Вход Регистрация
Файл: templates/default/js/fileuploader.js
Строк: 1492
<?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 and GNU LGPL 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>'+LANG_DROP_TO_UPLOAD+'</span></div>' +
                
'<div class="qq-upload-button">'+LANG_SELECT_UPLOAD+'</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="#">'+LANG_CANCEL+'</a>' +
                
'<span class="qq-upload-failed-text">'+LANG_ERROR+'</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'));
        
qq.remove(item);
        
//        if (result.success){
//            qq.addClass(item, this._classes.success);    
//        } else {
//            qq.addClass(item, this._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 && 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
Реклама