Вход Регистрация
Файл: protected/extensions/EAjaxUpload/assets/fileuploader.js
Строк: 1493
<?php
/**
 * http://github.com/valums/file-uploader
 *
 * Multiple file upload component with progress-bar, drag-and-drop.
 * © 2010 Andrew Valums ( andrew(at)valums.com )
 *
 * Licensed under GNU GPL 2 or later, see license.txt.
 */

//
// Helper functions
//

var qq qq || {};

/**
 * Adds all missing properties from second obj to first obj
 */
qq.extend = function(firstsecond){
    for (var 
prop in second){
        
first[prop] = second[prop];
    }
};

/**
 * Searches for a given element in the array, returns -1 if it is not present.
 * @param {Number} [from] The index at which to begin the search
 */
qq.indexOf = function(arreltfrom){
    if (
arr.indexOf) return arr.indexOf(eltfrom);

    
from from || 0;
    var 
len arr.length;

    if (
from 0from += len;

    for (; 
from lenfrom++){
        if (
from in arr && arr[from] === elt){
            return 
from;
        }
    }
    return -
1;
};

qq.getUniqueId = (function(){
    var 
id 0;
    return function(){ return 
id++; };
})();

//
// Events

qq.attach = function(elementtype, fn){
    if (
element.addEventListener){
        
element.addEventListener(type, fn, false);
    } else if (
element.attachEvent){
        
element.attachEvent('on' type, fn);
    }
};
qq.detach = function(elementtype, fn){
    if (
element.removeEventListener){
        
element.removeEventListener(type, fn, false);
    } else if (
element.attachEvent){
        
element.detachEvent('on' type, fn);
    }
};

qq.preventDefault = function(e){
    if (
e.preventDefault){
        
e.preventDefault();
    } else{
        
e.returnValue false;
    }
};

//
// Node manipulations

/**
 * Insert node a before node b.
 */
qq.insertBefore = function(ab){
    
b.parentNode.insertBefore(ab);
};
qq.remove = function(element){
    
element.parentNode.removeChild(element);
};

qq.contains = function(parentdescendant){
    
// compareposition returns false in this case
    
if (parent == descendant) return true;

    if (
parent.contains){
        return 
parent.contains(descendant);
    } else {
        return !!(
descendant.compareDocumentPosition(parent) & 8);
    }
};

/**
 * Creates and returns element from html string
 * Uses innerHTML to create an element
 */
qq.toElement = (function(){
    var 
div document.createElement('div');
    return function(
html){
        
div.innerHTML html;
        var 
element div.firstChild;
        
div.removeChild(element);
        return 
element;
    };
})();

//
// Node properties and attributes

/**
 * Sets styles for an element.
 * Fixes opacity in IE6-8.
 */
qq.css = function(elementstyles){
    if (
styles.opacity != null){
        if (
typeof element.style.opacity != 'string' && typeof(element.filters) != 'undefined'){
            
styles.filter 'alpha(opacity=' Math.round(100 styles.opacity) + ')';
        }
    }
    
qq.extend(element.stylestyles);
};
qq.hasClass = function(elementname){
    var 
re = new RegExp('(^| )' name '( |$)');
    return 
re.test(element.className);
};
qq.addClass = function(elementname){
    if (!
qq.hasClass(elementname)){
        
element.className += ' ' name;
    }
};
qq.removeClass = function(elementname){
    var 
re = new RegExp('(^| )' name '( |$)');
    
element.className element.className.replace(re' ').replace(/^s+|s+$/g"");
};
qq.setText = function(elementtext){
    
element.innerText text;
    
element.textContent text;
};

//
// Selecting elements

qq.children = function(element){
    var 
children = [],
    
child element.firstChild;

    while (
child){
        if (
child.nodeType == 1){
            
children.push(child);
        }
        
child child.nextSibling;
    }

    return 
children;
};

qq.getByClass = function(elementclassName){
    if (
element.querySelectorAll){
        return 
element.querySelectorAll('.' className);
    }

    var 
result = [];
    var 
candidates element.getElementsByTagName("*");
    var 
len candidates.length;

    for (var 
0leni++){
        if (
qq.hasClass(candidates[i], className)){
            
result.push(candidates[i]);
        }
    }
    return 
result;
};

/**
 * obj2url() takes a json-object as argument and generates
 * a querystring. pretty much like jQuery.param()
 *
 * how to use:
 *
 *    `qq.obj2url({a:'b',c:'d'},'http://any.url/upload?otherParam=value');`
 *
 * will result in:
 *
 *    `http://any.url/upload?otherParam=value&a=b&c=d`
 *
 * @param  Object JSON-Object
 * @param  String current querystring-part
 * @return String encoded querystring
 */
qq.obj2url = function(objtempprefixDone){
    var 
uristrings = [],
        
prefix '&',
        
add = function(nextObji){
            var 
nextTemp temp
                
? (/[]$/.test(temp)) // prevent double-encoding
                   
temp
                   
temp+'['+i+']'
                
i;
            if ((
nextTemp != 'undefined') && (!= 'undefined')) {
                
uristrings.push(
                    (
typeof nextObj === 'object')
                        ? 
qq.obj2url(nextObjnextTemptrue)
                        : (
Object.prototype.toString.call(nextObj) === '[object Function]')
                            ? 
encodeURIComponent(nextTemp) + '=' encodeURIComponent(nextObj())
                            : 
encodeURIComponent(nextTemp) + '=' encodeURIComponent(nextObj)
                );
            }
        };

    if (!
prefixDone && temp) {
      
prefix = (/?/.test(temp)) ? (/?$/.test(temp)) ? '' '&' '?';
      
uristrings.push(temp);
      
uristrings.push(qq.obj2url(obj));
    } else if ((
Object.prototype.toString.call(obj) === '[object Array]') && (typeof obj != 'undefined') ) {
        
// we wont use a for-in-loop on an array (performance)
        
for (var 0len obj.lengthlen; ++i){
            
add(obj[i], i);
        }
    } else if ((
typeof obj != 'undefined') && (obj !== null) && (typeof obj === "object")){
        
// for anything else but a scalar, we will use for-in-loop
        
for (var i in obj){
            
add(obj[i], i);
        }
    } else {
        
uristrings.push(encodeURIComponent(temp) + '=' encodeURIComponent(obj));
    }

    return 
uristrings.join(prefix)
                     .
replace(/^&/, '')
                     .
replace(/%20/g'+');
};

//
//
// Uploader Classes
//
//

var qq qq || {};

/**
 * Creates upload button, validates upload, but doesn't create file list or dd.
 */
qq.FileUploaderBasic = function(o){
    
this._options = {
        
// set to true to see the server response
        
debugfalse,
        
action'/server/upload',
        
params: {},
        
buttonnull,
        
multipletrue,
        
maxConnections3,
        
// validation
        
allowedExtensions: [],
        
sizeLimit0,
        
minSizeLimit0,
        
// events
        // return false to cancel submit
        
onSubmit: function(idfileName){},
        
onProgress: function(idfileNameloadedtotal){},
        
onComplete: function(idfileNameresponseJSON){},
        
onCancel: function(idfileName){},
        
// messages
        
messages: {
            
typeError"{file} has invalid extension. Only {extensions} are allowed.",
            
sizeError"{file} is too large, maximum file size is {sizeLimit}.",
            
minSizeError"{file} is too small, minimum file size is {minSizeLimit}.",
            
emptyError"{file} is empty, please select files again without it.",
            
onLeave"The files are being uploaded, if you leave now the upload will be cancelled."
        
},
        
showMessage: function(message){
            
alert(message);
        }
    };
    
qq.extend(this._optionso);

    
// number of files being uploaded
    
this._filesInProgress 0;
    
this._handler this._createUploadHandler();

    if (
this._options.button){
        
this._button this._createUploadButton(this._options.button);
    }

    
this._preventLeaveInProgress();
};

qq.FileUploaderBasic.prototype = {
    
setParams: function(params){
        
this._options.params params;
    },
    
getInProgress: function(){
        return 
this._filesInProgress;
    },
    
_createUploadButton: function(element){
        var 
self this;

        return new 
qq.UploadButton({
            
elementelement,
            
multiplethis._options.multiple && qq.UploadHandlerXhr.isSupported(),
            
onChange: function(input){
                
self._onInputChange(input);
            }
        });
    },
    
_createUploadHandler: function(){
        var 
self this,
            
handlerClass;

        if(
qq.UploadHandlerXhr.isSupported()){
            
handlerClass 'UploadHandlerXhr';
        } else {
            
handlerClass 'UploadHandlerForm';
        }

        var 
handler = new qq[handlerClass]({
            
debugthis._options.debug,
            
actionthis._options.action,
            
maxConnectionsthis._options.maxConnections,
            
onProgress: function(idfileNameloadedtotal){
                
self._onProgress(idfileNameloadedtotal);
                
self._options.onProgress(idfileNameloadedtotal);
            },
            
onComplete: function(idfileNameresult){
                
self._onComplete(idfileNameresult);
                
self._options.onComplete(idfileNameresult);
            },
            
onCancel: function(idfileName){
                
self._onCancel(idfileName);
                
self._options.onCancel(idfileName);
            }
        });

        return 
handler;
    },
    
_preventLeaveInProgress: function(){
        var 
self this;

        
qq.attach(window'beforeunload', function(e){
            if (!
self._filesInProgress){return;}

            var 
|| window.event;
            
// for ie, ff
            
e.returnValue self._options.messages.onLeave;
            
// for webkit
            
return self._options.messages.onLeave;
        });
    },
    
_onSubmit: function(idfileName){
        
this._filesInProgress++;
    },
    
_onProgress: function(idfileNameloadedtotal){
    },
    
_onComplete: function(idfileNameresult){
        
this._filesInProgress--;
        if (
result.error){
            
this._options.showMessage(result.error);
        }
    },
    
_onCancel: function(idfileName){
        
this._filesInProgress--;
    },
    
_onInputChange: function(input){
        if (
this._handler instanceof qq.UploadHandlerXhr){
            
this._uploadFileList(input.files);
        } else {
            if (
this._validateFile(input)){
                
this._uploadFile(input);
            }
        }
        
this._button.reset();
    },
    
_uploadFileList: function(files){
        for (var 
i=0i<files.lengthi++){
            if ( !
this._validateFile(files[i])){
                return;
            }
        }

        for (var 
i=0i<files.lengthi++){
            
this._uploadFile(files[i]);
        }
    },
    
_uploadFile: function(fileContainer){
        var 
id this._handler.add(fileContainer);
        var 
fileName this._handler.getName(id);

        if (
this._options.onSubmit(idfileName) !== false){
            
this._onSubmit(idfileName);
            
this._handler.upload(idthis._options.params);
        }
    },
    
_validateFile: function(file){
        var 
namesize;

        if (
file.value){
            
// it is a file input
            // get input value and remove path to normalize
            
name file.value.replace(/.*(/|\)/, "");
        } else {
            
// fix missing properties in Safari
            
name file.fileName != null file.fileName file.name;
            
size file.fileSize != null file.fileSize file.size;
        }

        if (! 
this._isAllowedExtension(name)){
            
this._error('typeError'name);
            return 
false;

        } else if (
size === 0){
            
this._error('emptyError'name);
            return 
false;

        } else if (
size && this._options.sizeLimit && size this._options.sizeLimit){
            
this._error('sizeError'name);
            return 
false;

        } else if (
size && size this._options.minSizeLimit){
            
this._error('minSizeError'name);
            return 
false;
        }

        return 
true;
    },
    
_error: function(codefileName){
        var 
message this._options.messages[code];
        function 
r(namereplacement){ message message.replace(namereplacement); }

        
r('{file}'this._formatFileName(fileName));
        
r('{extensions}'this._options.allowedExtensions.join(', '));
        
r('{sizeLimit}'this._formatSize(this._options.sizeLimit));
        
r('{minSizeLimit}'this._formatSize(this._options.minSizeLimit));

        
this._options.showMessage(message);
    },
    
_formatFileName: function(name){
        if (
name.length 33){
            
name name.slice(019) + '...' name.slice(-13);
        }
        return 
name;
    },
    
_isAllowedExtension: function(fileName){
        var 
ext = (-!== fileName.indexOf('.')) ? fileName.replace(/.*[.]/, '').toLowerCase() : '';
        var 
allowed this._options.allowedExtensions;

        if (!
allowed.length){return true;}

        for (var 
i=0i<allowed.lengthi++){
            if (
allowed[i].toLowerCase() == ext){ return true;}
        }

        return 
false;
    },
    
_formatSize: function(bytes){
        var 
= -1;
        do {
            
bytes bytes 1024;
            
i++;
        } while (
bytes 99);

        return 
Math.max(bytes0.1).toFixed(1) + ['kB''MB''GB''TB''PB''EB'][i];
    }
};


/**
 * Class that creates upload widget with drag-and-drop and file list
 * @inherits qq.FileUploaderBasic
 */
qq.FileUploader = function(o){
    
// call parent constructor
    
qq.FileUploaderBasic.apply(thisarguments);

    
// additional options
    
qq.extend(this._options, {
        
elementnull,
        
// if set, will be used instead of qq-upload-list in template
        
listElementnull,

        
template'<div class="qq-uploader">' +
               
/* '<div class="qq-upload-drop-area"><span>Drop files here to upload</span></div>' +*/
                
'<div class="qq-upload-button" style="line-height: 25px;">Новый файл</div>' +
                
'<ul class="qq-upload-list"></ul>' +
             
'</div>',

        
// template for one item in file list
        
fileTemplate'<li>' +
                
'<span class="qq-upload-file"></span>' +
                
'<span class="qq-upload-spinner"></span>' +
                
'<span class="qq-upload-size"></span>' +
                
'<a class="qq-upload-cancel" href="#">Cancel</a>' +
                
'<span class="qq-upload-failed-text">Failed</span>' +
            
'</li>',

        
classes: {
            
// used to get elements from templates
            
button'qq-upload-button',
            
drop'qq-upload-drop-area',
            
dropActive'qq-upload-drop-area-active',
            list: 
'qq-upload-list',

            
file'qq-upload-file',
            
spinner'qq-upload-spinner',
            
size'qq-upload-size',
            
cancel'qq-upload-cancel',

            
// added to list item when upload completes
            // used in css to hide progress spinner
            
success'qq-upload-success',
            
fail'qq-upload-fail'
        
}
    });
    
// overwrite options with user supplied
    
qq.extend(this._optionso);

    
this._element this._options.element;
    
this._element.innerHTML this._options.template;
    
this._listElement this._options.listElement || this._find(this._element'list');

    
this._classes this._options.classes;

    
this._button this._createUploadButton(this._find(this._element'button'));

    
this._bindCancelEvent();
    
this._setupDragDrop();
};

// inherit from Basic Uploader
qq.extend(qq.FileUploader.prototypeqq.FileUploaderBasic.prototype);

qq.extend(qq.FileUploader.prototype, {
    
/**
     * Gets one of the elements listed in this._options.classes
     **/
    
_find: function(parenttype){
        var 
element qq.getByClass(parentthis._options.classes[type])[0];
        if (!
element){
            throw new 
Error('element not found ' type);
        }

        return 
element;
    },
    
_setupDragDrop: function(){
        var 
self this,
            
dropArea this._find(this._element'drop');

        var 
dz = new qq.UploadDropZone({
            
elementdropArea,
            
onEnter: function(e){
                
qq.addClass(dropAreaself._classes.dropActive);
                
e.stopPropagation();
            },
            
onLeave: function(e){
                
e.stopPropagation();
            },
            
onLeaveNotDescendants: function(e){
                
qq.removeClass(dropAreaself._classes.dropActive);
            },
            
onDrop: function(e){
                
dropArea.style.display 'none';
                
qq.removeClass(dropAreaself._classes.dropActive);
                
self._uploadFileList(e.dataTransfer.files);
            }
        });

        
dropArea.style.display 'none';

        
qq.attach(document'dragenter', function(e){
            if (!
dz._isValidFileDrag(e)) return;

            
dropArea.style.display 'block';
        });
        
qq.attach(document'dragleave', function(e){
            if (!
dz._isValidFileDrag(e)) return;

            var 
relatedTarget document.elementFromPoint(e.clientXe.clientY);
            
// only fire when leaving document out
            
if ( ! relatedTarget || relatedTarget.nodeName == "HTML"){
                
dropArea.style.display 'none';
            }
        });
    },
    
_onSubmit: function(idfileName){
        
qq.FileUploaderBasic.prototype._onSubmit.apply(thisarguments);
        
this._addToList(idfileName);
    },
    
_onProgress: function(idfileNameloadedtotal){
        
qq.FileUploaderBasic.prototype._onProgress.apply(thisarguments);

        var 
item this._getItemByFileId(id);
        var 
size this._find(item'size');
        
size.style.display 'inline';

        var 
text;
        if (
loaded != total){
            
text Math.round(loaded total 100) + '% from ' this._formatSize(total);
        } else {
            
text this._formatSize(total);
        }

        
qq.setText(sizetext);
    },
    
_onComplete: function(idfileNameresult){
        
qq.FileUploaderBasic.prototype._onComplete.apply(thisarguments);

        
// mark completed
        
var item this._getItemByFileId(id);
        
qq.remove(this._find(item'cancel'));
        
qq.remove(this._find(item'spinner'));

        if (
result.success){
            
qq.addClass(itemthis._classes.success);
        } else {
            
qq.addClass(itemthis._classes.fail);
        }
    },
    
_addToList: function(idfileName){
        var 
item qq.toElement(this._options.fileTemplate);
        
item.qqFileId id;

        var 
fileElement this._find(item'file');
        
qq.setText(fileElementthis._formatFileName(fileName));
        
this._find(item'size').style.display 'none';

        
this._listElement.appendChild(item);
    },
    
_getItemByFileId: function(id){
        var 
item this._listElement.firstChild;

        
// there can't be txt nodes in dynamically created list
        // and we can  use nextSibling
        
while (item){
            if (
item.qqFileId == id) return item;
            
item item.nextSibling;
        }
    },
    
/**
     * delegate click event for cancel link
     **/
    
_bindCancelEvent: function(){
        var 
self this,
            list = 
this._listElement;

        
qq.attach(list, 'click', function(e){
            
|| window.event;
            var 
target e.target || e.srcElement;

            if (
qq.hasClass(targetself._classes.cancel)){
                
qq.preventDefault(e);

                var 
item target.parentNode;
                
self._handler.cancel(item.qqFileId);
                
qq.remove(item);
            }
        });
    }
});

qq.UploadDropZone = function(o){
    
this._options = {
        
elementnull,
        
onEnter: function(e){},
        
onLeave: function(e){},
        
// is not fired when leaving element by hovering descendants
        
onLeaveNotDescendants: function(e){},
        
onDrop: function(e){}
    };
    
qq.extend(this._optionso);

    
this._element this._options.element;

    
this._disableDropOutside();
    
this._attachEvents();
};

qq.UploadDropZone.prototype = {
    
_disableDropOutside: function(e){
        
// run only once for all instances
        
if (!qq.UploadDropZone.dropOutsideDisabled ){

            
qq.attach(document'dragover', function(e){
                if (
e.dataTransfer){
                    
e.dataTransfer.dropEffect 'none';
                    
e.preventDefault();
                }
            });

            
qq.UploadDropZone.dropOutsideDisabled true;
        }
    },
    
_attachEvents: function(){
        var 
self this;

        
qq.attach(self._element'dragover', function(e){
            if (!
self._isValidFileDrag(e)) return;

            var 
effect e.dataTransfer.effectAllowed;
            if (
effect == 'move' || effect == 'linkMove'){
                
e.dataTransfer.dropEffect 'move'// for FF (only move allowed)
            
} else {
                
e.dataTransfer.dropEffect 'copy'// for Chrome
            
}

            
e.stopPropagation();
            
e.preventDefault();
        });

        
qq.attach(self._element'dragenter', function(e){
            if (!
self._isValidFileDrag(e)) return;

            
self._options.onEnter(e);
        });

        
qq.attach(self._element'dragleave', function(e){
            if (!
self._isValidFileDrag(e)) return;

            
self._options.onLeave(e);

            var 
relatedTarget document.elementFromPoint(e.clientXe.clientY);
            
// do not fire when moving a mouse over a descendant
            
if (qq.contains(thisrelatedTarget)) return;

            
self._options.onLeaveNotDescendants(e);
        });

        
qq.attach(self._element'drop', function(e){
            if (!
self._isValidFileDrag(e)) return;

            
e.preventDefault();
            
self._options.onDrop(e);
        });
    },
    
_isValidFileDrag: function(e){
        var 
dt e.dataTransfer,
            
// do not check dt.types.contains in webkit, because it crashes safari 4
            
isWebkit navigator.userAgent.indexOf("AppleWebKit") > -1;

        
// dt.effectAllowed is none in Safari 5
        // dt.types.contains check is for firefox
        
return dt && dt.effectAllowed != 'none' &&
            (
dt.files || (!isWebkit && dt.types.contains && dt.types.contains('Files')));

    }
};

qq.UploadButton = function(o){
    
this._options = {
        
elementnull,
        
// if set to true adds multiple attribute to file input
        
multiplefalse,
        
// name attribute of file input
        
name'file',
        
onChange: function(input){},
        
hoverClass'qq-upload-button-hover',
        
focusClass'qq-upload-button-focus'
    
};

    
qq.extend(this._optionso);

    
this._element this._options.element;

    
// make button suitable container for input
    
qq.css(this._element, {
        
position'relative',
        
overflow'hidden',
        
// Make sure browse button is in the right side
        // in Internet Explorer
        
direction'ltr'
    
});

    
this._input this._createInput();
};

qq.UploadButton.prototype = {
    
/* returns file input element */
    
getInput: function(){
        return 
this._input;
    },
    
/* cleans/recreates the file input */
    
reset: function(){
        if (
this._input.parentNode){
            
qq.remove(this._input);
        }

        
qq.removeClass(this._elementthis._options.focusClass);
        
this._input this._createInput();
    },
    
_createInput: function(){
        var 
input document.createElement("input");

        if (
this._options.multiple){
            
input.setAttribute("multiple""multiple");
        }

        
input.setAttribute("type""file");
        
input.setAttribute("name"this._options.name);

        
qq.css(input, {
            
position'absolute',
            
// in Opera only 'browse' button
            // is clickable and it is located at
            // the right side of the input
            
right0,
            
top0,
            
fontFamily'Arial',
            
// 4 persons reported this, the max values that worked for them were 243, 236, 236, 118
            
fontSize'118px',
            
margin0,
            
padding0,
            
cursor'pointer',
            
opacity0
        
});

        
this._element.appendChild(input);

        var 
self this;
        
qq.attach(input'change', function(){
            
self._options.onChange(input);
        });

        
qq.attach(input'mouseover', function(){
            
qq.addClass(self._elementself._options.hoverClass);
        });
        
qq.attach(input'mouseout', function(){
            
qq.removeClass(self._elementself._options.hoverClass);
        });
        
qq.attach(input'focus', function(){
            
qq.addClass(self._elementself._options.focusClass);
        });
        
qq.attach(input'blur', function(){
            
qq.removeClass(self._elementself._options.focusClass);
        });

        
// IE and Opera, unfortunately have 2 tab stops on file input
        // which is unacceptable in our case, disable keyboard access
        
if (window.attachEvent){
            
// it is IE or Opera
            
input.setAttribute('tabIndex'"-1");
        }

        return 
input;
    }
};

/**
 * Class for uploading files, uploading itself is handled by child classes
 */
qq.UploadHandlerAbstract = function(o){
    
this._options = {
        
debugfalse,
        
action'/upload.php',
        
// maximum number of concurrent uploads
        
maxConnections999,
        
onProgress: function(idfileNameloadedtotal){},
        
onComplete: function(idfileNameresponse){},
        
onCancel: function(idfileName){}
    };
    
qq.extend(this._optionso);

    
this._queue = [];
    
// params for files in queue
    
this._params = [];
};
qq.UploadHandlerAbstract.prototype = {
    
log: function(str){
        if (
this._options.debug && window.consoleconsole.log('[uploader] ' str);
    },
    
/**
     * Adds file or file input to the queue
     * @returns id
     **/
    
add: function(file){},
    
/**
     * Sends the file identified by id and additional query params to the server
     */
    
upload: function(idparams){
        var 
len this._queue.push(id);

        var 
copy = {};
        
qq.extend(copyparams);
        
this._params[id] = copy;

        
// if too many active uploads, wait...
        
if (len <= this._options.maxConnections){
            
this._upload(idthis._params[id]);
        }
    },
    
/**
     * Cancels file upload by id
     */
    
cancel: function(id){
        
this._cancel(id);
        
this._dequeue(id);
    },
    
/**
     * Cancells all uploads
     */
    
cancelAll: function(){
        for (var 
i=0i<this._queue.lengthi++){
            
this._cancel(this._queue[i]);
        }
        
this._queue = [];
    },
    
/**
     * Returns name of the file identified by id
     */
    
getName: function(id){},
    
/**
     * Returns size of the file identified by id
     */
    
getSize: function(id){},
    
/**
     * Returns id of files being uploaded or
     * waiting for their turn
     */
    
getQueue: function(){
        return 
this._queue;
    },
    
/**
     * Actual upload method
     */
    
_upload: function(id){},
    
/**
     * Actual cancel method
     */
    
_cancel: function(id){},
    
/**
     * Removes element from queue, starts upload of next
     */
    
_dequeue: function(id){
        var 
qq.indexOf(this._queueid);
        
this._queue.splice(i1);

        var 
max this._options.maxConnections;

        if (
this._queue.length >= max){
            var 
nextId this._queue[max-1];
            
this._upload(nextIdthis._params[nextId]);
        }
    }
};

/**
 * Class for uploading files using form and iframe
 * @inherits qq.UploadHandlerAbstract
 */
qq.UploadHandlerForm = function(o){
    
qq.UploadHandlerAbstract.apply(thisarguments);

    
this._inputs = {};
};
// @inherits qq.UploadHandlerAbstract
qq.extend(qq.UploadHandlerForm.prototypeqq.UploadHandlerAbstract.prototype);

qq.extend(qq.UploadHandlerForm.prototype, {
    
add: function(fileInput){
        
fileInput.setAttribute('name''qqfile');
        var 
id 'qq-upload-handler-iframe' qq.getUniqueId();

        
this._inputs[id] = fileInput;

        
// remove file input from DOM
        
if (fileInput.parentNode){
            
qq.remove(fileInput);
        }

        return 
id;
    },
    
getName: function(id){
        
// get input value and remove path to normalize
        
return this._inputs[id].value.replace(/.*(/|\)/, "");
    },
    
_cancel: function(id){
        
this._options.onCancel(idthis.getName(id));

        
delete this._inputs[id];

        var 
iframe document.getElementById(id);
        if (
iframe){
            
// to cancel request set src to something else
            // we use src="javascript:false;" because it doesn't
            // trigger ie6 prompt on https
            
iframe.setAttribute('src''javascript:false;');

            
qq.remove(iframe);
        }
    },
    
_upload: function(idparams){
        var 
input this._inputs[id];

        if (!
input){
            throw new 
Error('file with passed id was not added, or already uploaded or cancelled');
        }

        var 
fileName this.getName(id);

        var 
iframe this._createIframe(id);
        var 
form this._createForm(iframeparams);
        
form.appendChild(input);

        var 
self this;
        
this._attachLoadEvent(iframe, function(){
            
self.log('iframe loaded');

            var 
response self._getIframeContentJSON(iframe);

            
self._options.onComplete(idfileNameresponse);
            
self._dequeue(id);

            
delete self._inputs[id];
            
// timeout added to fix busy state in FF3.6
            
setTimeout(function(){
                
qq.remove(iframe);
            }, 
1);
        });

        
form.submit();
        
qq.remove(form);

        return 
id;
    },
    
_attachLoadEvent: function(iframecallback){
        
qq.attach(iframe'load', function(){
            
// when we remove iframe from dom
            // the request stops, but in IE load
            // event fires
            
if (!iframe.parentNode){
                return;
            }

            
// fixing Opera 10.53
            
if (iframe.contentDocument &&
                
iframe.contentDocument.body &&
                
iframe.contentDocument.body.innerHTML == "false"){
                
// In Opera event is fired second time
                // when body.innerHTML changed from false
                // to server response approx. after 1 sec
                // when we upload file with iframe
                
return;
            }

            
callback();
        });
    },
    
/**
     * Returns json object received by iframe from server.
     */
    
_getIframeContentJSON: function(iframe){
        
// iframe.contentWindow.document - for IE<7
        
var doc iframe.contentDocument iframe.contentDocumentiframe.contentWindow.document,
            
response;

        
this.log("converting iframe's innerHTML to JSON");
        
this.log("innerHTML = " doc.body.innerHTML);

        try {
            
response = eval("(" doc.body.innerHTML ")");
        } catch(
err){
            
response = {};
        }

        return 
response;
    },
    
/**
     * Creates iframe with unique name
     */
    
_createIframe: function(id){
        
// We can't use following code as the name attribute
        // won't be properly registered in IE6, and new window
        // on form submit will open
        // var iframe = document.createElement('iframe');
        // iframe.setAttribute('name', id);

        
var iframe qq.toElement('<iframe src="javascript:false;" name="' id '" />');
        
// src="javascript:false;" removes ie6 prompt on https

        
iframe.setAttribute('id'id);

        
iframe.style.display 'none';
        
document.body.appendChild(iframe);

        return 
iframe;
    },
    
/**
     * Creates form, that will be submitted to iframe
     */
    
_createForm: function(iframeparams){
        
// We can't use the following code in IE6
        // var form = document.createElement('form');
        // form.setAttribute('method', 'post');
        // form.setAttribute('enctype', 'multipart/form-data');
        // Because in this case file won't be attached to request
        
var form qq.toElement('<form method="post" enctype="multipart/form-data"></form>');

        var 
queryString qq.obj2url(paramsthis._options.action);

        
form.setAttribute('action'queryString);
        
form.setAttribute('target'iframe.name);
        
form.style.display 'none';
        
document.body.appendChild(form);

        return 
form;
    }
});

/**
 * Class for uploading files using xhr
 * @inherits qq.UploadHandlerAbstract
 */
qq.UploadHandlerXhr = function(o){
    
qq.UploadHandlerAbstract.apply(thisarguments);

    
this._files = [];
    
this._xhrs = [];

    
// current loaded size in bytes for each file
    
this._loaded = [];
};

// static method
qq.UploadHandlerXhr.isSupported = function(){
    var 
input document.createElement('input');
    
input.type 'file';

    return (
        
'multiple' in input &&
        
typeof File != "undefined" &&
        
typeof (new XMLHttpRequest()).upload != "undefined" );
};

// @inherits qq.UploadHandlerAbstract
qq.extend(qq.UploadHandlerXhr.prototypeqq.UploadHandlerAbstract.prototype)

qq.extend(qq.UploadHandlerXhr.prototype, {
    
/**
     * Adds file to the queue
     * Returns id to use with upload, cancel
     **/
    
add: function(file){
        if (!(
file instanceof File)){
            throw new 
Error('Passed obj in not a File (in qq.UploadHandlerXhr)');
        }

        return 
this._files.push(file) - 1;
    },
    
getName: function(id){
        var 
file this._files[id];
        
// fix missing name in Safari 4
        
return file.fileName != null file.fileName file.name;
    },
    
getSize: function(id){
        var 
file this._files[id];
        return 
file.fileSize != null file.fileSize file.size;
    },
    
/**
     * Returns uploaded bytes for file identified by id
     */
    
getLoaded: function(id){
        return 
this._loaded[id] || 0;
    },
    
/**
     * Sends the file identified by id and additional query params to the server
     * @param {Object} params name-value string pairs
     */
    
_upload: function(idparams){
        var 
file this._files[id],
            
name this.getName(id),
            
size this.getSize(id);

        
this._loaded[id] = 0;

        var 
xhr this._xhrs[id] = new XMLHttpRequest();
        var 
self this;

        
xhr.upload.onprogress = function(e){
            if (
e.lengthComputable){
                
self._loaded[id] = e.loaded;
                
self._options.onProgress(idnamee.loadede.total);
            }
        };

        
xhr.onreadystatechange = function(){
            if (
xhr.readyState == 4){
                
self._onComplete(idxhr);
            }
        };

        
// build query string
        
params params || {};
        
params['qqfile'] = name;
        var 
queryString qq.obj2url(paramsthis._options.action);

        
xhr.open("POST"queryStringtrue);
        
xhr.setRequestHeader("X-Requested-With""XMLHttpRequest");
        
xhr.setRequestHeader("X-File-Name"encodeURIComponent(name));
        
xhr.setRequestHeader("Content-Type""application/octet-stream");
        
xhr.send(file);
    },
    
_onComplete: function(idxhr){
        
// the request was aborted/cancelled
        
if (!this._files[id]) return;

        var 
name this.getName(id);
        var 
size this.getSize(id);

        
this._options.onProgress(idnamesizesize);

        if (
xhr.status == 200){
            
this.log("xhr - server response received");
            
this.log("responseText = " xhr.responseText);

            var 
response;

            try {
                
response = eval("(" xhr.responseText ")");
            } catch(
err){
                
response = {};
            }

            
this._options.onComplete(idnameresponse);

        } else {
            
this._options.onComplete(idname, {});
        }

        
this._files[id] = null;
        
this._xhrs[id] = null;
        
this._dequeue(id);
    },
    
_cancel: function(id){
        
this._options.onCancel(idthis.getName(id));

        
this._files[id] = null;

        if (
this._xhrs[id]){
            
this._xhrs[id].abort();
            
this._xhrs[id] = null;
        }
    }
});
?>
Онлайн: 0
Реклама