{"version":3,"sources":["?","../client/js/util.js","../client/js/export.js","../client/js/error/error.js","../client/js/version.js","../client/js/features.js","../client/js/promise.js","../client/js/blob-proxy.js","../client/js/button.js","../client/js/upload-data.js","../client/js/uploader.basic.api.js","../client/js/uploader.basic.js","../client/js/ajax.requester.js","../client/js/upload-handler/upload.handler.js","../client/js/upload-handler/upload.handler.controller.js","../client/js/window.receive.message.js","../client/js/upload-handler/form.upload.handler.js","../client/js/upload-handler/xhr.upload.handler.js","../client/js/deletefile.ajax.requester.js","../client/js/image-support/megapix-image.js","../client/js/image-support/image.js","../client/js/image-support/exif.js","../client/js/identify.js","../client/js/image-support/validation.image.js","../client/js/session.js","../client/js/session.ajax.requester.js","../client/js/image-support/scaler.js","../client/js/third-party/ExifRestorer.js","../client/js/total-progress.js","../client/js/paste.js","../client/js/form-support.js","../client/js/traditional/traditional.form.upload.handler.js","../client/js/traditional/traditional.xhr.upload.handler.js","../client/js/traditional/all-chunks-done.ajax.requester.js"],"names":["global","qq","element","hide","style","display","this","attach","type","fn","addEventListener","attachEvent","detach","removeEventListener","detachEvent","contains","descendant","compareDocumentPosition","insertBefore","elementB","parentNode","remove","removeChild","css","styles","Error","opacity","filter","Math","round","extend","hasClass","name","considerParent","re","RegExp","test","className","addClass","removeClass","replace","getByClass","first","candidates","result","querySelector","querySelectorAll","getElementsByTagName","each","idx","val","push","getFirstByClass","children","child","firstChild","nodeType","nextSibling","setText","text","innerText","textContent","clearText","hasAttribute","attrName","attrVal","exec","getAttribute","undefined","canvasToBlob","canvas","mime","quality","dataUriToBlob","toDataURL","dataUri","arrayBuffer","byteString","createBlob","data","BlobBuilder","window","WebKitBlobBuilder","MozBlobBuilder","MSBlobBuilder","blobBuilder","append","getBlob","Blob","intArray","mimeString","split","indexOf","atob","decodeURI","ArrayBuffer","length","Uint8Array","character","charCodeAt","log","message","level","console","isObject","variable","Object","prototype","toString","call","isFunction","isArray","value","buffer","constructor","isItemList","maybeItemList","isNodeList","maybeNodeList","item","namedItem","isString","maybeString","trimStr","string","String","trim","format","str","args","Array","slice","arguments","newStr","nextIdxToReplace","strBefore","substring","strAfter","isFile","maybeFile","File","isFileList","maybeFileList","FileList","isFileOrInput","maybeFileOrInput","isInput","maybeInput","notFile","evaluateType","normalizedType","toLowerCase","HTMLInputElement","tagName","isBlob","maybeBlob","isXhrUploadSupported","input","document","createElement","multiple","FormData","createXhrInstance","upload","XMLHttpRequest","ActiveXObject","error","isFolderDropSupported","dataTransfer","items","webkitGetAsEntry","isFileChunkingSupported","androidStock","webkitSlice","mozSlice","sliceBlob","fileOrBlob","start","end","slicer","arrayBufferToHex","bytesAsHex","bytes","byt","byteAsHexStr","readBlobToHex","blob","startOffset","initialBlob","fileReader","FileReader","promise","Promise","onload","success","onerror","failure","readAsArrayBuffer","second","extendNested","prop","override","target","sourceFn","super_","source","srcPropName","srcPropVal","arr","elt","from","len","hasOwnProperty","getUniqueId","c","r","random","v","ie","navigator","userAgent","ie7","ie8","ie10","ie11","edge","safari","vendor","chrome","opera","firefox","windows","platform","android","ios6","ios","ios7","ios8","ios800","iosChrome","iosSafari","iosSafariWebView","preventDefault","e","returnValue","toElement","div","html","innerHTML","iterableItem","callback","keyOrIndex","retVal","Storage","key","getItem","charAt","bind","oldFunc","context","newArgs","concat","apply","obj2url","obj","temp","prefixDone","uristrings","prefix","add","nextObj","i","nextTemp","encodeURIComponent","join","obj2FormData","formData","arrayKeyName","obj2Inputs","form","setAttribute","appendChild","parseJson","json","JSON","parse","eval","getExtension","filename","extIdx","lastIndexOf","substr","getFilename","blobOrFileInput","fileName","DisposeSupport","disposers","dispose","disposer","shift","addDisposer","disposeFunction","define","amd","module","exports","version","supportedFeatures","supportsUploading","supportsUploadingBlobs","supportsFileDrop","supportsAjaxFileUploading","supportsFolderDrop","supportsChunking","supportsResume","supportsUploadViaPaste","supportsUploadCors","supportsDeleteFileXdr","supportsDeleteFileCorsXhr","supportsDeleteFileCors","supportsFolderSelection","supportsImagePreviews","supportsUploadProgress","testSupportsFileInputElement","supported","tempInput","disabled","ex","isChrome14OrHigher","match","isCrossOriginXhrSupported","xhr","withCredentials","isXdrSupported","XDomainRequest","isCrossOriginAjaxSupported","isFolderSelectionSupported","webkitdirectory","isLocalStorageSupported","localStorage","setItem","isDragAndDropSupported","span","postMessage","ajaxUploading","blobUploading","canDetermineSize","chunking","deleteFileCors","deleteFileCorsXdr","deleteFileCorsXhr","dialogElement","HTMLDialogElement","fileDrop","folderDrop","folderSelection","imagePreviews","imageValidation","itemSizeValidation","pause","progressBar","resume","scaling","tiffPreviews","unlimitedScaledImageSize","uploading","uploadCors","uploadCustomHeaders","uploadNonMultipart","uploadViaPaste","isGenericPromise","maybePromise","then","successArgs","failureArgs","successCallbacks","failureCallbacks","doneCallbacks","state","onSuccess","onFailure","done","BlobProxy","referenceBlob","onCreate","create","UploadButton","o","self","disposeSupport","options","acceptFiles","focusClass","folders","hoverClass","ios8BrowserCrashWorkaround","onChange","title","buttonId","createInput","BUTTON_ID_ATTR_NAME","setMultiple","position","right","top","fontFamily","fontSize","margin","padding","cursor","height","overflow","direction","getInput","getButtonId","isMultiple","optInput","removeAttribute","setAcceptFiles","reset","UploadData","uploaderProxy","byUuid","byStatus","byProxyGroupId","byBatchId","getDataByIds","idOrIds","entries","id","getDataByUuids","uuids","uuid","getDataByStatus","status","statusResults","statuses","index","statusEnum","statusResultIndexes","dataIndex","addFile","spec","SUBMITTING","originalName","size","file","batchId","proxyGroupId","onBeforeStatusChange","onStatusChange","retrieve","optionalFilter","removeFileRef","record","setStatus","newStatus","oldStatus","byStatusOldStatusIndex","splice","uuidChanged","newUuid","oldUuid","updateName","newName","updateSize","newSize","setParentId","targetId","parentId","getIdsInProxyGroup","getIdsInBatch","SUBMITTED","REJECTED","QUEUED","CANCELED","PAUSED","UPLOADING","UPLOAD_FINALIZING","UPLOAD_RETRYING","UPLOAD_SUCCESSFUL","UPLOAD_FAILED","DELETE_FAILED","DELETING","DELETED","basePublicApi","addBlobs","blobDataOrArray","params","endpoint","addFiles","addInitialFiles","cannedFileList","cannedFile","_addCannedFile","_maybeHandleIos8SafariWorkaround","_storedIds","_currentBatchId","processBlob","_handleNewFile","_options","blobs","defaultName","verifiedFiles","processBlobData","blobData","processCanvas","processCanvasData","canvasData","normalizedQuality","processFileOrInput","fileOrInput","files","normalizeData","fileContainer","_prepareItemsForUpload","cancel","uploadData","_uploadData","getName","_handler","cancelAll","storedIdsCopy","storedFileId","clearStoredFiles","continueUpload","enabled","_uploadFile","deleteFile","_onSubmitDelete","doesExist","fileOrBlobId","isValid","drawThumbnail","fileId","imgOrCanvas","maxSize","fromServer","customResizeFunction","promiseToReturn","fileOrUrl","_imageGenerator","_thumbnailUrls","scale","getFile","container","generate","modifiedContainer","reason","getButton","_getButton","_buttonIdsForFileIds","getEndpoint","_endpointStore","get","uploadDataRecord","getInProgress","getParentId","uploadDataEntry","getUploads","getResumableFilesData","getSize","getNetUploads","_netUploaded","getRemainingAllowedItems","allowedItems","_currentItemLimit","_netUploadedOrQueued","getUuid","isResumable","hasResumeRecord","debug","pauseUpload","expunge","_autoRetries","_retryTimeouts","_preventRetries","_buttons","button","_paramsStore","_pasteHandler","session","refreshOnReset","_refreshSessionData","_succeededSinceLastAllComplete","_failedSinceLastAllComplete","_totalProgress","_customResumeDataStore","retry","_manualRetry","scaleImage","specs","Scaler","setCustomHeaders","headers","_customHeadersStore","set","setCustomResumeData","setDeleteFileCustomHeaders","_deleteFileCustomHeadersStore","setDeleteFileEndpoint","_deleteFileEndpointStore","setDeleteFileParams","_deleteFileParamsStore","setEndpoint","setForm","elementOrId","_updateFormSupportAndParams","setItemLimit","newItemLimit","setName","setParams","setUuid","fileRecord","_onDeleteComplete","errorMessage","uploadStoredFiles","_itemError","_uploadStoredFiles","basePrivateApi","sessionData","deleteFileEndpoint","deleteFileParams","thumbnailUrl","_annotateWithButtonId","associatedInput","qqButtonId","_getButtonId","_batchError","callbacks","onError","_createDeleteHandler","DeleteFileAjaxRequester","method","toUpperCase","maxConnections","uuidParamName","request","uuidName","customHeaders","paramsStore","endpointStore","cors","onDelete","_onDelete","onDeleteComplete","xhrOrXdr","isError","_createPasteHandler","PasteSupport","targetElement","paste","pasteReceived","_handleCheckedCallback","onPasteReceived","_handlePasteSuccess","identifier","_createStore","initialValue","_readOnlyValues_","store","catchall","perIdReadOnlyValues","readOnlyValues","copy","orig","getReadOnlyValues","includeReadOnlyValues","existing","values","addReadOnly","_createUploadDataTracker","_onUploadStatusChange","_maybeAllComplete","setTimeout","_createUploadButton","accept","validation","allowedExtensions","allowMultiple","workarounds","iosEmptyVideos","_isAllowedExtension","classes","buttonFocus","buttonHover","ios8BrowserCrash","inputName","_onInputChange","fileInputTitle","_disposeSupport","_createUploadHandler","additionalOptions","namespace","lastOnProgress","preventRetryParam","preventRetryResponseProperty","onProgress","loaded","total","_onProgress","onComplete","_onComplete","onCancel","cancelFinalizationEffort","_onCancel","onUploadPrep","_onUploadPrep","onUpload","_onUpload","onUploadResult","onUploadChunk","chunkData","_onUploadChunk","onUploadChunkResult","partIndex","onUploadChunkSuccess","_onUploadChunkSuccess","onResume","customResumeData","onAutoRetry","responseJSON","_onAutoRetry","onUuidChanged","setSize","_setSize","getDataByUuid","isQueued","isInProgress","getCustomResumeData","_getCustomResumeData","UploadHandlerController","_fileOrBlobRejected","_formatSize","sizeSymbols","max","toFixed","_generateExtraButtonSpecs","_extraButtonSpecs","extraButtons","extraButtonOptionEntry","extraButtonSpec","_initExtraButton","extraButtonsSpec","_defaultButtonId","buttonOrFileInputOrFile","inputs","fileInput","fileBlobOrInput","_getNotFinished","_getValidationBase","_getValidationDescriptor","fileWrapper","_getValidationDescriptors","fileWrappers","fileDescriptors","_handleCameraAccess","camera","acceptIosCamera","optionRoot","details","callbackRetVal","successParam","newFileWrapperList","actualFile","handler","_customNewFileHandler","_handleNewFileGeneric","addFileToHandler","_trackButton","fileList","extSuppliedName","extension","_handleDeleteSuccess","_handleDeleteFailed","_initFormSupportAndParams","_formSupport","FormSupport","attachedToForm","getFormInputsAsObject","autoUpload","newAutoUpload","newEndpoint","_isDeletePossible","expected","allowXdr","allowed","valid","allowedExt","extRegex","code","maybeNameOrNames","messages","names","validationBase","extensionsForMessage","placeholderMatch","replacement","allowedExtension","formatFileName","sizeLimit","minSizeLimit","placeholder","_onBeforeManualRetry","notFinished","_onAllComplete","ios8SafariUploads","alert","unsupportedBrowserIos8Safari","_maybeParseAndSendUploadError","response","errorReason","defaultResponseError","_maybeProcessNextItemAfterOnValidateCallback","validItem","stopOnFirstInvalidFile","validationDescriptor","onValidate","_onValidateCallbackSuccess","_onValidateCallbackFailure","successful","failed","onAllComplete","_shouldAutoRetry","retryWaitPeriod","autoAttemptDelay","_onBeforeAutoRetry","itemLimit","onManualRetry","clearTimeout","storedItemIndex","fileIndex","onIndividualProgress","_onSubmit","_onSubmitCallbackSuccess","_onSubmitted","onSubmitted","_storeForLater","onSuccessCallback","additionalMandatedParams","adjustedOnSuccessCallback","onSubmitDelete","_deleteHandler","sendDelete","_onTotalProgress","onTotalProgress","enableAuto","_onValidateBatchCallbackFailure","_onValidateBatchCallbackSuccess","validationDescriptors","proposedNetFilesUploadedOrQueued","tooManyItemsError","nextIndex","_validateFileOrBlobData","_upload","onValidateBatch","_preventLeaveInProgress","event","onLeave","Session","_session","addFileRecord","refresh","_sessionRequestComplete","onSessionRequestComplete","onNewSize","maxAutoAttempts","formElementOrId","onSubmit","idToUpload","stillSubmitting","validityChecker","allowEmpty","ImageValidation","validate","image","errorCode","_wrapCallbacks","safeCallback","errorMsg","exception","callbackName","callbackFunc","FineUploaderBasic","disableCancelForFormUploads","warnBeforeUnload","filenameParam","forceMultipart","omitDefaultParams","paramsInBody","requireSuccessJson","totalFileSizeName","maxHeight","maxWidth","minHeight","minWidth","maybeXhr","maybeXhrOrXdr","attemptNumber","fileOrBlobData","typeError","sizeError","minSizeError","emptyError","noFilesError","maxHeightImageError","maxWidthImageError","minHeightImageError","minWidthImageError","retryFailTooManyItems","concurrent","mandatory","paramNames","partByteOffset","chunkSize","totalFileSize","totalParts","partSize","jsonPayload","resetOnStatus","recordsExpireIn","resuming","customKeys","fileOrBlobName","sendCredentials","interceptSubmit","customResizer","sendOriginal","orient","defaultType","defaultQuality","failureText","includeExif","sizes","ImageGenerator","_scaler","handleNewFile","TotalProgress","entry","AjaxRequester","shouldParamsBeInQueryString","queue","requestData","acceptHeader","validMethods","contentType","mandatedParams","allowXRequestedWithAndCacheControl","successfulResponseCodes","DELETE","PATCH","POST","PUT","GET","onSend","isSimpleMethod","containsNonSimpleHeaders","containsNonSimple","header","isXdr","getCorsAjaxTransport","ontimeout","onprogress","getXhrOrXdr","suppliedXhr","dequeue","nextId","sendRequest","xdrError","isResponseSuccessful","getParams","onDemandParams","additionalParams","optXhr","payload","url","createUrl","additionalQueryParams","getXdrLoadHandler","getXdrErrorHandler","onreadystatechange","getXhrReadyStateChangeHandler","registerForUploadProgress","open","setHeaders","send","stringify","addToPath","readyState","lengthComputable","onDemandHeaders","additionalHeaders","allHeaders","setRequestHeader","responseCode","prepareToSend","initTransport","path","cacheBuster","withPath","appendToPath","withParams","withQueryParams","_additionalQueryParams_","withHeaders","withPayload","thePayload","withCacheBuster","qqtimestamp","Date","getTime","canceled","UploadHandler","proxy","fileState","fileItem","onCancelRetVal","getThirdPartyFileId","_getFileState","_setThirdPartyFileId","thirdPartyFileId","_wasCanceled","controller","chunkingPossible","concurrentChunkingPossible","preventRetryResponse","chunked","chunkIdx","_getChunkData","attemptingResume","chunkProgress","_getChunkDataForCallback","finalize","finalizeChunks","normaizedResponse","normalizeResponse","_maybeDeletePersistedChunkData","cleanup","normalizedResponse","handleFailure","clearCachedChunk","responseToReport","inProgressIdx","inProgressChunksArray","inProgress","remaining","unshift","ignoreFailure","keys","_getXhrs","ckid","ckXhr","abort","_cancelled","moveInProgressToRemaining","connectionManager","free","hasMoreParts","nextPart","nextIdx","_getTotalChunks","reevaluateChunking","sendNext","inProgressChunks","available","onUploadChunkPromise","requestOverrides","uploadChunkData","overrides","uploadChunk","inProgressChunkIdx","_maybePersistChunkedState","clearXhr","_open","_openChunks","_waiting","openChunkEntriesCount","openChunksCount","openChunkIndexes","dontAllowNext","allowNext","waitingIndex","connectionsIndex","getProxyOrBlob","getWaitingOrConnected","waitingOrConnected","chunks","parseInt","isUsingConnection","pop","openChunksEntry","simple","uploadFile","maybeNewUuid","_clearXhrs","getProxy","initHandler","handlerType","traditional","handlerModuleSubtype","onFinalizing","_removeExpiredChunkingRecords","isDeferredEligibleForUpload","maybeDefer","generatedBlob","updateBlob","maybeSendDeferredFiles","errorResponse","idsInGroup","uploadedThisId","idInGroup","now","originalResponse","_shouldChunkThisFile","blobToUpload","cancelRetVal","isProxied","_getLocalStorageId","isAttemptingResume","WindowReceiveMessage","callbackWrapperDetachers","receiveMessage","onMessageCallbackWrapper","stopReceivingMessages","detacher","FormUploadHandler","formHandlerInstanceId","onloadCallbacks","detachLoadEvents","postMessageCallbackTimers","isCors","corsMessageReceiver","expungeFile","iframe","getElementById","_getIframeName","getFileIdForIframeName","iframeName","initIframeForUpload","body","registerPostMessageCallback","_parseJsonResponse","onloadCallback","_detachLoadEvent","_attachLoadEvent","responseDescriptor","contentDocument","_createIframe","_initFormForUpload","targetName","innerHtmlOrMessage","XhrUploadHandler","getChunkSize","chunkFiles","resumeEnabled","xhrId","ajaxRequester","_getAjaxRequester","blobOrProxy","_initTempState","_maybePrepareForResume","cachedChunks","tempState","xhrs","ajaxRequesters","responseParser","lastChunkIdx","_getXhr","resumableFilesData","_iterateResumeRecords","notResumable","optInProgress","optRemaining","chunkingState","reverse","paused","totalChunks","parts","newBlob","chunkId","_createXhr","optChunkIdx","_registerXhr","chunkIndex","fileSize","startBytes","endBytes","part","count","startByte","endByte","formatVersion","localStorageId","forEach","_getMimeType","_getPersistableData","ceil","_markNotResumable","removeItem","persistedData","lastUpdated","_registerProgressHandler","progressCalculator","totalSuccessfullyLoadedForFile","loadedForRequest","totalForRequest","estActualChunkLoaded","totalLoadedForFile","chunkLoaded","optAjaxRequester","xhrsId","expirationDays","expirationDate","setDate","getDate","requester","getMandatedParams","_method","detectSubsampling","img","iw","naturalWidth","ih","naturalHeight","ctx","width","getContext","drawImage","getImageData","detectVerticalSquash","sy","ey","py","alpha","ratio","renderImageToDataURL","doSquash","renderImageToCanvas","maybeCalculateDownsampledDimensions","maxPixels","origHeight","origWidth","newHeight","sqrt","newWidth","modifiedDimensions","save","resize","renderImageToCanvasWithCustomResizer","imageHeight","imageWidth","orientation","targetHeight","targetWidth","transformCoordinate","d","tmpCanvas","vertSquashRatio","dw","dh","dy","tmpCtx","sx","dx","clearRect","restore","qqImageRendered","resizeInfo","sourceCanvas","sourceCanvasContext","targetCanvas","translate","rotate","PI","MegaPixImage","srcImage","errorCallback","Image","URL","createObjectURL","webkitURL","src","listeners","imageLoadListeners","render","imgWidth","imgHeight","opt","optionsKey","optionsValue","oldTargetSrc","onrender","isImg","el","isCanvas","isImgCorsSupported","crossOrigin","isCanvasSupported","determineMimeOfFileName","nameWithPath","pathSegments","isCrossOrigin","targetAnchor","targetProtocol","targetHostname","targetPort","href","protocol","port","hostname","location","registerImgLoadListeners","registerCanvasDrawImageListener","registerThumbnailRenderedListener","registered","draw","drawPreview","Identify","megapixErrorHandler","isPreviewable","dummyExif","exif","Exif","mpImg","Orientation","failureMsg","drawOnCanvasOrImgFromUrl","canvasOrImg","tempImg","tempImgRender","rendered","drawOnImgFromUrlWithCssScaling","drawFromUrl","fileBlobOrUrl","_testing","TAG_IDS","TAG_INFO","274","parseLittleEndian","hex","pow","seekToApp1","offset","theOffset","thePromise","segmentLength","getApp1Offset","isLittleEndian","app1Start","getDirEntryCount","littleEndian","getIfd","dirEntries","getDirEntries","ifdHex","getTagValues","TAG_VAL_OFFSET","tagsToFind","vals","idHex","tagsToFindIdx","tagValHex","tagValLength","parser","onParseFailure","app1Offset","dirEntryCount","tagValues","isIdentifiable","magicBytes","questionableBytes","identifiable","magicBytesEntries","magicBytesArrayEntry","previewable","isPreviewableSync","PREVIEWABLE_MIME_TYPES","fileMime","isRecognizedImage","image/jpeg","image/gif","image/png","image/bmp","image/tiff","hasNonZeroLimits","limits","atLeastOne","limit","getWidthHeight","sizeDetermination","getFailingLimit","dimensions","failingLimit","limitName","limitValue","limitMatcher","dimensionPropName","actualValue","validationEffort","isJsonResponseValid","handleFileItems","fileItems","someItemsIgnored","err","refreshEffort","refreshCompleteCallback","requesterOptions","SessionAjaxRequester","queryServer","responseText","includeOriginal","failedToScaleText","_getSortedSizes","getFileRecords","originalFileUuid","originalFileName","originalBlobOrBlobData","records","originalBlob","sizeRecord","outputType","_determineOutputType","requestedType","refType","_getName","_generateScaledImage","failedText","api","scaledIds","originalId","blobSize","scaledId","qqparentuuid","qqparentsize","param","scalingEffort","scalingOptions","scaler","referenceType","scaledVersionProperties","startOfExt","versionType","scaledName","scaledExt","nameAppendage","sort","a","b","sourceFile","imageGenerator","scaledImageDataUri","signalSuccess","_insertExifHeader","scaledImageDataUriWithExif","originalImage","reader","insertionEffort","originalImageDataUri","ExifRestorer","readAsDataURL","_dataUriToBlob","_createBlob","KEY_STR","encode64","output","chr1","chr2","chr3","enc1","enc2","enc3","enc4","isNaN","origFileBase64","resizedFileBase64","expectedBase64Header","rawImage","decode64","segments","slice2Segments","exifManipulation","exifArray","getExifArray","newImageArray","insertExif","aBuffer","seg","x","imageData","buf","separatePoint","mae","ato","array","rawImageArray","head","endPoint","base64test","perFileProgress","totalLoaded","totalSize","lastLoadedSent","lastTotalSent","callbackProxy","noRetryableFiles","retryable","none","failedId","updateTotalProgress","onNew","newLoaded","newTotal","oldLoaded","oldTotal","detachPasteHandler","isImage","registerPasteHandler","clipboardData","getAsFile","unregisterPasteHandler","startUpload","formEl","_form2Obj","determineNewEndpoint","validateForm","nativeSubmit","checkValidity","maybeUploadOnSubmit","submit","determineFormEl","notIrrelevantType","irrelevantTypes","radioOrCheckbox","ignoreValue","checked","selectValue","select","selected","elements","getIframeContentJson","doc","innerHtml","contentWindow","nodeValue","createForm","responseFromMessage","multipart","addChunkingSpecificParams","allChunksDoneRequester","AllChunksDoneAjaxRequester","createReadyStateChangedHandler","onUploadOrChunkComplete","getChunksCompleteParams","isErrorUploadResponse","parseResponse","sendChunksCompleteRequest","complete","setParamsAndGetEntityToSend","entityToSendParams","xhrOverrides","customParams","defaultParams","xhrOverrideParams","setUploadHeaders","headersOptions","headerOverrides","headerName","headerValue","extraHeaders","uploadChunkParams","toSend","promises","endpointHandler"],"mappings":"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","file":"fine-uploader.core.js","sourcesContent":[null,"/*globals window, navigator, document, FormData, File, HTMLInputElement, XMLHttpRequest, Blob, Storage, ActiveXObject */\n/* jshint -W079 */\nvar qq = function(element) {\n    \"use strict\";\n\n    return {\n        hide: function() {\n            element.style.display = \"none\";\n            return this;\n        },\n\n        /** Returns the function which detaches attached event */\n        attach: function(type, fn) {\n            if (element.addEventListener) {\n                element.addEventListener(type, fn, false);\n            } else if (element.attachEvent) {\n                element.attachEvent(\"on\" + type, fn);\n            }\n            return function() {\n                qq(element).detach(type, fn);\n            };\n        },\n\n        detach: function(type, fn) {\n            if (element.removeEventListener) {\n                element.removeEventListener(type, fn, false);\n            } else if (element.attachEvent) {\n                element.detachEvent(\"on\" + type, fn);\n            }\n            return this;\n        },\n\n        contains: function(descendant) {\n            // The [W3C spec](http://www.w3.org/TR/domcore/#dom-node-contains)\n            // says a `null` (or ostensibly `undefined`) parameter\n            // passed into `Node.contains` should result in a false return value.\n            // IE7 throws an exception if the parameter is `undefined` though.\n            if (!descendant) {\n                return false;\n            }\n\n            // compareposition returns false in this case\n            if (element === descendant) {\n                return true;\n            }\n\n            if (element.contains) {\n                return element.contains(descendant);\n            } else {\n                /*jslint bitwise: true*/\n                return !!(descendant.compareDocumentPosition(element) & 8);\n            }\n        },\n\n        /**\n         * Insert this element before elementB.\n         */\n        insertBefore: function(elementB) {\n            elementB.parentNode.insertBefore(element, elementB);\n            return this;\n        },\n\n        remove: function() {\n            element.parentNode.removeChild(element);\n            return this;\n        },\n\n        /**\n         * Sets styles for an element.\n         * Fixes opacity in IE6-8.\n         */\n        css: function(styles) {\n            /*jshint eqnull: true*/\n            if (element.style == null) {\n                throw new qq.Error(\"Can't apply style to node as it is not on the HTMLElement prototype chain!\");\n            }\n\n            /*jshint -W116*/\n            if (styles.opacity != null) {\n                if (typeof element.style.opacity !== \"string\" && typeof (element.filters) !== \"undefined\") {\n                    styles.filter = \"alpha(opacity=\" + Math.round(100 * styles.opacity) + \")\";\n                }\n            }\n            qq.extend(element.style, styles);\n\n            return this;\n        },\n\n        hasClass: function(name, considerParent) {\n            var re = new RegExp(\"(^| )\" + name + \"( |$)\");\n            return re.test(element.className) || !!(considerParent && re.test(element.parentNode.className));\n        },\n\n        addClass: function(name) {\n            if (!qq(element).hasClass(name)) {\n                element.className += \" \" + name;\n            }\n            return this;\n        },\n\n        removeClass: function(name) {\n            var re = new RegExp(\"(^| )\" + name + \"( |$)\");\n            element.className = element.className.replace(re, \" \").replace(/^\\s+|\\s+$/g, \"\");\n            return this;\n        },\n\n        getByClass: function(className, first) {\n            var candidates,\n                result = [];\n\n            if (first && element.querySelector) {\n                return element.querySelector(\".\" + className);\n            }\n            else if (element.querySelectorAll) {\n                return element.querySelectorAll(\".\" + className);\n            }\n\n            candidates = element.getElementsByTagName(\"*\");\n\n            qq.each(candidates, function(idx, val) {\n                if (qq(val).hasClass(className)) {\n                    result.push(val);\n                }\n            });\n            return first ? result[0] : result;\n        },\n\n        getFirstByClass: function(className) {\n            return qq(element).getByClass(className, true);\n        },\n\n        children: function() {\n            var children = [],\n                child = element.firstChild;\n\n            while (child) {\n                if (child.nodeType === 1) {\n                    children.push(child);\n                }\n                child = child.nextSibling;\n            }\n\n            return children;\n        },\n\n        setText: function(text) {\n            element.innerText = text;\n            element.textContent = text;\n            return this;\n        },\n\n        clearText: function() {\n            return qq(element).setText(\"\");\n        },\n\n        // Returns true if the attribute exists on the element\n        // AND the value of the attribute is NOT \"false\" (case-insensitive)\n        hasAttribute: function(attrName) {\n            var attrVal;\n\n            if (element.hasAttribute) {\n\n                if (!element.hasAttribute(attrName)) {\n                    return false;\n                }\n\n                /*jshint -W116*/\n                return (/^false$/i).exec(element.getAttribute(attrName)) == null;\n            }\n            else {\n                attrVal = element[attrName];\n\n                if (attrVal === undefined) {\n                    return false;\n                }\n\n                /*jshint -W116*/\n                return (/^false$/i).exec(attrVal) == null;\n            }\n        }\n    };\n};\n\n(function() {\n    \"use strict\";\n\n    qq.canvasToBlob = function(canvas, mime, quality) {\n        return qq.dataUriToBlob(canvas.toDataURL(mime, quality));\n    };\n\n    qq.dataUriToBlob = function(dataUri) {\n        var arrayBuffer, byteString,\n            createBlob = function(data, mime) {\n                var BlobBuilder = window.BlobBuilder ||\n                        window.WebKitBlobBuilder ||\n                        window.MozBlobBuilder ||\n                        window.MSBlobBuilder,\n                    blobBuilder = BlobBuilder && new BlobBuilder();\n\n                if (blobBuilder) {\n                    blobBuilder.append(data);\n                    return blobBuilder.getBlob(mime);\n                }\n                else {\n                    return new Blob([data], {type: mime});\n                }\n            },\n            intArray, mimeString;\n\n        // convert base64 to raw binary data held in a string\n        if (dataUri.split(\",\")[0].indexOf(\"base64\") >= 0) {\n            byteString = atob(dataUri.split(\",\")[1]);\n        }\n        else {\n            byteString = decodeURI(dataUri.split(\",\")[1]);\n        }\n\n        // extract the MIME\n        mimeString = dataUri.split(\",\")[0]\n            .split(\":\")[1]\n            .split(\";\")[0];\n\n        // write the bytes of the binary string to an ArrayBuffer\n        arrayBuffer = new ArrayBuffer(byteString.length);\n        intArray = new Uint8Array(arrayBuffer);\n        qq.each(byteString, function(idx, character) {\n            intArray[idx] = character.charCodeAt(0);\n        });\n\n        return createBlob(arrayBuffer, mimeString);\n    };\n\n    qq.log = function(message, level) {\n        if (window.console) {\n            if (!level || level === \"info\") {\n                window.console.log(message);\n            }\n            else\n            {\n                if (window.console[level]) {\n                    window.console[level](message);\n                }\n                else {\n                    window.console.log(\"<\" + level + \"> \" + message);\n                }\n            }\n        }\n    };\n\n    qq.isObject = function(variable) {\n        return variable && !variable.nodeType && Object.prototype.toString.call(variable) === \"[object Object]\";\n    };\n\n    qq.isFunction = function(variable) {\n        return typeof (variable) === \"function\";\n    };\n\n    /**\n     * Check the type of a value.  Is it an \"array\"?\n     *\n     * @param value value to test.\n     * @returns true if the value is an array or associated with an `ArrayBuffer`\n     */\n    qq.isArray = function(value) {\n        return Object.prototype.toString.call(value) === \"[object Array]\" ||\n            (value && window.ArrayBuffer && value.buffer && value.buffer.constructor === ArrayBuffer);\n    };\n\n    // Looks for an object on a `DataTransfer` object that is associated with drop events when utilizing the Filesystem API.\n    qq.isItemList = function(maybeItemList) {\n        return Object.prototype.toString.call(maybeItemList) === \"[object DataTransferItemList]\";\n    };\n\n    // Looks for an object on a `NodeList` or an `HTMLCollection`|`HTMLFormElement`|`HTMLSelectElement`\n    // object that is associated with collections of Nodes.\n    qq.isNodeList = function(maybeNodeList) {\n        return Object.prototype.toString.call(maybeNodeList) === \"[object NodeList]\" ||\n            // If `HTMLCollection` is the actual type of the object, we must determine this\n            // by checking for expected properties/methods on the object\n            (maybeNodeList.item && maybeNodeList.namedItem);\n    };\n\n    qq.isString = function(maybeString) {\n        return Object.prototype.toString.call(maybeString) === \"[object String]\";\n    };\n\n    qq.trimStr = function(string) {\n        if (String.prototype.trim) {\n            return string.trim();\n        }\n\n        return string.replace(/^\\s+|\\s+$/g, \"\");\n    };\n\n    /**\n     * @param str String to format.\n     * @returns {string} A string, swapping argument values with the associated occurrence of {} in the passed string.\n     */\n    qq.format = function(str) {\n\n        var args =  Array.prototype.slice.call(arguments, 1),\n            newStr = str,\n            nextIdxToReplace = newStr.indexOf(\"{}\");\n\n        qq.each(args, function(idx, val) {\n            var strBefore = newStr.substring(0, nextIdxToReplace),\n                strAfter = newStr.substring(nextIdxToReplace + 2);\n\n            newStr = strBefore + val + strAfter;\n            nextIdxToReplace = newStr.indexOf(\"{}\", nextIdxToReplace + val.length);\n\n            // End the loop if we have run out of tokens (when the arguments exceed the # of tokens)\n            if (nextIdxToReplace < 0) {\n                return false;\n            }\n        });\n\n        return newStr;\n    };\n\n    qq.isFile = function(maybeFile) {\n        return window.File && Object.prototype.toString.call(maybeFile) === \"[object File]\";\n    };\n\n    qq.isFileList = function(maybeFileList) {\n        return window.FileList && Object.prototype.toString.call(maybeFileList) === \"[object FileList]\";\n    };\n\n    qq.isFileOrInput = function(maybeFileOrInput) {\n        return qq.isFile(maybeFileOrInput) || qq.isInput(maybeFileOrInput);\n    };\n\n    qq.isInput = function(maybeInput, notFile) {\n        var evaluateType = function(type) {\n            var normalizedType = type.toLowerCase();\n\n            if (notFile) {\n                return normalizedType !== \"file\";\n            }\n\n            return normalizedType === \"file\";\n        };\n\n        if (window.HTMLInputElement) {\n            if (Object.prototype.toString.call(maybeInput) === \"[object HTMLInputElement]\") {\n                if (maybeInput.type && evaluateType(maybeInput.type)) {\n                    return true;\n                }\n            }\n        }\n        if (maybeInput.tagName) {\n            if (maybeInput.tagName.toLowerCase() === \"input\") {\n                if (maybeInput.type && evaluateType(maybeInput.type)) {\n                    return true;\n                }\n            }\n        }\n\n        return false;\n    };\n\n    qq.isBlob = function(maybeBlob) {\n        if (window.Blob && Object.prototype.toString.call(maybeBlob) === \"[object Blob]\") {\n            return true;\n        }\n    };\n\n    qq.isXhrUploadSupported = function() {\n        var input = document.createElement(\"input\");\n        input.type = \"file\";\n\n        return (\n            input.multiple !== undefined &&\n                typeof File !== \"undefined\" &&\n                typeof FormData !== \"undefined\" &&\n                typeof (qq.createXhrInstance()).upload !== \"undefined\");\n    };\n\n    // Fall back to ActiveX is native XHR is disabled (possible in any version of IE).\n    qq.createXhrInstance = function() {\n        if (window.XMLHttpRequest) {\n            return new XMLHttpRequest();\n        }\n\n        try {\n            return new ActiveXObject(\"MSXML2.XMLHTTP.3.0\");\n        }\n        catch (error) {\n            qq.log(\"Neither XHR or ActiveX are supported!\", \"error\");\n            return null;\n        }\n    };\n\n    qq.isFolderDropSupported = function(dataTransfer) {\n        return dataTransfer.items &&\n            dataTransfer.items.length > 0 &&\n            dataTransfer.items[0].webkitGetAsEntry;\n    };\n\n    qq.isFileChunkingSupported = function() {\n        return !qq.androidStock() && //Android's stock browser cannot upload Blobs correctly\n            qq.isXhrUploadSupported() &&\n            (File.prototype.slice !== undefined || File.prototype.webkitSlice !== undefined || File.prototype.mozSlice !== undefined);\n    };\n\n    qq.sliceBlob = function(fileOrBlob, start, end) {\n        var slicer = fileOrBlob.slice || fileOrBlob.mozSlice || fileOrBlob.webkitSlice;\n\n        return slicer.call(fileOrBlob, start, end);\n    };\n\n    qq.arrayBufferToHex = function(buffer) {\n        var bytesAsHex = \"\",\n            bytes = new Uint8Array(buffer);\n\n        qq.each(bytes, function(idx, byt) {\n            var byteAsHexStr = byt.toString(16);\n\n            if (byteAsHexStr.length < 2) {\n                byteAsHexStr = \"0\" + byteAsHexStr;\n            }\n\n            bytesAsHex += byteAsHexStr;\n        });\n\n        return bytesAsHex;\n    };\n\n    qq.readBlobToHex = function(blob, startOffset, length) {\n        var initialBlob = qq.sliceBlob(blob, startOffset, startOffset + length),\n            fileReader = new FileReader(),\n            promise = new qq.Promise();\n\n        fileReader.onload = function() {\n            promise.success(qq.arrayBufferToHex(fileReader.result));\n        };\n\n        fileReader.onerror = promise.failure;\n\n        fileReader.readAsArrayBuffer(initialBlob);\n\n        return promise;\n    };\n\n    qq.extend = function(first, second, extendNested) {\n        qq.each(second, function(prop, val) {\n            if (extendNested && qq.isObject(val)) {\n                if (first[prop] === undefined) {\n                    first[prop] = {};\n                }\n                qq.extend(first[prop], val, true);\n            }\n            else {\n                first[prop] = val;\n            }\n        });\n\n        return first;\n    };\n\n    /**\n     * Allow properties in one object to override properties in another,\n     * keeping track of the original values from the target object.\n     *\n     * Note that the pre-overriden properties to be overriden by the source will be passed into the `sourceFn` when it is invoked.\n     *\n     * @param target Update properties in this object from some source\n     * @param sourceFn A function that, when invoked, will return properties that will replace properties with the same name in the target.\n     * @returns {object} The target object\n     */\n    qq.override = function(target, sourceFn) {\n        var super_ = {},\n            source = sourceFn(super_);\n\n        qq.each(source, function(srcPropName, srcPropVal) {\n            if (target[srcPropName] !== undefined) {\n                super_[srcPropName] = target[srcPropName];\n            }\n\n            target[srcPropName] = srcPropVal;\n        });\n\n        return target;\n    };\n\n    /**\n     * Searches for a given element (elt) in the array, returns -1 if it is not present.\n     */\n    qq.indexOf = function(arr, elt, from) {\n        if (arr.indexOf) {\n            return arr.indexOf(elt, from);\n        }\n\n        from = from || 0;\n        var len = arr.length;\n\n        if (from < 0) {\n            from += len;\n        }\n\n        for (; from < len; from += 1) {\n            if (arr.hasOwnProperty(from) && arr[from] === elt) {\n                return from;\n            }\n        }\n        return -1;\n    };\n\n    //this is a version 4 UUID\n    qq.getUniqueId = function() {\n        return \"xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx\".replace(/[xy]/g, function(c) {\n            /*jslint eqeq: true, bitwise: true*/\n            var r = Math.random() * 16 | 0, v = c == \"x\" ? r : (r & 0x3 | 0x8);\n            return v.toString(16);\n        });\n    };\n\n    //\n    // Browsers and platforms detection\n    qq.ie = function() {\n        return navigator.userAgent.indexOf(\"MSIE\") !== -1 ||\n            navigator.userAgent.indexOf(\"Trident\") !== -1;\n    };\n\n    qq.ie7 = function() {\n        return navigator.userAgent.indexOf(\"MSIE 7\") !== -1;\n    };\n\n    qq.ie8 = function() {\n        return navigator.userAgent.indexOf(\"MSIE 8\") !== -1;\n    };\n\n    qq.ie10 = function() {\n        return navigator.userAgent.indexOf(\"MSIE 10\") !== -1;\n    };\n\n    qq.ie11 = function() {\n        return qq.ie() && navigator.userAgent.indexOf(\"rv:11\") !== -1;\n    };\n\n    qq.edge = function() {\n        return navigator.userAgent.indexOf(\"Edge\") >= 0;\n    };\n\n    qq.safari = function() {\n        return navigator.vendor !== undefined && navigator.vendor.indexOf(\"Apple\") !== -1;\n    };\n\n    qq.chrome = function() {\n        return navigator.vendor !== undefined && navigator.vendor.indexOf(\"Google\") !== -1;\n    };\n\n    qq.opera = function() {\n        return navigator.vendor !== undefined && navigator.vendor.indexOf(\"Opera\") !== -1;\n    };\n\n    qq.firefox = function() {\n        return (!qq.edge() && !qq.ie11() && navigator.userAgent.indexOf(\"Mozilla\") !== -1 && navigator.vendor !== undefined && navigator.vendor === \"\");\n    };\n\n    qq.windows = function() {\n        return navigator.platform === \"Win32\";\n    };\n\n    qq.android = function() {\n        return navigator.userAgent.toLowerCase().indexOf(\"android\") !== -1;\n    };\n\n    // We need to identify the Android stock browser via the UA string to work around various bugs in this browser,\n    // such as the one that prevents a `Blob` from being uploaded.\n    qq.androidStock = function() {\n        return qq.android() && navigator.userAgent.toLowerCase().indexOf(\"chrome\") < 0;\n    };\n\n    qq.ios6 = function() {\n        return qq.ios() && navigator.userAgent.indexOf(\" OS 6_\") !== -1;\n    };\n\n    qq.ios7 = function() {\n        return qq.ios() && navigator.userAgent.indexOf(\" OS 7_\") !== -1;\n    };\n\n    qq.ios8 = function() {\n        return qq.ios() && navigator.userAgent.indexOf(\" OS 8_\") !== -1;\n    };\n\n    // iOS 8.0.0\n    qq.ios800 = function() {\n        return qq.ios() && navigator.userAgent.indexOf(\" OS 8_0 \") !== -1;\n    };\n\n    qq.ios = function() {\n        /*jshint -W014 */\n        return navigator.userAgent.indexOf(\"iPad\") !== -1\n            || navigator.userAgent.indexOf(\"iPod\") !== -1\n            || navigator.userAgent.indexOf(\"iPhone\") !== -1;\n    };\n\n    qq.iosChrome = function() {\n        return qq.ios() && navigator.userAgent.indexOf(\"CriOS\") !== -1;\n    };\n\n    qq.iosSafari = function() {\n        return qq.ios() && !qq.iosChrome() && navigator.userAgent.indexOf(\"Safari\") !== -1;\n    };\n\n    qq.iosSafariWebView = function() {\n        return qq.ios() && !qq.iosChrome() && !qq.iosSafari();\n    };\n\n    //\n    // Events\n\n    qq.preventDefault = function(e) {\n        if (e.preventDefault) {\n            e.preventDefault();\n        } else {\n            e.returnValue = false;\n        }\n    };\n\n    /**\n     * Creates and returns element from html string\n     * Uses innerHTML to create an element\n     */\n    qq.toElement = (function() {\n        var div = document.createElement(\"div\");\n        return function(html) {\n            div.innerHTML = html;\n            var element = div.firstChild;\n            div.removeChild(element);\n            return element;\n        };\n    }());\n\n    //key and value are passed to callback for each entry in the iterable item\n    qq.each = function(iterableItem, callback) {\n        var keyOrIndex, retVal;\n\n        if (iterableItem) {\n            // Iterate through [`Storage`](http://www.w3.org/TR/webstorage/#the-storage-interface) items\n            if (window.Storage && iterableItem.constructor === window.Storage) {\n                for (keyOrIndex = 0; keyOrIndex < iterableItem.length; keyOrIndex++) {\n                    retVal = callback(iterableItem.key(keyOrIndex), iterableItem.getItem(iterableItem.key(keyOrIndex)));\n                    if (retVal === false) {\n                        break;\n                    }\n                }\n            }\n            // `DataTransferItemList` & `NodeList` objects are array-like and should be treated as arrays\n            // when iterating over items inside the object.\n            else if (qq.isArray(iterableItem) || qq.isItemList(iterableItem) || qq.isNodeList(iterableItem)) {\n                for (keyOrIndex = 0; keyOrIndex < iterableItem.length; keyOrIndex++) {\n                    retVal = callback(keyOrIndex, iterableItem[keyOrIndex]);\n                    if (retVal === false) {\n                        break;\n                    }\n                }\n            }\n            else if (qq.isString(iterableItem)) {\n                for (keyOrIndex = 0; keyOrIndex < iterableItem.length; keyOrIndex++) {\n                    retVal = callback(keyOrIndex, iterableItem.charAt(keyOrIndex));\n                    if (retVal === false) {\n                        break;\n                    }\n                }\n            }\n            else {\n                for (keyOrIndex in iterableItem) {\n                    if (Object.prototype.hasOwnProperty.call(iterableItem, keyOrIndex)) {\n                        retVal = callback(keyOrIndex, iterableItem[keyOrIndex]);\n                        if (retVal === false) {\n                            break;\n                        }\n                    }\n                }\n            }\n        }\n    };\n\n    //include any args that should be passed to the new function after the context arg\n    qq.bind = function(oldFunc, context) {\n        if (qq.isFunction(oldFunc)) {\n            var args =  Array.prototype.slice.call(arguments, 2);\n\n            return function() {\n                var newArgs = qq.extend([], args);\n                if (arguments.length) {\n                    newArgs = newArgs.concat(Array.prototype.slice.call(arguments));\n                }\n                return oldFunc.apply(context, newArgs);\n            };\n        }\n\n        throw new Error(\"first parameter must be a function!\");\n    };\n\n    /**\n     * obj2url() takes a json-object as argument and generates\n     * a querystring. pretty much like jQuery.param()\n     *\n     * how to use:\n     *\n     *    `qq.obj2url({a:'b',c:'d'},'http://any.url/upload?otherParam=value');`\n     *\n     * will result in:\n     *\n     *    `http://any.url/upload?otherParam=value&a=b&c=d`\n     *\n     * @param  Object JSON-Object\n     * @param  String current querystring-part\n     * @return String encoded querystring\n     */\n    qq.obj2url = function(obj, temp, prefixDone) {\n        /*jshint laxbreak: true*/\n        var uristrings = [],\n            prefix = \"&\",\n            add = function(nextObj, i) {\n                var nextTemp = temp\n                    ? (/\\[\\]$/.test(temp)) // prevent double-encoding\n                    ? temp\n                    : temp + \"[\" + i + \"]\"\n                    : i;\n                if ((nextTemp !== \"undefined\") && (i !== \"undefined\")) {\n                    uristrings.push(\n                        (typeof nextObj === \"object\")\n                            ? qq.obj2url(nextObj, nextTemp, true)\n                            : (Object.prototype.toString.call(nextObj) === \"[object Function]\")\n                            ? encodeURIComponent(nextTemp) + \"=\" + encodeURIComponent(nextObj())\n                            : encodeURIComponent(nextTemp) + \"=\" + encodeURIComponent(nextObj)\n                    );\n                }\n            };\n\n        if (!prefixDone && temp) {\n            prefix = (/\\?/.test(temp)) ? (/\\?$/.test(temp)) ? \"\" : \"&\" : \"?\";\n            uristrings.push(temp);\n            uristrings.push(qq.obj2url(obj));\n        } else if ((Object.prototype.toString.call(obj) === \"[object Array]\") && (typeof obj !== \"undefined\")) {\n            qq.each(obj, function(idx, val) {\n                add(val, idx);\n            });\n        } else if ((typeof obj !== \"undefined\") && (obj !== null) && (typeof obj === \"object\")) {\n            qq.each(obj, function(prop, val) {\n                add(val, prop);\n            });\n        } else {\n            uristrings.push(encodeURIComponent(temp) + \"=\" + encodeURIComponent(obj));\n        }\n\n        if (temp) {\n            return uristrings.join(prefix);\n        } else {\n            return uristrings.join(prefix)\n                .replace(/^&/, \"\")\n                .replace(/%20/g, \"+\");\n        }\n    };\n\n    qq.obj2FormData = function(obj, formData, arrayKeyName) {\n        if (!formData) {\n            formData = new FormData();\n        }\n\n        qq.each(obj, function(key, val) {\n            key = arrayKeyName ? arrayKeyName + \"[\" + key + \"]\" : key;\n\n            if (qq.isObject(val)) {\n                qq.obj2FormData(val, formData, key);\n            }\n            else if (qq.isFunction(val)) {\n                formData.append(key, val());\n            }\n            else {\n                formData.append(key, val);\n            }\n        });\n\n        return formData;\n    };\n\n    qq.obj2Inputs = function(obj, form) {\n        var input;\n\n        if (!form) {\n            form = document.createElement(\"form\");\n        }\n\n        qq.obj2FormData(obj, {\n            append: function(key, val) {\n                input = document.createElement(\"input\");\n                input.setAttribute(\"name\", key);\n                input.setAttribute(\"value\", val);\n                form.appendChild(input);\n            }\n        });\n\n        return form;\n    };\n\n    /**\n     * Not recommended for use outside of Fine Uploader since this falls back to an unchecked eval if JSON.parse is not\n     * implemented.  For a more secure JSON.parse polyfill, use Douglas Crockford's json2.js.\n     */\n    qq.parseJson = function(json) {\n        /*jshint evil: true*/\n        if (window.JSON && qq.isFunction(JSON.parse)) {\n            return JSON.parse(json);\n        } else {\n            return eval(\"(\" + json + \")\");\n        }\n    };\n\n    /**\n     * Retrieve the extension of a file, if it exists.\n     *\n     * @param filename\n     * @returns {string || undefined}\n     */\n    qq.getExtension = function(filename) {\n        var extIdx = filename.lastIndexOf(\".\") + 1;\n\n        if (extIdx > 0) {\n            return filename.substr(extIdx, filename.length - extIdx);\n        }\n    };\n\n    qq.getFilename = function(blobOrFileInput) {\n        /*jslint regexp: true*/\n\n        if (qq.isInput(blobOrFileInput)) {\n            // get input value and remove path to normalize\n            return blobOrFileInput.value.replace(/.*(\\/|\\\\)/, \"\");\n        }\n        else if (qq.isFile(blobOrFileInput)) {\n            if (blobOrFileInput.fileName !== null && blobOrFileInput.fileName !== undefined) {\n                return blobOrFileInput.fileName;\n            }\n        }\n\n        return blobOrFileInput.name;\n    };\n\n    /**\n     * A generic module which supports object disposing in dispose() method.\n     * */\n    qq.DisposeSupport = function() {\n        var disposers = [];\n\n        return {\n            /** Run all registered disposers */\n            dispose: function() {\n                var disposer;\n                do {\n                    disposer = disposers.shift();\n                    if (disposer) {\n                        disposer();\n                    }\n                }\n                while (disposer);\n            },\n\n            /** Attach event handler and register de-attacher as a disposer */\n            attach: function() {\n                var args = arguments;\n                /*jslint undef:true*/\n                this.addDisposer(qq(args[0]).attach.apply(this, Array.prototype.slice.call(arguments, 1)));\n            },\n\n            /** Add disposer to the collection */\n            addDisposer: function(disposeFunction) {\n                disposers.push(disposeFunction);\n            }\n        };\n    };\n}());\n","/* globals define, module, global, qq */\n(function() {\n    \"use strict\";\n    if (typeof define === \"function\" && define.amd) {\n        define(function() {\n            return qq;\n        });\n    }\n    else if (typeof module !== \"undefined\" && module.exports) {\n        module.exports = qq;\n    }\n    else {\n        global.qq = qq;\n    }\n}());\n","/* globals qq */\n/**\n * Fine Uploader top-level Error container.  Inherits from `Error`.\n */\n(function() {\n    \"use strict\";\n\n    qq.Error = function(message) {\n        this.message = \"[Fine Uploader \" + qq.version + \"] \" + message;\n    };\n\n    qq.Error.prototype = new Error();\n}());\n","/*global qq */\nqq.version = \"5.16.2\";\n","/* globals qq */\nqq.supportedFeatures = (function() {\n    \"use strict\";\n\n    var supportsUploading,\n        supportsUploadingBlobs,\n        supportsFileDrop,\n        supportsAjaxFileUploading,\n        supportsFolderDrop,\n        supportsChunking,\n        supportsResume,\n        supportsUploadViaPaste,\n        supportsUploadCors,\n        supportsDeleteFileXdr,\n        supportsDeleteFileCorsXhr,\n        supportsDeleteFileCors,\n        supportsFolderSelection,\n        supportsImagePreviews,\n        supportsUploadProgress;\n\n    function testSupportsFileInputElement() {\n        var supported = true,\n            tempInput;\n\n        try {\n            tempInput = document.createElement(\"input\");\n            tempInput.type = \"file\";\n            qq(tempInput).hide();\n\n            if (tempInput.disabled) {\n                supported = false;\n            }\n        }\n        catch (ex) {\n            supported = false;\n        }\n\n        return supported;\n    }\n\n    //only way to test for complete Clipboard API support at this time\n    function isChrome14OrHigher() {\n        return (qq.chrome() || qq.opera()) &&\n            navigator.userAgent.match(/Chrome\\/[1][4-9]|Chrome\\/[2-9][0-9]/) !== undefined;\n    }\n\n    //Ensure we can send cross-origin `XMLHttpRequest`s\n    function isCrossOriginXhrSupported() {\n        if (window.XMLHttpRequest) {\n            var xhr = qq.createXhrInstance();\n\n            //Commonly accepted test for XHR CORS support.\n            return xhr.withCredentials !== undefined;\n        }\n\n        return false;\n    }\n\n    //Test for (terrible) cross-origin ajax transport fallback for IE9 and IE8\n    function isXdrSupported() {\n        return window.XDomainRequest !== undefined;\n    }\n\n    // CORS Ajax requests are supported if it is either possible to send credentialed `XMLHttpRequest`s,\n    // or if `XDomainRequest` is an available alternative.\n    function isCrossOriginAjaxSupported() {\n        if (isCrossOriginXhrSupported()) {\n            return true;\n        }\n\n        return isXdrSupported();\n    }\n\n    function isFolderSelectionSupported() {\n        // We know that folder selection is only supported in Chrome via this proprietary attribute for now\n        return document.createElement(\"input\").webkitdirectory !== undefined;\n    }\n\n    function isLocalStorageSupported() {\n        try {\n            return !!window.localStorage &&\n                // unpatched versions of IE10/11 have buggy impls of localStorage where setItem is a string\n                qq.isFunction(window.localStorage.setItem);\n        }\n        catch (error) {\n            // probably caught a security exception, so no localStorage for you\n            return false;\n        }\n    }\n\n    function isDragAndDropSupported() {\n        var span = document.createElement(\"span\");\n\n        return (\"draggable\" in span || (\"ondragstart\" in span && \"ondrop\" in span)) &&\n            !qq.android() && !qq.ios();\n    }\n\n    supportsUploading = testSupportsFileInputElement();\n\n    supportsAjaxFileUploading = supportsUploading && qq.isXhrUploadSupported();\n\n    supportsUploadingBlobs = supportsAjaxFileUploading && !qq.androidStock();\n\n    supportsFileDrop = supportsAjaxFileUploading && isDragAndDropSupported();\n\n    // adapted from https://stackoverflow.com/a/23278460/486979\n    supportsFolderDrop = supportsFileDrop && (function() {\n        var input = document.createElement(\"input\");\n\n        input.type = \"file\";\n        return !!(\"webkitdirectory\" in (input || document.querySelectorAll(\"input[type=file]\")[0]));\n    }());\n\n    supportsChunking = supportsAjaxFileUploading && qq.isFileChunkingSupported();\n\n    supportsResume = supportsAjaxFileUploading && supportsChunking && isLocalStorageSupported();\n\n    supportsUploadViaPaste = supportsAjaxFileUploading && isChrome14OrHigher();\n\n    supportsUploadCors = supportsUploading && (window.postMessage !== undefined || supportsAjaxFileUploading);\n\n    supportsDeleteFileCorsXhr = isCrossOriginXhrSupported();\n\n    supportsDeleteFileXdr = isXdrSupported();\n\n    supportsDeleteFileCors = isCrossOriginAjaxSupported();\n\n    supportsFolderSelection = isFolderSelectionSupported();\n\n    supportsImagePreviews = supportsAjaxFileUploading && window.FileReader !== undefined;\n\n    supportsUploadProgress = (function() {\n        if (supportsAjaxFileUploading) {\n            return !qq.androidStock() && !qq.iosChrome();\n        }\n        return false;\n    }());\n\n    return {\n        ajaxUploading: supportsAjaxFileUploading,\n        blobUploading: supportsUploadingBlobs,\n        canDetermineSize: supportsAjaxFileUploading,\n        chunking: supportsChunking,\n        deleteFileCors: supportsDeleteFileCors,\n        deleteFileCorsXdr: supportsDeleteFileXdr, //NOTE: will also return true in IE10, where XDR is also supported\n        deleteFileCorsXhr: supportsDeleteFileCorsXhr,\n        dialogElement: !!window.HTMLDialogElement,\n        fileDrop: supportsFileDrop,\n        folderDrop: supportsFolderDrop,\n        folderSelection: supportsFolderSelection,\n        imagePreviews: supportsImagePreviews,\n        imageValidation: supportsImagePreviews,\n        itemSizeValidation: supportsAjaxFileUploading,\n        pause: supportsChunking,\n        progressBar: supportsUploadProgress,\n        resume: supportsResume,\n        scaling: supportsImagePreviews && supportsUploadingBlobs,\n        tiffPreviews: qq.safari(), // Not the best solution, but simple and probably accurate enough (for now)\n        unlimitedScaledImageSize: !qq.ios(), // false simply indicates that there is some known limit\n        uploading: supportsUploading,\n        uploadCors: supportsUploadCors,\n        uploadCustomHeaders: supportsAjaxFileUploading,\n        uploadNonMultipart: supportsAjaxFileUploading,\n        uploadViaPaste: supportsUploadViaPaste\n    };\n\n}());\n","/*globals qq*/\n\n// Is the passed object a promise instance?\nqq.isGenericPromise = function(maybePromise) {\n    \"use strict\";\n    return !!(maybePromise && maybePromise.then && qq.isFunction(maybePromise.then));\n};\n\nqq.Promise = function() {\n    \"use strict\";\n\n    var successArgs, failureArgs,\n        successCallbacks = [],\n        failureCallbacks = [],\n        doneCallbacks = [],\n        state = 0;\n\n    qq.extend(this, {\n        then: function(onSuccess, onFailure) {\n            if (state === 0) {\n                if (onSuccess) {\n                    successCallbacks.push(onSuccess);\n                }\n                if (onFailure) {\n                    failureCallbacks.push(onFailure);\n                }\n            }\n            else if (state === -1) {\n                onFailure && onFailure.apply(null, failureArgs);\n            }\n            else if (onSuccess) {\n                onSuccess.apply(null, successArgs);\n            }\n\n            return this;\n        },\n\n        done: function(callback) {\n            if (state === 0) {\n                doneCallbacks.push(callback);\n            }\n            else {\n                callback.apply(null, failureArgs === undefined ? successArgs : failureArgs);\n            }\n\n            return this;\n        },\n\n        success: function() {\n            state = 1;\n            successArgs = arguments;\n\n            if (successCallbacks.length) {\n                qq.each(successCallbacks, function(idx, callback) {\n                    callback.apply(null, successArgs);\n                });\n            }\n\n            if (doneCallbacks.length) {\n                qq.each(doneCallbacks, function(idx, callback) {\n                    callback.apply(null, successArgs);\n                });\n            }\n\n            return this;\n        },\n\n        failure: function() {\n            state = -1;\n            failureArgs = arguments;\n\n            if (failureCallbacks.length) {\n                qq.each(failureCallbacks, function(idx, callback) {\n                    callback.apply(null, failureArgs);\n                });\n            }\n\n            if (doneCallbacks.length) {\n                qq.each(doneCallbacks, function(idx, callback) {\n                    callback.apply(null, failureArgs);\n                });\n            }\n\n            return this;\n        }\n    });\n};\n","/* globals qq */\n/**\n * Placeholder for a Blob that will be generated on-demand.\n *\n * @param referenceBlob Parent of the generated blob\n * @param onCreate Function to invoke when the blob must be created.  Must be promissory.\n * @constructor\n */\nqq.BlobProxy = function(referenceBlob, onCreate) {\n    \"use strict\";\n\n    qq.extend(this, {\n        referenceBlob: referenceBlob,\n\n        create: function() {\n            return onCreate(referenceBlob);\n        }\n    });\n};\n","/*globals qq*/\n\n/**\n * This module represents an upload or \"Select File(s)\" button.  It's job is to embed an opaque `<input type=\"file\">`\n * element as a child of a provided \"container\" element.  This \"container\" element (`options.element`) is used to provide\n * a custom style for the `<input type=\"file\">` element.  The ability to change the style of the container element is also\n * provided here by adding CSS classes to the container on hover/focus.\n *\n * TODO Eliminate the mouseover and mouseout event handlers since the :hover CSS pseudo-class should now be\n * available on all supported browsers.\n *\n * @param o Options to override the default values\n */\nqq.UploadButton = function(o) {\n    \"use strict\";\n\n    var self = this,\n\n        disposeSupport = new qq.DisposeSupport(),\n\n        options = {\n            // Corresponds to the `accept` attribute on the associated `<input type=\"file\">`\n            acceptFiles: null,\n\n            // \"Container\" element\n            element: null,\n\n            focusClass: \"qq-upload-button-focus\",\n\n            // A true value allows folders to be selected, if supported by the UA\n            folders: false,\n\n            // **This option will be removed** in the future as the :hover CSS pseudo-class is available on all supported browsers\n            hoverClass: \"qq-upload-button-hover\",\n\n            ios8BrowserCrashWorkaround: false,\n\n            // If true adds `multiple` attribute to `<input type=\"file\">`\n            multiple: false,\n\n            // `name` attribute of `<input type=\"file\">`\n            name: \"qqfile\",\n\n            // Called when the browser invokes the onchange handler on the `<input type=\"file\">`\n            onChange: function(input) {},\n\n            title: null\n        },\n        input, buttonId;\n\n    // Overrides any of the default option values with any option values passed in during construction.\n    qq.extend(options, o);\n\n    buttonId = qq.getUniqueId();\n\n    // Embed an opaque `<input type=\"file\">` element as a child of `options.element`.\n    function createInput() {\n        var input = document.createElement(\"input\");\n\n        input.setAttribute(qq.UploadButton.BUTTON_ID_ATTR_NAME, buttonId);\n        input.setAttribute(\"title\", options.title);\n\n        self.setMultiple(options.multiple, input);\n\n        if (options.folders && qq.supportedFeatures.folderSelection) {\n            // selecting directories is only possible in Chrome now, via a vendor-specific prefixed attribute\n            input.setAttribute(\"webkitdirectory\", \"\");\n        }\n\n        if (options.acceptFiles) {\n            input.setAttribute(\"accept\", options.acceptFiles);\n        }\n\n        input.setAttribute(\"type\", \"file\");\n        input.setAttribute(\"name\", options.name);\n\n        qq(input).css({\n            position: \"absolute\",\n            // in Opera only 'browse' button\n            // is clickable and it is located at\n            // the right side of the input\n            right: 0,\n            top: 0,\n            fontFamily: \"Arial\",\n            // It's especially important to make this an arbitrarily large value\n            // to ensure the rendered input button in IE takes up the entire\n            // space of the container element.  Otherwise, the left side of the\n            // button will require a double-click to invoke the file chooser.\n            // In other browsers, this might cause other issues, so a large font-size\n            // is only used in IE.  There is a bug in IE8 where the opacity style is  ignored\n            // in some cases when the font-size is large.  So, this workaround is not applied\n            // to IE8.\n            fontSize: qq.ie() && !qq.ie8() ? \"3500px\" : \"118px\",\n            margin: 0,\n            padding: 0,\n            cursor: \"pointer\",\n            opacity: 0\n        });\n\n        // Setting the file input's height to 100% in IE7 causes\n        // most of the visible button to be unclickable.\n        !qq.ie7() && qq(input).css({height: \"100%\"});\n\n        options.element.appendChild(input);\n\n        disposeSupport.attach(input, \"change\", function() {\n            options.onChange(input);\n        });\n\n        // **These event handlers will be removed** in the future as the :hover CSS pseudo-class is available on all supported browsers\n        disposeSupport.attach(input, \"mouseover\", function() {\n            qq(options.element).addClass(options.hoverClass);\n        });\n        disposeSupport.attach(input, \"mouseout\", function() {\n            qq(options.element).removeClass(options.hoverClass);\n        });\n\n        disposeSupport.attach(input, \"focus\", function() {\n            qq(options.element).addClass(options.focusClass);\n        });\n        disposeSupport.attach(input, \"blur\", function() {\n            qq(options.element).removeClass(options.focusClass);\n        });\n\n        return input;\n    }\n\n    // Make button suitable container for input\n    qq(options.element).css({\n        position: \"relative\",\n        overflow: \"hidden\",\n        // Make sure browse button is in the right side in Internet Explorer\n        direction: \"ltr\"\n    });\n\n    // Exposed API\n    qq.extend(this, {\n        getInput: function() {\n            return input;\n        },\n\n        getButtonId: function() {\n            return buttonId;\n        },\n\n        setMultiple: function(isMultiple, optInput) {\n            var input = optInput || this.getInput();\n\n            // Temporary workaround for bug in in iOS8 UIWebView that causes the browser to crash\n            // before the file chooser appears if the file input doesn't contain a multiple attribute.\n            // See #1283.\n            if (options.ios8BrowserCrashWorkaround && qq.ios8() && (qq.iosChrome() || qq.iosSafariWebView())) {\n                input.setAttribute(\"multiple\", \"\");\n            }\n\n            else {\n                if (isMultiple) {\n                    input.setAttribute(\"multiple\", \"\");\n                }\n                else {\n                    input.removeAttribute(\"multiple\");\n                }\n            }\n        },\n\n        setAcceptFiles: function(acceptFiles) {\n            if (acceptFiles !== options.acceptFiles) {\n                input.setAttribute(\"accept\", acceptFiles);\n            }\n        },\n\n        reset: function() {\n            if (input.parentNode) {\n                qq(input).remove();\n            }\n\n            qq(options.element).removeClass(options.focusClass);\n            input = null;\n            input = createInput();\n        }\n    });\n\n    input = createInput();\n};\n\nqq.UploadButton.BUTTON_ID_ATTR_NAME = \"qq-button-id\";\n","/*globals qq */\nqq.UploadData = function(uploaderProxy) {\n    \"use strict\";\n\n    var data = [],\n        byUuid = {},\n        byStatus = {},\n        byProxyGroupId = {},\n        byBatchId = {};\n\n    function getDataByIds(idOrIds) {\n        if (qq.isArray(idOrIds)) {\n            var entries = [];\n\n            qq.each(idOrIds, function(idx, id) {\n                entries.push(data[id]);\n            });\n\n            return entries;\n        }\n\n        return data[idOrIds];\n    }\n\n    function getDataByUuids(uuids) {\n        if (qq.isArray(uuids)) {\n            var entries = [];\n\n            qq.each(uuids, function(idx, uuid) {\n                entries.push(data[byUuid[uuid]]);\n            });\n\n            return entries;\n        }\n\n        return data[byUuid[uuids]];\n    }\n\n    function getDataByStatus(status) {\n        var statusResults = [],\n            statuses = [].concat(status);\n\n        qq.each(statuses, function(index, statusEnum) {\n            var statusResultIndexes = byStatus[statusEnum];\n\n            if (statusResultIndexes !== undefined) {\n                qq.each(statusResultIndexes, function(i, dataIndex) {\n                    statusResults.push(data[dataIndex]);\n                });\n            }\n        });\n\n        return statusResults;\n    }\n\n    qq.extend(this, {\n        /**\n         * Adds a new file to the data cache for tracking purposes.\n         *\n         * @param spec Data that describes this file.  Possible properties are:\n         *\n         * - uuid: Initial UUID for this file.\n         * - name: Initial name of this file.\n         * - size: Size of this file, omit if this cannot be determined\n         * - status: Initial `qq.status` for this file.  Omit for `qq.status.SUBMITTING`.\n         * - batchId: ID of the batch this file belongs to\n         * - proxyGroupId: ID of the proxy group associated with this file\n         * - onBeforeStatusChange(fileId): callback that is executed before the status change is broadcast\n         *\n         * @returns {number} Internal ID for this file.\n         */\n        addFile: function(spec) {\n            var status = spec.status || qq.status.SUBMITTING,\n                id = data.push({\n                    name: spec.name,\n                    originalName: spec.name,\n                    uuid: spec.uuid,\n                    size: spec.size == null ? -1 : spec.size,\n                    status: status,\n                    file: spec.file\n                }) - 1;\n\n            if (spec.batchId) {\n                data[id].batchId = spec.batchId;\n\n                if (byBatchId[spec.batchId] === undefined) {\n                    byBatchId[spec.batchId] = [];\n                }\n                byBatchId[spec.batchId].push(id);\n            }\n\n            if (spec.proxyGroupId) {\n                data[id].proxyGroupId = spec.proxyGroupId;\n\n                if (byProxyGroupId[spec.proxyGroupId] === undefined) {\n                    byProxyGroupId[spec.proxyGroupId] = [];\n                }\n                byProxyGroupId[spec.proxyGroupId].push(id);\n            }\n\n            data[id].id = id;\n            byUuid[spec.uuid] = id;\n\n            if (byStatus[status] === undefined) {\n                byStatus[status] = [];\n            }\n            byStatus[status].push(id);\n\n            spec.onBeforeStatusChange && spec.onBeforeStatusChange(id);\n            uploaderProxy.onStatusChange(id, null, status);\n\n            return id;\n        },\n\n        retrieve: function(optionalFilter) {\n            if (qq.isObject(optionalFilter) && data.length)  {\n                if (optionalFilter.id !== undefined) {\n                    return getDataByIds(optionalFilter.id);\n                }\n\n                else if (optionalFilter.uuid !== undefined) {\n                    return getDataByUuids(optionalFilter.uuid);\n                }\n\n                else if (optionalFilter.status) {\n                    return getDataByStatus(optionalFilter.status);\n                }\n            }\n            else {\n                return qq.extend([], data, true);\n            }\n        },\n\n        removeFileRef: function(id) {\n            var record = getDataByIds(id);\n\n            if (record) {\n                delete record.file;\n            }\n        },\n\n        reset: function() {\n            data = [];\n            byUuid = {};\n            byStatus = {};\n            byBatchId = {};\n        },\n\n        setStatus: function(id, newStatus) {\n            var oldStatus = data[id].status,\n                byStatusOldStatusIndex = qq.indexOf(byStatus[oldStatus], id);\n\n            byStatus[oldStatus].splice(byStatusOldStatusIndex, 1);\n\n            data[id].status = newStatus;\n\n            if (byStatus[newStatus] === undefined) {\n                byStatus[newStatus] = [];\n            }\n            byStatus[newStatus].push(id);\n\n            uploaderProxy.onStatusChange(id, oldStatus, newStatus);\n        },\n\n        uuidChanged: function(id, newUuid) {\n            var oldUuid = data[id].uuid;\n\n            data[id].uuid = newUuid;\n            byUuid[newUuid] = id;\n            delete byUuid[oldUuid];\n        },\n\n        updateName: function(id, newName) {\n            data[id].name = newName;\n        },\n\n        updateSize: function(id, newSize) {\n            data[id].size = newSize;\n        },\n\n        // Only applicable if this file has a parent that we may want to reference later.\n        setParentId: function(targetId, parentId) {\n            data[targetId].parentId = parentId;\n        },\n\n        getIdsInProxyGroup: function(id) {\n            var proxyGroupId = data[id].proxyGroupId;\n\n            if (proxyGroupId) {\n                return byProxyGroupId[proxyGroupId];\n            }\n            return [];\n        },\n\n        getIdsInBatch: function(id) {\n            var batchId = data[id].batchId;\n\n            return byBatchId[batchId];\n        }\n    });\n};\n\nqq.status = {\n    SUBMITTING: \"submitting\",\n    SUBMITTED: \"submitted\",\n    REJECTED: \"rejected\",\n    QUEUED: \"queued\",\n    CANCELED: \"canceled\",\n    PAUSED: \"paused\",\n    UPLOADING: \"uploading\",\n    UPLOAD_FINALIZING: \"upload finalizing\",\n    UPLOAD_RETRYING: \"retrying upload\",\n    UPLOAD_SUCCESSFUL: \"upload successful\",\n    UPLOAD_FAILED: \"upload failed\",\n    DELETE_FAILED: \"delete failed\",\n    DELETING: \"deleting\",\n    DELETED: \"deleted\"\n};\n","/*globals qq*/\n/**\n * Defines the public API for FineUploaderBasic mode.\n */\n(function() {\n    \"use strict\";\n\n    qq.basePublicApi = {\n        // DEPRECATED - TODO REMOVE IN NEXT MAJOR RELEASE (replaced by addFiles)\n        addBlobs: function(blobDataOrArray, params, endpoint) {\n            this.addFiles(blobDataOrArray, params, endpoint);\n        },\n\n        addInitialFiles: function(cannedFileList) {\n            var self = this;\n\n            qq.each(cannedFileList, function(index, cannedFile) {\n                self._addCannedFile(cannedFile);\n            });\n        },\n\n        addFiles: function(data, params, endpoint) {\n            this._maybeHandleIos8SafariWorkaround();\n\n            var batchId = this._storedIds.length === 0 ? qq.getUniqueId() : this._currentBatchId,\n\n                processBlob = qq.bind(function(blob) {\n                    this._handleNewFile({\n                        blob: blob,\n                        name: this._options.blobs.defaultName\n                    }, batchId, verifiedFiles);\n                }, this),\n\n                processBlobData = qq.bind(function(blobData) {\n                    this._handleNewFile(blobData, batchId, verifiedFiles);\n                }, this),\n\n                processCanvas = qq.bind(function(canvas) {\n                    var blob = qq.canvasToBlob(canvas);\n\n                    this._handleNewFile({\n                        blob: blob,\n                        name: this._options.blobs.defaultName + \".png\"\n                    }, batchId, verifiedFiles);\n                }, this),\n\n                processCanvasData = qq.bind(function(canvasData) {\n                    var normalizedQuality = canvasData.quality && canvasData.quality / 100,\n                        blob = qq.canvasToBlob(canvasData.canvas, canvasData.type, normalizedQuality);\n\n                    this._handleNewFile({\n                        blob: blob,\n                        name: canvasData.name\n                    }, batchId, verifiedFiles);\n                }, this),\n\n                processFileOrInput = qq.bind(function(fileOrInput) {\n                    if (qq.isInput(fileOrInput) && qq.supportedFeatures.ajaxUploading) {\n                        var files = Array.prototype.slice.call(fileOrInput.files),\n                            self = this;\n\n                        qq.each(files, function(idx, file) {\n                            self._handleNewFile(file, batchId, verifiedFiles);\n                        });\n                    }\n                    else {\n                        this._handleNewFile(fileOrInput, batchId, verifiedFiles);\n                    }\n                }, this),\n\n                normalizeData = function() {\n                    if (qq.isFileList(data)) {\n                        data = Array.prototype.slice.call(data);\n                    }\n                    data = [].concat(data);\n                },\n\n                self = this,\n                verifiedFiles = [];\n\n            this._currentBatchId = batchId;\n\n            if (data) {\n                normalizeData();\n\n                qq.each(data, function(idx, fileContainer) {\n                    if (qq.isFileOrInput(fileContainer)) {\n                        processFileOrInput(fileContainer);\n                    }\n                    else if (qq.isBlob(fileContainer)) {\n                        processBlob(fileContainer);\n                    }\n                    else if (qq.isObject(fileContainer)) {\n                        if (fileContainer.blob && fileContainer.name) {\n                            processBlobData(fileContainer);\n                        }\n                        else if (fileContainer.canvas && fileContainer.name) {\n                            processCanvasData(fileContainer);\n                        }\n                    }\n                    else if (fileContainer.tagName && fileContainer.tagName.toLowerCase() === \"canvas\") {\n                        processCanvas(fileContainer);\n                    }\n                    else {\n                        self.log(fileContainer + \" is not a valid file container!  Ignoring!\", \"warn\");\n                    }\n                });\n\n                this.log(\"Received \" + verifiedFiles.length + \" files.\");\n                this._prepareItemsForUpload(verifiedFiles, params, endpoint);\n            }\n        },\n\n        cancel: function(id) {\n            var uploadData = this._uploadData.retrieve({id: id});\n\n            if (uploadData && uploadData.status === qq.status.UPLOAD_FINALIZING) {\n                this.log(qq.format(\"Ignoring cancel for file ID {} ({}).  Finalizing upload.\", id, this.getName(id)), \"error\");\n            }\n            else {\n                this._handler.cancel(id);\n            }\n        },\n\n        cancelAll: function() {\n            var storedIdsCopy = [],\n                self = this;\n\n            qq.extend(storedIdsCopy, this._storedIds);\n            qq.each(storedIdsCopy, function(idx, storedFileId) {\n                self.cancel(storedFileId);\n            });\n\n            this._handler.cancelAll();\n        },\n\n        clearStoredFiles: function() {\n            this._storedIds = [];\n        },\n\n        continueUpload: function(id) {\n            var uploadData = this._uploadData.retrieve({id: id});\n\n            if (!qq.supportedFeatures.pause || !this._options.chunking.enabled) {\n                return false;\n            }\n\n            if (uploadData.status === qq.status.PAUSED) {\n                this.log(qq.format(\"Paused file ID {} ({}) will be continued.  Not paused.\", id, this.getName(id)));\n                this._uploadFile(id);\n                return true;\n            }\n            else {\n                this.log(qq.format(\"Ignoring continue for file ID {} ({}).  Not paused.\", id, this.getName(id)), \"error\");\n            }\n\n            return false;\n        },\n\n        deleteFile: function(id) {\n            return this._onSubmitDelete(id);\n        },\n\n        // TODO document?\n        doesExist: function(fileOrBlobId) {\n            return this._handler.isValid(fileOrBlobId);\n        },\n\n        // Generate a variable size thumbnail on an img or canvas,\n        // returning a promise that is fulfilled when the attempt completes.\n        // Thumbnail can either be based off of a URL for an image returned\n        // by the server in the upload response, or the associated `Blob`.\n        drawThumbnail: function(fileId, imgOrCanvas, maxSize, fromServer, customResizeFunction) {\n            var promiseToReturn = new qq.Promise(),\n                fileOrUrl, options;\n\n            if (this._imageGenerator) {\n                fileOrUrl = this._thumbnailUrls[fileId];\n                options = {\n                    customResizeFunction: customResizeFunction,\n                    maxSize: maxSize > 0 ? maxSize : null,\n                    scale: maxSize > 0\n                };\n\n                // If client-side preview generation is possible\n                // and we are not specifically looking for the image URl returned by the server...\n                if (!fromServer && qq.supportedFeatures.imagePreviews) {\n                    fileOrUrl = this.getFile(fileId);\n                }\n\n                /* jshint eqeqeq:false,eqnull:true */\n                if (fileOrUrl == null) {\n                    promiseToReturn.failure({container: imgOrCanvas, error: \"File or URL not found.\"});\n                }\n                else {\n                    this._imageGenerator.generate(fileOrUrl, imgOrCanvas, options).then(\n                        function success(modifiedContainer) {\n                            promiseToReturn.success(modifiedContainer);\n                        },\n\n                        function failure(container, reason) {\n                            promiseToReturn.failure({container: container, error: reason || \"Problem generating thumbnail\"});\n                        }\n                    );\n                }\n            }\n            else {\n                promiseToReturn.failure({container: imgOrCanvas, error: \"Missing image generator module\"});\n            }\n\n            return promiseToReturn;\n        },\n\n        getButton: function(fileId) {\n            return this._getButton(this._buttonIdsForFileIds[fileId]);\n        },\n\n        getEndpoint: function(fileId) {\n            return this._endpointStore.get(fileId);\n        },\n\n        getFile: function(fileOrBlobId) {\n            var file = this._handler.getFile(fileOrBlobId);\n            var uploadDataRecord;\n\n            if (!file) {\n                uploadDataRecord = this._uploadData.retrieve({id: fileOrBlobId});\n\n                if (uploadDataRecord) {\n                    file = uploadDataRecord.file;\n                }\n            }\n\n            return file || null;\n        },\n\n        getInProgress: function() {\n            return this._uploadData.retrieve({\n                status: [\n                    qq.status.UPLOADING,\n                    qq.status.UPLOAD_RETRYING,\n                    qq.status.QUEUED\n                ]\n            }).length;\n        },\n\n        getName: function(id) {\n            return this._uploadData.retrieve({id: id}).name;\n        },\n\n        // Parent ID for a specific file, or null if this is the parent, or if it has no parent.\n        getParentId: function(id) {\n            var uploadDataEntry = this.getUploads({id: id}),\n                parentId = null;\n\n            if (uploadDataEntry) {\n                if (uploadDataEntry.parentId !== undefined) {\n                    parentId = uploadDataEntry.parentId;\n                }\n            }\n\n            return parentId;\n        },\n\n        getResumableFilesData: function() {\n            return this._handler.getResumableFilesData();\n        },\n\n        getSize: function(id) {\n            return this._uploadData.retrieve({id: id}).size;\n        },\n\n        getNetUploads: function() {\n            return this._netUploaded;\n        },\n\n        getRemainingAllowedItems: function() {\n            var allowedItems = this._currentItemLimit;\n\n            if (allowedItems > 0) {\n                return allowedItems - this._netUploadedOrQueued;\n            }\n\n            return null;\n        },\n\n        getUploads: function(optionalFilter) {\n            return this._uploadData.retrieve(optionalFilter);\n        },\n\n        getUuid: function(id) {\n            return this._uploadData.retrieve({id: id}).uuid;\n        },\n\n        isResumable: function(id) {\n            return this._handler.hasResumeRecord(id);\n        },\n\n        log: function(str, level) {\n            if (this._options.debug && (!level || level === \"info\")) {\n                qq.log(\"[Fine Uploader \" + qq.version + \"] \" + str);\n            }\n            else if (level && level !== \"info\") {\n                qq.log(\"[Fine Uploader \" + qq.version + \"] \" + str, level);\n\n            }\n        },\n\n        pauseUpload: function(id) {\n            var uploadData = this._uploadData.retrieve({id: id});\n\n            if (!qq.supportedFeatures.pause || !this._options.chunking.enabled) {\n                return false;\n            }\n\n            // Pause only really makes sense if the file is uploading or retrying\n            if (qq.indexOf([qq.status.UPLOADING, qq.status.UPLOAD_RETRYING], uploadData.status) >= 0) {\n                if (this._handler.pause(id)) {\n                    this._uploadData.setStatus(id, qq.status.PAUSED);\n                    return true;\n                }\n                else {\n                    this.log(qq.format(\"Unable to pause file ID {} ({}).\", id, this.getName(id)), \"error\");\n                }\n            }\n            else {\n                this.log(qq.format(\"Ignoring pause for file ID {} ({}).  Not in progress.\", id, this.getName(id)), \"error\");\n            }\n\n            return false;\n        },\n\n        removeFileRef: function(id) {\n            this._handler.expunge(id);\n            this._uploadData.removeFileRef(id);\n        },\n\n        reset: function() {\n            this.log(\"Resetting uploader...\");\n\n            this._handler.reset();\n            this._storedIds = [];\n            this._autoRetries = [];\n            this._retryTimeouts = [];\n            this._preventRetries = [];\n            this._thumbnailUrls = [];\n\n            qq.each(this._buttons, function(idx, button) {\n                button.reset();\n            });\n\n            this._paramsStore.reset();\n            this._endpointStore.reset();\n            this._netUploadedOrQueued = 0;\n            this._netUploaded = 0;\n            this._uploadData.reset();\n            this._buttonIdsForFileIds = [];\n\n            this._pasteHandler && this._pasteHandler.reset();\n            this._options.session.refreshOnReset && this._refreshSessionData();\n\n            this._succeededSinceLastAllComplete = [];\n            this._failedSinceLastAllComplete = [];\n\n            this._totalProgress && this._totalProgress.reset();\n\n            this._customResumeDataStore.reset();\n        },\n\n        retry: function(id) {\n            return this._manualRetry(id);\n        },\n\n        scaleImage: function(id, specs) {\n            var self = this;\n\n            return qq.Scaler.prototype.scaleImage(id, specs, {\n                log: qq.bind(self.log, self),\n                getFile: qq.bind(self.getFile, self),\n                uploadData: self._uploadData\n            });\n        },\n\n        setCustomHeaders: function(headers, id) {\n            this._customHeadersStore.set(headers, id);\n        },\n\n        setCustomResumeData: function(id, data) {\n            this._customResumeDataStore.set(data, id);\n        },\n\n        setDeleteFileCustomHeaders: function(headers, id) {\n            this._deleteFileCustomHeadersStore.set(headers, id);\n        },\n\n        setDeleteFileEndpoint: function(endpoint, id) {\n            this._deleteFileEndpointStore.set(endpoint, id);\n        },\n\n        setDeleteFileParams: function(params, id) {\n            this._deleteFileParamsStore.set(params, id);\n        },\n\n        // Re-sets the default endpoint, an endpoint for a specific file, or an endpoint for a specific button\n        setEndpoint: function(endpoint, id) {\n            this._endpointStore.set(endpoint, id);\n        },\n\n        setForm: function(elementOrId) {\n            this._updateFormSupportAndParams(elementOrId);\n        },\n\n        setItemLimit: function(newItemLimit) {\n            this._currentItemLimit = newItemLimit;\n        },\n\n        setName: function(id, newName) {\n            this._uploadData.updateName(id, newName);\n        },\n\n        setParams: function(params, id) {\n            this._paramsStore.set(params, id);\n        },\n\n        setUuid: function(id, newUuid) {\n            return this._uploadData.uuidChanged(id, newUuid);\n        },\n\n        /**\n         * Expose the internal status of a file id to the public api for manual state changes\n         * @public\n         *\n         * @param {Number} id,\n         * @param {String} newStatus\n         *\n         * @todo Implement the remaining methods\n         */\n        setStatus: function(id, newStatus) {\n            var fileRecord = this.getUploads({id: id});\n            if (!fileRecord) {\n                throw new qq.Error(id + \" is not a valid file ID.\");\n            }\n\n            switch (newStatus) {\n                case qq.status.DELETED:\n                    this._onDeleteComplete(id, null, false);\n                    break;\n                case qq.status.DELETE_FAILED:\n                    this._onDeleteComplete(id, null, true);\n                    break;\n                default:\n                    var errorMessage = \"Method setStatus called on '\" + name + \"' not implemented yet for \" + newStatus;\n                    this.log(errorMessage);\n                    throw new qq.Error(errorMessage);\n            }\n        },\n\n        uploadStoredFiles: function() {\n            if (this._storedIds.length === 0) {\n                this._itemError(\"noFilesError\");\n            }\n            else {\n                this._uploadStoredFiles();\n            }\n        }\n    };\n\n    /**\n     * Defines the private (internal) API for FineUploaderBasic mode.\n     */\n    qq.basePrivateApi = {\n        // Updates internal state with a file record (not backed by a live file).  Returns the assigned ID.\n        _addCannedFile: function(sessionData) {\n            var self = this;\n\n            return this._uploadData.addFile({\n                uuid: sessionData.uuid,\n                name: sessionData.name,\n                size: sessionData.size,\n                status: qq.status.UPLOAD_SUCCESSFUL,\n                onBeforeStatusChange: function(id) {\n                    sessionData.deleteFileEndpoint && self.setDeleteFileEndpoint(sessionData.deleteFileEndpoint, id);\n                    sessionData.deleteFileParams && self.setDeleteFileParams(sessionData.deleteFileParams, id);\n\n                    if (sessionData.thumbnailUrl) {\n                        self._thumbnailUrls[id] = sessionData.thumbnailUrl;\n                    }\n\n                    self._netUploaded++;\n                    self._netUploadedOrQueued++;\n                }\n            });\n        },\n\n        _annotateWithButtonId: function(file, associatedInput) {\n            if (qq.isFile(file)) {\n                file.qqButtonId = this._getButtonId(associatedInput);\n            }\n        },\n\n        _batchError: function(message) {\n            this._options.callbacks.onError(null, null, message, undefined);\n        },\n\n        _createDeleteHandler: function() {\n            var self = this;\n\n            return new qq.DeleteFileAjaxRequester({\n                method: this._options.deleteFile.method.toUpperCase(),\n                maxConnections: this._options.maxConnections,\n                uuidParamName: this._options.request.uuidName,\n                customHeaders: this._deleteFileCustomHeadersStore,\n                paramsStore: this._deleteFileParamsStore,\n                endpointStore: this._deleteFileEndpointStore,\n                cors: this._options.cors,\n                log: qq.bind(self.log, self),\n                onDelete: function(id) {\n                    self._onDelete(id);\n                    self._options.callbacks.onDelete(id);\n                },\n                onDeleteComplete: function(id, xhrOrXdr, isError) {\n                    self._onDeleteComplete(id, xhrOrXdr, isError);\n                    self._options.callbacks.onDeleteComplete(id, xhrOrXdr, isError);\n                }\n\n            });\n        },\n\n        _createPasteHandler: function() {\n            var self = this;\n\n            return new qq.PasteSupport({\n                targetElement: this._options.paste.targetElement,\n                callbacks: {\n                    log: qq.bind(self.log, self),\n                    pasteReceived: function(blob) {\n                        self._handleCheckedCallback({\n                            name: \"onPasteReceived\",\n                            callback: qq.bind(self._options.callbacks.onPasteReceived, self, blob),\n                            onSuccess: qq.bind(self._handlePasteSuccess, self, blob),\n                            identifier: \"pasted image\"\n                        });\n                    }\n                }\n            });\n        },\n\n        _createStore: function(initialValue, _readOnlyValues_) {\n            var store = {},\n                catchall = initialValue,\n                perIdReadOnlyValues = {},\n                readOnlyValues = _readOnlyValues_,\n                copy = function(orig) {\n                    if (qq.isObject(orig)) {\n                        return qq.extend({}, orig);\n                    }\n                    return orig;\n                },\n                getReadOnlyValues = function() {\n                    if (qq.isFunction(readOnlyValues)) {\n                        return readOnlyValues();\n                    }\n                    return readOnlyValues;\n                },\n                includeReadOnlyValues = function(id, existing) {\n                    if (readOnlyValues && qq.isObject(existing)) {\n                        qq.extend(existing, getReadOnlyValues());\n                    }\n\n                    if (perIdReadOnlyValues[id]) {\n                        qq.extend(existing, perIdReadOnlyValues[id]);\n                    }\n                };\n\n            return {\n                set: function(val, id) {\n                    /*jshint eqeqeq: true, eqnull: true*/\n                    if (id == null) {\n                        store = {};\n                        catchall = copy(val);\n                    }\n                    else {\n                        store[id] = copy(val);\n                    }\n                },\n\n                get: function(id) {\n                    var values;\n\n                    /*jshint eqeqeq: true, eqnull: true*/\n                    if (id != null && store[id]) {\n                        values = store[id];\n                    }\n                    else {\n                        values = copy(catchall);\n                    }\n\n                    includeReadOnlyValues(id, values);\n\n                    return copy(values);\n                },\n\n                addReadOnly: function(id, values) {\n                    // Only applicable to Object stores\n                    if (qq.isObject(store)) {\n                        // If null ID, apply readonly values to all files\n                        if (id === null) {\n                            if (qq.isFunction(values)) {\n                                readOnlyValues = values;\n                            }\n                            else {\n                                readOnlyValues = readOnlyValues || {};\n                                qq.extend(readOnlyValues, values);\n                            }\n                        }\n                        else {\n                            perIdReadOnlyValues[id] = perIdReadOnlyValues[id] || {};\n                            qq.extend(perIdReadOnlyValues[id], values);\n                        }\n                    }\n                },\n\n                remove: function(fileId) {\n                    return delete store[fileId];\n                },\n\n                reset: function() {\n                    store = {};\n                    perIdReadOnlyValues = {};\n                    catchall = initialValue;\n                }\n            };\n        },\n\n        _createUploadDataTracker: function() {\n            var self = this;\n\n            return new qq.UploadData({\n                getName: function(id) {\n                    return self.getName(id);\n                },\n                getUuid: function(id) {\n                    return self.getUuid(id);\n                },\n                getSize: function(id) {\n                    return self.getSize(id);\n                },\n                onStatusChange: function(id, oldStatus, newStatus) {\n                    self._onUploadStatusChange(id, oldStatus, newStatus);\n                    self._options.callbacks.onStatusChange(id, oldStatus, newStatus);\n                    self._maybeAllComplete(id, newStatus);\n\n                    if (self._totalProgress) {\n                        setTimeout(function() {\n                            self._totalProgress.onStatusChange(id, oldStatus, newStatus);\n                        }, 0);\n                    }\n                }\n            });\n        },\n\n        /**\n         * Generate a tracked upload button.\n         *\n         * @param spec Object containing a required `element` property\n         * along with optional `multiple`, `accept`, and `folders`.\n         * @returns {qq.UploadButton}\n         * @private\n         */\n        _createUploadButton: function(spec) {\n            var self = this,\n                acceptFiles = spec.accept || this._options.validation.acceptFiles,\n                allowedExtensions = spec.allowedExtensions || this._options.validation.allowedExtensions,\n                button;\n\n            function allowMultiple() {\n                if (qq.supportedFeatures.ajaxUploading) {\n                    // Workaround for bug in iOS7+ (see #1039)\n                    if (self._options.workarounds.iosEmptyVideos &&\n                        qq.ios() &&\n                        !qq.ios6() &&\n                        self._isAllowedExtension(allowedExtensions, \".mov\")) {\n\n                        return false;\n                    }\n\n                    if (spec.multiple === undefined) {\n                        return self._options.multiple;\n                    }\n\n                    return spec.multiple;\n                }\n\n                return false;\n            }\n\n            button = new qq.UploadButton({\n                acceptFiles: acceptFiles,\n                element: spec.element,\n                focusClass: this._options.classes.buttonFocus,\n                folders: spec.folders,\n                hoverClass: this._options.classes.buttonHover,\n                ios8BrowserCrashWorkaround: this._options.workarounds.ios8BrowserCrash,\n                multiple: allowMultiple(),\n                name: this._options.request.inputName,\n                onChange: function(input) {\n                    self._onInputChange(input);\n                },\n                title: spec.title == null ? this._options.text.fileInputTitle : spec.title\n            });\n\n            this._disposeSupport.addDisposer(function() {\n                button.dispose();\n            });\n\n            self._buttons.push(button);\n\n            return button;\n        },\n\n        _createUploadHandler: function(additionalOptions, namespace) {\n            var self = this,\n                lastOnProgress = {},\n                options = {\n                    debug: this._options.debug,\n                    maxConnections: this._options.maxConnections,\n                    cors: this._options.cors,\n                    paramsStore: this._paramsStore,\n                    endpointStore: this._endpointStore,\n                    chunking: this._options.chunking,\n                    resume: this._options.resume,\n                    blobs: this._options.blobs,\n                    log: qq.bind(self.log, self),\n                    preventRetryParam: this._options.retry.preventRetryResponseProperty,\n                    onProgress: function(id, name, loaded, total) {\n                        if (loaded < 0 || total < 0) {\n                            return;\n                        }\n\n                        if (lastOnProgress[id]) {\n                            if (lastOnProgress[id].loaded !== loaded || lastOnProgress[id].total !== total) {\n                                self._onProgress(id, name, loaded, total);\n                                self._options.callbacks.onProgress(id, name, loaded, total);\n                            }\n                        }\n                        else {\n                            self._onProgress(id, name, loaded, total);\n                            self._options.callbacks.onProgress(id, name, loaded, total);\n                        }\n\n                        lastOnProgress[id] = {loaded: loaded, total: total};\n\n                    },\n                    onComplete: function(id, name, result, xhr) {\n                        delete lastOnProgress[id];\n\n                        var status = self.getUploads({id: id}).status,\n                            retVal;\n\n                        // This is to deal with some observed cases where the XHR readyStateChange handler is\n                        // invoked by the browser multiple times for the same XHR instance with the same state\n                        // readyState value.  Higher level: don't invoke complete-related code if we've already\n                        // done this.\n                        if (status === qq.status.UPLOAD_SUCCESSFUL || status === qq.status.UPLOAD_FAILED) {\n                            return;\n                        }\n\n                        retVal = self._onComplete(id, name, result, xhr);\n\n                        // If the internal `_onComplete` handler returns a promise, don't invoke the `onComplete` callback\n                        // until the promise has been fulfilled.\n                        if (retVal instanceof  qq.Promise) {\n                            retVal.done(function() {\n                                self._options.callbacks.onComplete(id, name, result, xhr);\n                            });\n                        }\n                        else {\n                            self._options.callbacks.onComplete(id, name, result, xhr);\n                        }\n                    },\n                    onCancel: function(id, name, cancelFinalizationEffort) {\n                        var promise = new qq.Promise();\n\n                        self._handleCheckedCallback({\n                            name: \"onCancel\",\n                            callback: qq.bind(self._options.callbacks.onCancel, self, id, name),\n                            onFailure: promise.failure,\n                            onSuccess: function() {\n                                cancelFinalizationEffort.then(function() {\n                                    self._onCancel(id, name);\n                                });\n\n                                promise.success();\n                            },\n                            identifier: id\n                        });\n\n                        return promise;\n                    },\n                    onUploadPrep: qq.bind(this._onUploadPrep, this),\n                    onUpload: function(id, name) {\n                        self._onUpload(id, name);\n                        var onUploadResult = self._options.callbacks.onUpload(id, name);\n\n                        if (qq.isGenericPromise(onUploadResult)) {\n                            self.log(qq.format(\"onUpload for {} returned a Promise - waiting for resolution.\", id));\n                            return onUploadResult;\n                        }\n\n                        return new qq.Promise().success();\n                    },\n                    onUploadChunk: function(id, name, chunkData) {\n                        self._onUploadChunk(id, chunkData);\n                        var onUploadChunkResult = self._options.callbacks.onUploadChunk(id, name, chunkData);\n\n                        if (qq.isGenericPromise(onUploadChunkResult)) {\n                            self.log(qq.format(\"onUploadChunk for {}.{} returned a Promise - waiting for resolution.\", id, chunkData.partIndex));\n                            return onUploadChunkResult;\n                        }\n\n                        return new qq.Promise().success();\n                    },\n                    onUploadChunkSuccess: function(id, chunkData, result, xhr) {\n                        self._onUploadChunkSuccess(id, chunkData);\n                        self._options.callbacks.onUploadChunkSuccess.apply(self, arguments);\n                    },\n                    onResume: function(id, name, chunkData, customResumeData) {\n                        return self._options.callbacks.onResume(id, name, chunkData, customResumeData);\n                    },\n                    onAutoRetry: function(id, name, responseJSON, xhr) {\n                        return self._onAutoRetry.apply(self, arguments);\n                    },\n                    onUuidChanged: function(id, newUuid) {\n                        self.log(\"Server requested UUID change from '\" + self.getUuid(id) + \"' to '\" + newUuid + \"'\");\n                        self.setUuid(id, newUuid);\n                    },\n                    getName: qq.bind(self.getName, self),\n                    getUuid: qq.bind(self.getUuid, self),\n                    getSize: qq.bind(self.getSize, self),\n                    setSize: qq.bind(self._setSize, self),\n                    getDataByUuid: function(uuid) {\n                        return self.getUploads({uuid: uuid});\n                    },\n                    isQueued: function(id) {\n                        var status = self.getUploads({id: id}).status;\n                        return status === qq.status.QUEUED ||\n                            status === qq.status.SUBMITTED ||\n                            status === qq.status.UPLOAD_RETRYING ||\n                            status === qq.status.PAUSED;\n                    },\n                    getIdsInProxyGroup: self._uploadData.getIdsInProxyGroup,\n                    getIdsInBatch: self._uploadData.getIdsInBatch,\n                    isInProgress: function(id) {\n                        return self.getUploads({id: id}).status === qq.status.UPLOADING;\n                    },\n                    getCustomResumeData: qq.bind(self._getCustomResumeData, self),\n                    setStatus: function(id, status) {\n                        self._uploadData.setStatus(id, status);\n                    }\n                };\n\n            qq.each(this._options.request, function(prop, val) {\n                options[prop] = val;\n            });\n\n            options.customHeaders = this._customHeadersStore;\n\n            if (additionalOptions) {\n                qq.each(additionalOptions, function(key, val) {\n                    options[key] = val;\n                });\n            }\n\n            return new qq.UploadHandlerController(options, namespace);\n        },\n\n        _fileOrBlobRejected: function(id) {\n            this._netUploadedOrQueued--;\n            this._uploadData.setStatus(id, qq.status.REJECTED);\n        },\n\n        _formatSize: function(bytes) {\n            if (bytes === 0) {\n                return bytes + this._options.text.sizeSymbols[0];\n            }\n            var i = -1;\n            do {\n                bytes = bytes / 1000;\n                i++;\n            } while (bytes > 999);\n\n            return Math.max(bytes, 0.1).toFixed(1) + this._options.text.sizeSymbols[i];\n        },\n\n        // Creates an internal object that tracks various properties of each extra button,\n        // and then actually creates the extra button.\n        _generateExtraButtonSpecs: function() {\n            var self = this;\n\n            this._extraButtonSpecs = {};\n\n            qq.each(this._options.extraButtons, function(idx, extraButtonOptionEntry) {\n                var multiple = extraButtonOptionEntry.multiple,\n                    validation = qq.extend({}, self._options.validation, true),\n                    extraButtonSpec = qq.extend({}, extraButtonOptionEntry);\n\n                if (multiple === undefined) {\n                    multiple = self._options.multiple;\n                }\n\n                if (extraButtonSpec.validation) {\n                    qq.extend(validation, extraButtonOptionEntry.validation, true);\n                }\n\n                qq.extend(extraButtonSpec, {\n                    multiple: multiple,\n                    validation: validation\n                }, true);\n\n                self._initExtraButton(extraButtonSpec);\n            });\n        },\n\n        _getButton: function(buttonId) {\n            var extraButtonsSpec = this._extraButtonSpecs[buttonId];\n\n            if (extraButtonsSpec) {\n                return extraButtonsSpec.element;\n            }\n            else if (buttonId === this._defaultButtonId) {\n                return this._options.button;\n            }\n        },\n\n        /**\n         * Gets the internally used tracking ID for a button.\n         *\n         * @param buttonOrFileInputOrFile `File`, `<input type=\"file\">`, or a button container element\n         * @returns {*} The button's ID, or undefined if no ID is recoverable\n         * @private\n         */\n        _getButtonId: function(buttonOrFileInputOrFile) {\n            var inputs, fileInput,\n                fileBlobOrInput = buttonOrFileInputOrFile;\n\n            // We want the reference file/blob here if this is a proxy (a file that will be generated on-demand later)\n            if (fileBlobOrInput instanceof qq.BlobProxy) {\n                fileBlobOrInput = fileBlobOrInput.referenceBlob;\n            }\n\n            // If the item is a `Blob` it will never be associated with a button or drop zone.\n            if (fileBlobOrInput && !qq.isBlob(fileBlobOrInput)) {\n                if (qq.isFile(fileBlobOrInput)) {\n                    return fileBlobOrInput.qqButtonId;\n                }\n                else if (fileBlobOrInput.tagName.toLowerCase() === \"input\" &&\n                    fileBlobOrInput.type.toLowerCase() === \"file\") {\n\n                    return fileBlobOrInput.getAttribute(qq.UploadButton.BUTTON_ID_ATTR_NAME);\n                }\n\n                inputs = fileBlobOrInput.getElementsByTagName(\"input\");\n\n                qq.each(inputs, function(idx, input) {\n                    if (input.getAttribute(\"type\") === \"file\") {\n                        fileInput = input;\n                        return false;\n                    }\n                });\n\n                if (fileInput) {\n                    return fileInput.getAttribute(qq.UploadButton.BUTTON_ID_ATTR_NAME);\n                }\n            }\n        },\n\n        _getCustomResumeData: function(fileId) {\n            return this._customResumeDataStore.get(fileId);\n        },\n\n        _getNotFinished: function() {\n            return this._uploadData.retrieve({\n                status: [\n                    qq.status.UPLOADING,\n                    qq.status.UPLOAD_RETRYING,\n                    qq.status.QUEUED,\n                    qq.status.SUBMITTING,\n                    qq.status.SUBMITTED,\n                    qq.status.PAUSED\n                ]\n            }).length;\n        },\n\n        // Get the validation options for this button.  Could be the default validation option\n        // or a specific one assigned to this particular button.\n        _getValidationBase: function(buttonId) {\n            var extraButtonSpec = this._extraButtonSpecs[buttonId];\n\n            return extraButtonSpec ? extraButtonSpec.validation : this._options.validation;\n        },\n\n        _getValidationDescriptor: function(fileWrapper) {\n            if (fileWrapper.file instanceof qq.BlobProxy) {\n                return {\n                    name: qq.getFilename(fileWrapper.file.referenceBlob),\n                    size: fileWrapper.file.referenceBlob.size\n                };\n            }\n\n            return {\n                name: this.getUploads({id: fileWrapper.id}).name,\n                size: this.getUploads({id: fileWrapper.id}).size\n            };\n        },\n\n        _getValidationDescriptors: function(fileWrappers) {\n            var self = this,\n                fileDescriptors = [];\n\n            qq.each(fileWrappers, function(idx, fileWrapper) {\n                fileDescriptors.push(self._getValidationDescriptor(fileWrapper));\n            });\n\n            return fileDescriptors;\n        },\n\n        // Allows camera access on either the default or an extra button for iOS devices.\n        _handleCameraAccess: function() {\n            if (this._options.camera.ios && qq.ios()) {\n                var acceptIosCamera = \"image/*;capture=camera\",\n                    button = this._options.camera.button,\n                    buttonId = button ? this._getButtonId(button) : this._defaultButtonId,\n                    optionRoot = this._options;\n\n                // If we are not targeting the default button, it is an \"extra\" button\n                if (buttonId && buttonId !== this._defaultButtonId) {\n                    optionRoot = this._extraButtonSpecs[buttonId];\n                }\n\n                // Camera access won't work in iOS if the `multiple` attribute is present on the file input\n                optionRoot.multiple = false;\n\n                // update the options\n                if (optionRoot.validation.acceptFiles === null) {\n                    optionRoot.validation.acceptFiles = acceptIosCamera;\n                }\n                else {\n                    optionRoot.validation.acceptFiles += \",\" + acceptIosCamera;\n                }\n\n                // update the already-created button\n                qq.each(this._buttons, function(idx, button) {\n                    if (button.getButtonId() === buttonId) {\n                        button.setMultiple(optionRoot.multiple);\n                        button.setAcceptFiles(optionRoot.acceptFiles);\n\n                        return false;\n                    }\n                });\n            }\n        },\n\n        _handleCheckedCallback: function(details) {\n            var self = this,\n                callbackRetVal = details.callback();\n\n            if (qq.isGenericPromise(callbackRetVal)) {\n                this.log(details.name + \" - waiting for \" + details.name + \" promise to be fulfilled for \" + details.identifier);\n                return callbackRetVal.then(\n                    function(successParam) {\n                        self.log(details.name + \" promise success for \" + details.identifier);\n                        details.onSuccess(successParam);\n                    },\n                    function() {\n                        if (details.onFailure) {\n                            self.log(details.name + \" promise failure for \" + details.identifier);\n                            details.onFailure();\n                        }\n                        else {\n                            self.log(details.name + \" promise failure for \" + details.identifier);\n                        }\n                    });\n            }\n\n            if (callbackRetVal !== false) {\n                details.onSuccess(callbackRetVal);\n            }\n            else {\n                if (details.onFailure) {\n                    this.log(details.name + \" - return value was 'false' for \" + details.identifier + \".  Invoking failure callback.\");\n                    details.onFailure();\n                }\n                else {\n                    this.log(details.name + \" - return value was 'false' for \" + details.identifier + \".  Will not proceed.\");\n                }\n            }\n\n            return callbackRetVal;\n        },\n\n        // Updates internal state when a new file has been received, and adds it along with its ID to a passed array.\n        _handleNewFile: function(file, batchId, newFileWrapperList) {\n            var self = this,\n                uuid = qq.getUniqueId(),\n                size = -1,\n                name = qq.getFilename(file),\n                actualFile = file.blob || file,\n                handler = this._customNewFileHandler ?\n                    this._customNewFileHandler :\n                    qq.bind(self._handleNewFileGeneric, self);\n\n            if (!qq.isInput(actualFile) && actualFile.size >= 0) {\n                size = actualFile.size;\n            }\n\n            handler(actualFile, name, uuid, size, newFileWrapperList, batchId, this._options.request.uuidName, {\n                uploadData: self._uploadData,\n                paramsStore: self._paramsStore,\n                addFileToHandler: function(id, file) {\n                    self._handler.add(id, file);\n                    self._netUploadedOrQueued++;\n                    self._trackButton(id);\n                }\n            });\n        },\n\n        _handleNewFileGeneric: function(file, name, uuid, size, fileList, batchId) {\n            var id = this._uploadData.addFile({\n                uuid: uuid,\n                name: name,\n                size: size,\n                batchId: batchId,\n                file: file\n            });\n\n            this._handler.add(id, file);\n\n            this._trackButton(id);\n\n            this._netUploadedOrQueued++;\n\n            fileList.push({id: id, file: file});\n        },\n\n        _handlePasteSuccess: function(blob, extSuppliedName) {\n            var extension = blob.type.split(\"/\")[1],\n                name = extSuppliedName;\n\n            /*jshint eqeqeq: true, eqnull: true*/\n            if (name == null) {\n                name = this._options.paste.defaultName;\n            }\n\n            name += \".\" + extension;\n\n            this.addFiles({\n                name: name,\n                blob: blob\n            });\n        },\n\n        _handleDeleteSuccess: function(id) {\n            if (this.getUploads({id: id}).status !== qq.status.DELETED) {\n                var name = this.getName(id);\n\n                this._netUploadedOrQueued--;\n                this._netUploaded--;\n                this._handler.expunge(id);\n                this._uploadData.setStatus(id, qq.status.DELETED);\n                this.log(\"Delete request for '\" + name + \"' has succeeded.\");\n            }\n        },\n\n        _handleDeleteFailed: function(id, xhrOrXdr) {\n            var name = this.getName(id);\n\n            this._uploadData.setStatus(id, qq.status.DELETE_FAILED);\n            this.log(\"Delete request for '\" + name + \"' has failed.\", \"error\");\n\n            // Check first if xhrOrXdr is actually passed or valid\n            // For error reporting, we only have access to the response status if this is not\n            // an `XDomainRequest`.\n            if (!xhrOrXdr || xhrOrXdr.withCredentials === undefined) {\n                this._options.callbacks.onError(id, name, \"Delete request failed\", xhrOrXdr);\n            }\n            else {\n                this._options.callbacks.onError(id, name, \"Delete request failed with response code \" + xhrOrXdr.status, xhrOrXdr);\n            }\n        },\n\n        // Creates an extra button element\n        _initExtraButton: function(spec) {\n            var button = this._createUploadButton({\n                accept: spec.validation.acceptFiles,\n                allowedExtensions: spec.validation.allowedExtensions,\n                element: spec.element,\n                folders: spec.folders,\n                multiple: spec.multiple,\n                title: spec.fileInputTitle\n            });\n\n            this._extraButtonSpecs[button.getButtonId()] = spec;\n        },\n\n        _initFormSupportAndParams: function() {\n            this._formSupport = qq.FormSupport && new qq.FormSupport(\n                this._options.form, qq.bind(this.uploadStoredFiles, this), qq.bind(this.log, this)\n            );\n\n            if (this._formSupport && this._formSupport.attachedToForm) {\n                this._paramsStore = this._createStore(\n                    this._options.request.params,  this._formSupport.getFormInputsAsObject\n                );\n\n                this._options.autoUpload = this._formSupport.newAutoUpload;\n                if (this._formSupport.newEndpoint) {\n                    this._options.request.endpoint = this._formSupport.newEndpoint;\n                }\n            }\n            else {\n                this._paramsStore = this._createStore(this._options.request.params);\n            }\n        },\n\n        _isDeletePossible: function() {\n            if (!qq.DeleteFileAjaxRequester || !this._options.deleteFile.enabled) {\n                return false;\n            }\n\n            if (this._options.cors.expected) {\n                if (qq.supportedFeatures.deleteFileCorsXhr) {\n                    return true;\n                }\n\n                if (qq.supportedFeatures.deleteFileCorsXdr && this._options.cors.allowXdr) {\n                    return true;\n                }\n\n                return false;\n            }\n\n            return true;\n        },\n\n        _isAllowedExtension: function(allowed, fileName) {\n            var valid = false;\n\n            if (!allowed.length) {\n                return true;\n            }\n\n            qq.each(allowed, function(idx, allowedExt) {\n                /**\n                 * If an argument is not a string, ignore it.  Added when a possible issue with MooTools hijacking the\n                 * `allowedExtensions` array was discovered.  See case #735 in the issue tracker for more details.\n                 */\n                if (qq.isString(allowedExt)) {\n                    /*jshint eqeqeq: true, eqnull: true*/\n                    var extRegex = new RegExp(\"\\\\.\" + allowedExt + \"$\", \"i\");\n\n                    if (fileName.match(extRegex) != null) {\n                        valid = true;\n                        return false;\n                    }\n                }\n            });\n\n            return valid;\n        },\n\n        /**\n         * Constructs and returns a message that describes an item/file error.  Also calls `onError` callback.\n         *\n         * @param code REQUIRED - a code that corresponds to a stock message describing this type of error\n         * @param maybeNameOrNames names of the items that have failed, if applicable\n         * @param item `File`, `Blob`, or `<input type=\"file\">`\n         * @private\n         */\n        _itemError: function(code, maybeNameOrNames, item) {\n            var message = this._options.messages[code],\n                allowedExtensions = [],\n                names = [].concat(maybeNameOrNames),\n                name = names[0],\n                buttonId = this._getButtonId(item),\n                validationBase = this._getValidationBase(buttonId),\n                extensionsForMessage, placeholderMatch;\n\n            function r(name, replacement) { message = message.replace(name, replacement); }\n\n            qq.each(validationBase.allowedExtensions, function(idx, allowedExtension) {\n                /**\n                 * If an argument is not a string, ignore it.  Added when a possible issue with MooTools hijacking the\n                 * `allowedExtensions` array was discovered.  See case #735 in the issue tracker for more details.\n                 */\n                if (qq.isString(allowedExtension)) {\n                    allowedExtensions.push(allowedExtension);\n                }\n            });\n\n            extensionsForMessage = allowedExtensions.join(\", \").toLowerCase();\n\n            r(\"{file}\", this._options.formatFileName(name));\n            r(\"{extensions}\", extensionsForMessage);\n            r(\"{sizeLimit}\", this._formatSize(validationBase.sizeLimit));\n            r(\"{minSizeLimit}\", this._formatSize(validationBase.minSizeLimit));\n\n            placeholderMatch = message.match(/(\\{\\w+\\})/g);\n            if (placeholderMatch !== null) {\n                qq.each(placeholderMatch, function(idx, placeholder) {\n                    r(placeholder, names[idx]);\n                });\n            }\n\n            this._options.callbacks.onError(null, name, message, undefined);\n\n            return message;\n        },\n\n        /**\n         * Conditionally orders a manual retry of a failed upload.\n         *\n         * @param id File ID of the failed upload\n         * @param callback Optional callback to invoke if a retry is prudent.\n         * In lieu of asking the upload handler to retry.\n         * @returns {boolean} true if a manual retry will occur\n         * @private\n         */\n        _manualRetry: function(id, callback) {\n            if (this._onBeforeManualRetry(id)) {\n                this._netUploadedOrQueued++;\n                this._uploadData.setStatus(id, qq.status.UPLOAD_RETRYING);\n\n                if (callback) {\n                    callback(id);\n                }\n                else {\n                    this._handler.retry(id);\n                }\n\n                return true;\n            }\n        },\n\n        _maybeAllComplete: function(id, status) {\n            var self = this,\n                notFinished = this._getNotFinished();\n\n            if (status === qq.status.UPLOAD_SUCCESSFUL) {\n                this._succeededSinceLastAllComplete.push(id);\n            }\n            else if (status === qq.status.UPLOAD_FAILED) {\n                this._failedSinceLastAllComplete.push(id);\n            }\n\n            if (notFinished === 0 &&\n                (this._succeededSinceLastAllComplete.length || this._failedSinceLastAllComplete.length)) {\n                // Attempt to ensure onAllComplete is not invoked before other callbacks, such as onCancel & onComplete\n                setTimeout(function() {\n                    self._onAllComplete(self._succeededSinceLastAllComplete, self._failedSinceLastAllComplete);\n                }, 0);\n            }\n        },\n\n        _maybeHandleIos8SafariWorkaround: function() {\n            var self = this;\n\n            if (this._options.workarounds.ios8SafariUploads && qq.ios800() && qq.iosSafari()) {\n                setTimeout(function() {\n                    window.alert(self._options.messages.unsupportedBrowserIos8Safari);\n                }, 0);\n                throw new qq.Error(this._options.messages.unsupportedBrowserIos8Safari);\n            }\n        },\n\n        _maybeParseAndSendUploadError: function(id, name, response, xhr) {\n            // Assuming no one will actually set the response code to something other than 200\n            // and still set 'success' to true...\n            if (!response.success) {\n                if (xhr && xhr.status !== 200 && !response.error) {\n                    this._options.callbacks.onError(id, name, \"XHR returned response code \" + xhr.status, xhr);\n                }\n                else {\n                    var errorReason = response.error ? response.error : this._options.text.defaultResponseError;\n                    this._options.callbacks.onError(id, name, errorReason, xhr);\n                }\n            }\n        },\n\n        _maybeProcessNextItemAfterOnValidateCallback: function(validItem, items, index, params, endpoint) {\n            var self = this;\n\n            if (items.length > index) {\n                if (validItem || !this._options.validation.stopOnFirstInvalidFile) {\n                    //use setTimeout to prevent a stack overflow with a large number of files in the batch & non-promissory callbacks\n                    setTimeout(function() {\n                        var validationDescriptor = self._getValidationDescriptor(items[index]),\n                            buttonId = self._getButtonId(items[index].file),\n                            button = self._getButton(buttonId);\n\n                        self._handleCheckedCallback({\n                            name: \"onValidate\",\n                            callback: qq.bind(self._options.callbacks.onValidate, self, validationDescriptor, button),\n                            onSuccess: qq.bind(self._onValidateCallbackSuccess, self, items, index, params, endpoint),\n                            onFailure: qq.bind(self._onValidateCallbackFailure, self, items, index, params, endpoint),\n                            identifier: \"Item '\" + validationDescriptor.name + \"', size: \" + validationDescriptor.size\n                        });\n                    }, 0);\n                }\n                else if (!validItem) {\n                    for (; index < items.length; index++) {\n                        self._fileOrBlobRejected(items[index].id);\n                    }\n                }\n            }\n        },\n\n        _onAllComplete: function(successful, failed) {\n            this._totalProgress && this._totalProgress.onAllComplete(successful, failed, this._preventRetries);\n\n            this._options.callbacks.onAllComplete(qq.extend([], successful), qq.extend([], failed));\n\n            this._succeededSinceLastAllComplete = [];\n            this._failedSinceLastAllComplete = [];\n        },\n\n        /**\n         * Attempt to automatically retry a failed upload.\n         *\n         * @param id The file ID of the failed upload\n         * @param name The name of the file associated with the failed upload\n         * @param responseJSON Response from the server, parsed into a javascript object\n         * @param xhr Ajax transport used to send the failed request\n         * @param callback Optional callback to be invoked if a retry is prudent.\n         * Invoked in lieu of asking the upload handler to retry.\n         * @returns {boolean} true if an auto-retry will occur\n         * @private\n         */\n        _onAutoRetry: function(id, name, responseJSON, xhr, callback) {\n            var self = this;\n\n            self._preventRetries[id] = responseJSON[self._options.retry.preventRetryResponseProperty];\n\n            if (self._shouldAutoRetry(id)) {\n                var retryWaitPeriod = self._options.retry.autoAttemptDelay * 1000;\n\n                self._maybeParseAndSendUploadError.apply(self, arguments);\n                self._options.callbacks.onAutoRetry(id, name, self._autoRetries[id]);\n                self._onBeforeAutoRetry(id, name);\n\n                self._uploadData.setStatus(id, qq.status.UPLOAD_RETRYING);\n                self._retryTimeouts[id] = setTimeout(function() {\n                    self.log(\"Starting retry for \" + name + \"...\");\n\n                    if (callback) {\n                        callback(id);\n                    }\n                    else {\n                        self._handler.retry(id);\n                    }\n                }, retryWaitPeriod);\n\n                return true;\n            }\n        },\n\n        _onBeforeAutoRetry: function(id, name) {\n            this.log(\"Waiting \" + this._options.retry.autoAttemptDelay + \" seconds before retrying \" + name + \"...\");\n        },\n\n        //return false if we should not attempt the requested retry\n        _onBeforeManualRetry: function(id) {\n            var itemLimit = this._currentItemLimit,\n                fileName;\n\n            if (this._preventRetries[id]) {\n                this.log(\"Retries are forbidden for id \" + id, \"warn\");\n                return false;\n            }\n            else if (this._handler.isValid(id)) {\n                fileName = this.getName(id);\n\n                if (this._options.callbacks.onManualRetry(id, fileName) === false) {\n                    return false;\n                }\n\n                if (itemLimit > 0 && this._netUploadedOrQueued + 1 > itemLimit) {\n                    this._itemError(\"retryFailTooManyItems\");\n                    return false;\n                }\n\n                this.log(\"Retrying upload for '\" + fileName + \"' (id: \" + id + \")...\");\n                return true;\n            }\n            else {\n                this.log(\"'\" + id + \"' is not a valid file ID\", \"error\");\n                return false;\n            }\n        },\n\n        _onCancel: function(id, name) {\n            this._netUploadedOrQueued--;\n\n            clearTimeout(this._retryTimeouts[id]);\n\n            var storedItemIndex = qq.indexOf(this._storedIds, id);\n            if (!this._options.autoUpload && storedItemIndex >= 0) {\n                this._storedIds.splice(storedItemIndex, 1);\n            }\n\n            this._uploadData.setStatus(id, qq.status.CANCELED);\n        },\n\n        _onComplete: function(id, name, result, xhr) {\n            if (!result.success) {\n                this._netUploadedOrQueued--;\n                this._uploadData.setStatus(id, qq.status.UPLOAD_FAILED);\n\n                if (result[this._options.retry.preventRetryResponseProperty] === true) {\n                    this._preventRetries[id] = true;\n                }\n            }\n            else {\n                if (result.thumbnailUrl) {\n                    this._thumbnailUrls[id] = result.thumbnailUrl;\n                }\n\n                this._netUploaded++;\n                this._uploadData.setStatus(id, qq.status.UPLOAD_SUCCESSFUL);\n            }\n\n            this._maybeParseAndSendUploadError(id, name, result, xhr);\n\n            return result.success ? true : false;\n        },\n\n        _onDelete: function(id) {\n            this._uploadData.setStatus(id, qq.status.DELETING);\n        },\n\n        _onDeleteComplete: function(id, xhrOrXdr, isError) {\n            var name = this.getName(id);\n\n            if (isError) {\n                this._handleDeleteFailed(id, xhrOrXdr);\n            }\n            else {\n                this._handleDeleteSuccess(id);\n            }\n        },\n\n        _onInputChange: function(input) {\n            var fileIndex;\n\n            if (qq.supportedFeatures.ajaxUploading) {\n                for (fileIndex = 0; fileIndex < input.files.length; fileIndex++) {\n                    this._annotateWithButtonId(input.files[fileIndex], input);\n                }\n\n                this.addFiles(input.files);\n            }\n            // Android 2.3.x will fire `onchange` even if no file has been selected\n            else if (input.value.length > 0) {\n                this.addFiles(input);\n            }\n\n            qq.each(this._buttons, function(idx, button) {\n                button.reset();\n            });\n        },\n\n        _onProgress: function(id, name, loaded, total) {\n            this._totalProgress && this._totalProgress.onIndividualProgress(id, loaded, total);\n        },\n\n        _onSubmit: function(id, name) {\n            //nothing to do yet in core uploader\n        },\n\n        _onSubmitCallbackSuccess: function(id, name) {\n            this._onSubmit.apply(this, arguments);\n            this._uploadData.setStatus(id, qq.status.SUBMITTED);\n            this._onSubmitted.apply(this, arguments);\n\n            if (this._options.autoUpload) {\n                this._options.callbacks.onSubmitted.apply(this, arguments);\n                this._uploadFile(id);\n            }\n            else {\n                this._storeForLater(id);\n                this._options.callbacks.onSubmitted.apply(this, arguments);\n            }\n        },\n\n        _onSubmitDelete: function(id, onSuccessCallback, additionalMandatedParams) {\n            var uuid = this.getUuid(id),\n                adjustedOnSuccessCallback;\n\n            if (onSuccessCallback) {\n                adjustedOnSuccessCallback = qq.bind(onSuccessCallback, this, id, uuid, additionalMandatedParams);\n            }\n\n            if (this._isDeletePossible()) {\n                this._handleCheckedCallback({\n                    name: \"onSubmitDelete\",\n                    callback: qq.bind(this._options.callbacks.onSubmitDelete, this, id),\n                    onSuccess: adjustedOnSuccessCallback ||\n                        qq.bind(this._deleteHandler.sendDelete, this, id, uuid, additionalMandatedParams),\n                    identifier: id\n                });\n                return true;\n            }\n            else {\n                this.log(\"Delete request ignored for ID \" + id + \", delete feature is disabled or request not possible \" +\n                    \"due to CORS on a user agent that does not support pre-flighting.\", \"warn\");\n                return false;\n            }\n        },\n\n        _onSubmitted: function(id) {\n            //nothing to do in the base uploader\n        },\n\n        _onTotalProgress: function(loaded, total) {\n            this._options.callbacks.onTotalProgress(loaded, total);\n        },\n\n        _onUploadPrep: function(id) {\n            // nothing to do in the core uploader for now\n        },\n\n        _onUpload: function(id, name) {\n            this._uploadData.setStatus(id, qq.status.UPLOADING);\n        },\n\n        _onUploadChunk: function(id, chunkData) {\n            //nothing to do in the base uploader\n        },\n\n        _onUploadChunkSuccess: function(id, chunkData) {\n            if (!this._preventRetries[id] && this._options.retry.enableAuto) {\n                this._autoRetries[id] = 0;\n            }\n        },\n\n        _onUploadStatusChange: function(id, oldStatus, newStatus) {\n            // Make sure a \"queued\" retry attempt is canceled if the upload has been paused\n            if (newStatus === qq.status.PAUSED) {\n                clearTimeout(this._retryTimeouts[id]);\n            }\n        },\n\n        _onValidateBatchCallbackFailure: function(fileWrappers) {\n            var self = this;\n\n            qq.each(fileWrappers, function(idx, fileWrapper) {\n                self._fileOrBlobRejected(fileWrapper.id);\n            });\n        },\n\n        _onValidateBatchCallbackSuccess: function(validationDescriptors, items, params, endpoint, button) {\n            var errorMessage,\n                itemLimit = this._currentItemLimit,\n                proposedNetFilesUploadedOrQueued = this._netUploadedOrQueued;\n\n            if (itemLimit === 0 || proposedNetFilesUploadedOrQueued <= itemLimit) {\n                if (items.length > 0) {\n                    this._handleCheckedCallback({\n                        name: \"onValidate\",\n                        callback: qq.bind(this._options.callbacks.onValidate, this, validationDescriptors[0], button),\n                        onSuccess: qq.bind(this._onValidateCallbackSuccess, this, items, 0, params, endpoint),\n                        onFailure: qq.bind(this._onValidateCallbackFailure, this, items, 0, params, endpoint),\n                        identifier: \"Item '\" + items[0].file.name + \"', size: \" + items[0].file.size\n                    });\n                }\n                else {\n                    this._itemError(\"noFilesError\");\n                }\n            }\n            else {\n                this._onValidateBatchCallbackFailure(items);\n                errorMessage = this._options.messages.tooManyItemsError\n                    .replace(/\\{netItems\\}/g, proposedNetFilesUploadedOrQueued)\n                    .replace(/\\{itemLimit\\}/g, itemLimit);\n                this._batchError(errorMessage);\n            }\n        },\n\n        _onValidateCallbackFailure: function(items, index, params, endpoint) {\n            var nextIndex = index + 1;\n\n            this._fileOrBlobRejected(items[index].id, items[index].file.name);\n\n            this._maybeProcessNextItemAfterOnValidateCallback(false, items, nextIndex, params, endpoint);\n        },\n\n        _onValidateCallbackSuccess: function(items, index, params, endpoint) {\n            var self = this,\n                nextIndex = index + 1,\n                validationDescriptor = this._getValidationDescriptor(items[index]);\n\n            this._validateFileOrBlobData(items[index], validationDescriptor)\n                .then(\n                function() {\n                    self._upload(items[index].id, params, endpoint);\n                    self._maybeProcessNextItemAfterOnValidateCallback(true, items, nextIndex, params, endpoint);\n                },\n                function() {\n                    self._maybeProcessNextItemAfterOnValidateCallback(false, items, nextIndex, params, endpoint);\n                }\n            );\n        },\n\n        _prepareItemsForUpload: function(items, params, endpoint) {\n            if (items.length === 0) {\n                this._itemError(\"noFilesError\");\n                return;\n            }\n\n            var validationDescriptors = this._getValidationDescriptors(items),\n                buttonId = this._getButtonId(items[0].file),\n                button = this._getButton(buttonId);\n\n            this._handleCheckedCallback({\n                name: \"onValidateBatch\",\n                callback: qq.bind(this._options.callbacks.onValidateBatch, this, validationDescriptors, button),\n                onSuccess: qq.bind(this._onValidateBatchCallbackSuccess, this, validationDescriptors, items, params, endpoint, button),\n                onFailure: qq.bind(this._onValidateBatchCallbackFailure, this, items),\n                identifier: \"batch validation\"\n            });\n        },\n\n        _preventLeaveInProgress: function() {\n            var self = this;\n\n            this._disposeSupport.attach(window, \"beforeunload\", function(e) {\n                if (self.getInProgress()) {\n                    e = e || window.event;\n                    // for ie, ff\n                    e.returnValue = self._options.messages.onLeave;\n                    // for webkit\n                    return self._options.messages.onLeave;\n                }\n            });\n        },\n\n        // Attempts to refresh session data only if the `qq.Session` module exists\n        // and a session endpoint has been specified.  The `onSessionRequestComplete`\n        // callback will be invoked once the refresh is complete.\n        _refreshSessionData: function() {\n            var self = this,\n                options = this._options.session;\n\n            /* jshint eqnull:true */\n            if (qq.Session && this._options.session.endpoint != null) {\n                if (!this._session) {\n                    qq.extend(options, {cors: this._options.cors});\n\n                    options.log = qq.bind(this.log, this);\n                    options.addFileRecord = qq.bind(this._addCannedFile, this);\n\n                    this._session = new qq.Session(options);\n                }\n\n                setTimeout(function() {\n                    self._session.refresh().then(function(response, xhrOrXdr) {\n                        self._sessionRequestComplete();\n                        self._options.callbacks.onSessionRequestComplete(response, true, xhrOrXdr);\n\n                    }, function(response, xhrOrXdr) {\n\n                        self._options.callbacks.onSessionRequestComplete(response, false, xhrOrXdr);\n                    });\n                }, 0);\n            }\n        },\n\n        _sessionRequestComplete: function() {},\n\n        _setSize: function(id, newSize) {\n            this._uploadData.updateSize(id, newSize);\n            this._totalProgress && this._totalProgress.onNewSize(id);\n        },\n\n        _shouldAutoRetry: function(id) {\n            var uploadData = this._uploadData.retrieve({id: id});\n\n            /*jshint laxbreak: true */\n            if (!this._preventRetries[id]\n                && this._options.retry.enableAuto\n                && uploadData.status !== qq.status.PAUSED) {\n\n                if (this._autoRetries[id] === undefined) {\n                    this._autoRetries[id] = 0;\n                }\n\n                if (this._autoRetries[id] < this._options.retry.maxAutoAttempts) {\n                    this._autoRetries[id] += 1;\n                    return true;\n                }\n            }\n\n            return false;\n        },\n\n        _storeForLater: function(id) {\n            this._storedIds.push(id);\n        },\n\n        // Maps a file with the button that was used to select it.\n        _trackButton: function(id) {\n            var buttonId;\n\n            if (qq.supportedFeatures.ajaxUploading) {\n                buttonId = this._handler.getFile(id).qqButtonId;\n            }\n            else {\n                buttonId = this._getButtonId(this._handler.getInput(id));\n            }\n\n            if (buttonId) {\n                this._buttonIdsForFileIds[id] = buttonId;\n            }\n        },\n\n        _updateFormSupportAndParams: function(formElementOrId) {\n            this._options.form.element = formElementOrId;\n\n            this._formSupport = qq.FormSupport && new qq.FormSupport(\n                    this._options.form, qq.bind(this.uploadStoredFiles, this), qq.bind(this.log, this)\n                );\n\n            if (this._formSupport && this._formSupport.attachedToForm) {\n                this._paramsStore.addReadOnly(null, this._formSupport.getFormInputsAsObject);\n\n                this._options.autoUpload = this._formSupport.newAutoUpload;\n                if (this._formSupport.newEndpoint) {\n                    this.setEndpoint(this._formSupport.newEndpoint);\n                }\n            }\n        },\n\n        _upload: function(id, params, endpoint) {\n            var name = this.getName(id);\n\n            if (params) {\n                this.setParams(params, id);\n            }\n\n            if (endpoint) {\n                this.setEndpoint(endpoint, id);\n            }\n\n            this._handleCheckedCallback({\n                name: \"onSubmit\",\n                callback: qq.bind(this._options.callbacks.onSubmit, this, id, name),\n                onSuccess: qq.bind(this._onSubmitCallbackSuccess, this, id, name),\n                onFailure: qq.bind(this._fileOrBlobRejected, this, id, name),\n                identifier: id\n            });\n        },\n\n        _uploadFile: function(id) {\n            if (!this._handler.upload(id)) {\n                this._uploadData.setStatus(id, qq.status.QUEUED);\n            }\n        },\n\n        _uploadStoredFiles: function() {\n            var idToUpload, stillSubmitting,\n                self = this;\n\n            while (this._storedIds.length) {\n                idToUpload = this._storedIds.shift();\n                this._uploadFile(idToUpload);\n            }\n\n            // If we are still waiting for some files to clear validation, attempt to upload these again in a bit\n            stillSubmitting = this.getUploads({status: qq.status.SUBMITTING}).length;\n            if (stillSubmitting) {\n                qq.log(\"Still waiting for \" + stillSubmitting + \" files to clear submit queue. Will re-parse stored IDs array shortly.\");\n                setTimeout(function() {\n                    self._uploadStoredFiles();\n                }, 1000);\n            }\n        },\n\n        /**\n         * Performs some internal validation checks on an item, defined in the `validation` option.\n         *\n         * @param fileWrapper Wrapper containing a `file` along with an `id`\n         * @param validationDescriptor Normalized information about the item (`size`, `name`).\n         * @returns qq.Promise with appropriate callbacks invoked depending on the validity of the file\n         * @private\n         */\n        _validateFileOrBlobData: function(fileWrapper, validationDescriptor) {\n            var self = this,\n                file = (function() {\n                    if (fileWrapper.file instanceof qq.BlobProxy) {\n                        return fileWrapper.file.referenceBlob;\n                    }\n                    return fileWrapper.file;\n                }()),\n                name = validationDescriptor.name,\n                size = validationDescriptor.size,\n                buttonId = this._getButtonId(fileWrapper.file),\n                validationBase = this._getValidationBase(buttonId),\n                validityChecker = new qq.Promise();\n\n            validityChecker.then(\n                function() {},\n                function() {\n                    self._fileOrBlobRejected(fileWrapper.id, name);\n                });\n\n            if (qq.isFileOrInput(file) && !this._isAllowedExtension(validationBase.allowedExtensions, name)) {\n                this._itemError(\"typeError\", name, file);\n                return validityChecker.failure();\n            }\n\n            if (!this._options.validation.allowEmpty && size === 0) {\n                this._itemError(\"emptyError\", name, file);\n                return validityChecker.failure();\n            }\n\n            if (size > 0 && validationBase.sizeLimit && size > validationBase.sizeLimit) {\n                this._itemError(\"sizeError\", name, file);\n                return validityChecker.failure();\n            }\n\n            if (size > 0 && size < validationBase.minSizeLimit) {\n                this._itemError(\"minSizeError\", name, file);\n                return validityChecker.failure();\n            }\n\n            if (qq.ImageValidation && qq.supportedFeatures.imagePreviews && qq.isFile(file)) {\n                new qq.ImageValidation(file, qq.bind(self.log, self)).validate(validationBase.image).then(\n                    validityChecker.success,\n                    function(errorCode) {\n                        self._itemError(errorCode + \"ImageError\", name, file);\n                        validityChecker.failure();\n                    }\n                );\n            }\n            else {\n                validityChecker.success();\n            }\n\n            return validityChecker;\n        },\n\n        _wrapCallbacks: function() {\n            var self, safeCallback, prop;\n\n            self = this;\n\n            safeCallback = function(name, callback, args) {\n                var errorMsg;\n\n                try {\n                    return callback.apply(self, args);\n                }\n                catch (exception) {\n                    errorMsg = exception.message || exception.toString();\n                    self.log(\"Caught exception in '\" + name + \"' callback - \" + errorMsg, \"error\");\n                }\n            };\n\n            /* jshint forin: false, loopfunc: true */\n            for (prop in this._options.callbacks) {\n                (function() {\n                    var callbackName, callbackFunc;\n                    callbackName = prop;\n                    callbackFunc = self._options.callbacks[callbackName];\n                    self._options.callbacks[callbackName] = function() {\n                        return safeCallback(callbackName, callbackFunc, arguments);\n                    };\n                }());\n            }\n        }\n    };\n}());\n","/*globals qq*/\n(function() {\n    \"use strict\";\n\n    qq.FineUploaderBasic = function(o) {\n        var self = this;\n\n        // These options define FineUploaderBasic mode.\n        this._options = {\n            debug: false,\n            button: null,\n            multiple: true,\n            maxConnections: 3,\n            disableCancelForFormUploads: false,\n            autoUpload: true,\n            warnBeforeUnload: true,\n\n            request: {\n                customHeaders: {},\n                endpoint: \"/server/upload\",\n                filenameParam: \"qqfilename\",\n                forceMultipart: true,\n                inputName: \"qqfile\",\n                method: \"POST\",\n                omitDefaultParams: false,\n                params: {},\n                paramsInBody: true,\n                requireSuccessJson: true,\n                totalFileSizeName: \"qqtotalfilesize\",\n                uuidName: \"qquuid\"\n            },\n\n            validation: {\n                allowedExtensions: [],\n                sizeLimit: 0,\n                minSizeLimit: 0,\n                itemLimit: 0,\n                stopOnFirstInvalidFile: true,\n                acceptFiles: null,\n                image: {\n                    maxHeight: 0,\n                    maxWidth: 0,\n                    minHeight: 0,\n                    minWidth: 0\n                },\n                allowEmpty: false\n            },\n\n            callbacks: {\n                onSubmit: function(id, name) {},\n                onSubmitted: function(id, name) {},\n                onComplete: function(id, name, responseJSON, maybeXhr) {},\n                onAllComplete: function(successful, failed) {},\n                onCancel: function(id, name) {},\n                onUpload: function(id, name) {},\n                onUploadChunk: function(id, name, chunkData) {},\n                onUploadChunkSuccess: function(id, chunkData, responseJSON, xhr) {},\n                onResume: function(id, fileName, chunkData, customResumeData) {},\n                onProgress: function(id, name, loaded, total) {},\n                onTotalProgress: function(loaded, total) {},\n                onError: function(id, name, reason, maybeXhrOrXdr) {},\n                onAutoRetry: function(id, name, attemptNumber) {},\n                onManualRetry: function(id, name) {},\n                onValidateBatch: function(fileOrBlobData) {},\n                onValidate: function(fileOrBlobData) {},\n                onSubmitDelete: function(id) {},\n                onDelete: function(id) {},\n                onDeleteComplete: function(id, xhrOrXdr, isError) {},\n                onPasteReceived: function(blob) {},\n                onStatusChange: function(id, oldStatus, newStatus) {},\n                onSessionRequestComplete: function(response, success, xhrOrXdr) {}\n            },\n\n            messages: {\n                typeError: \"{file} has an invalid extension. Valid extension(s): {extensions}.\",\n                sizeError: \"{file} is too large, maximum file size is {sizeLimit}.\",\n                minSizeError: \"{file} is too small, minimum file size is {minSizeLimit}.\",\n                emptyError: \"{file} is empty, please select files again without it.\",\n                noFilesError: \"No files to upload.\",\n                tooManyItemsError: \"Too many items ({netItems}) would be uploaded.  Item limit is {itemLimit}.\",\n                maxHeightImageError: \"Image is too tall.\",\n                maxWidthImageError: \"Image is too wide.\",\n                minHeightImageError: \"Image is not tall enough.\",\n                minWidthImageError: \"Image is not wide enough.\",\n                retryFailTooManyItems: \"Retry failed - you have reached your file limit.\",\n                onLeave: \"The files are being uploaded, if you leave now the upload will be canceled.\",\n                unsupportedBrowserIos8Safari: \"Unrecoverable error - this browser does not permit file uploading of any kind due to serious bugs in iOS8 Safari.  Please use iOS8 Chrome until Apple fixes these issues.\"\n            },\n\n            retry: {\n                enableAuto: false,\n                maxAutoAttempts: 3,\n                autoAttemptDelay: 5,\n                preventRetryResponseProperty: \"preventRetry\"\n            },\n\n            classes: {\n                buttonHover: \"qq-upload-button-hover\",\n                buttonFocus: \"qq-upload-button-focus\"\n            },\n\n            chunking: {\n                enabled: false,\n                concurrent: {\n                    enabled: false\n                },\n                mandatory: false,\n                paramNames: {\n                    partIndex: \"qqpartindex\",\n                    partByteOffset: \"qqpartbyteoffset\",\n                    chunkSize: \"qqchunksize\",\n                    totalFileSize: \"qqtotalfilesize\",\n                    totalParts: \"qqtotalparts\"\n                },\n                partSize: function(id) {\n                    return 2000000;\n                },\n                // only relevant for traditional endpoints, only required when concurrent.enabled === true\n                success: {\n                    endpoint: null,\n\n                    headers: function(id) {\n                        return null;\n                    },\n\n                    jsonPayload: false,\n\n                    method: \"POST\",\n\n                    params: function(id) {\n                        return null;\n                    },\n\n                    resetOnStatus: []\n                }\n            },\n\n            resume: {\n                enabled: false,\n                recordsExpireIn: 7, //days\n                paramNames: {\n                    resuming: \"qqresume\"\n                },\n                customKeys: function(fileId) {\n                    return [];\n                }\n            },\n\n            formatFileName: function(fileOrBlobName) {\n                return fileOrBlobName;\n            },\n\n            text: {\n                defaultResponseError: \"Upload failure reason unknown\",\n                fileInputTitle: \"file input\",\n                sizeSymbols: [\"kB\", \"MB\", \"GB\", \"TB\", \"PB\", \"EB\"]\n            },\n\n            deleteFile: {\n                enabled: false,\n                method: \"DELETE\",\n                endpoint: \"/server/upload\",\n                customHeaders: {},\n                params: {}\n            },\n\n            cors: {\n                expected: false,\n                sendCredentials: false,\n                allowXdr: false\n            },\n\n            blobs: {\n                defaultName: \"misc_data\"\n            },\n\n            paste: {\n                targetElement: null,\n                defaultName: \"pasted_image\"\n            },\n\n            camera: {\n                ios: false,\n\n                // if ios is true: button is null means target the default button, otherwise target the button specified\n                button: null\n            },\n\n            // This refers to additional upload buttons to be handled by Fine Uploader.\n            // Each element is an object, containing `element` as the only required\n            // property.  The `element` must be a container that will ultimately\n            // contain an invisible `<input type=\"file\">` created by Fine Uploader.\n            // Optional properties of each object include `multiple`, `validation`,\n            // and `folders`.\n            extraButtons: [],\n\n            // Depends on the session module.  Used to query the server for an initial file list\n            // during initialization and optionally after a `reset`.\n            session: {\n                endpoint: null,\n                params: {},\n                customHeaders: {},\n                refreshOnReset: true\n            },\n\n            // Send parameters associated with an existing form along with the files\n            form: {\n                // Element ID, HTMLElement, or null\n                element: \"qq-form\",\n\n                // Overrides the base `autoUpload`, unless `element` is null.\n                autoUpload: false,\n\n                // true = upload files on form submission (and squelch submit event)\n                interceptSubmit: true\n            },\n\n            // scale images client side, upload a new file for each scaled version\n            scaling: {\n                customResizer: null,\n\n                // send the original file as well\n                sendOriginal: true,\n\n                // fox orientation for scaled images\n                orient: true,\n\n                // If null, scaled image type will match reference image type.  This value will be referred to\n                // for any size record that does not specific a type.\n                defaultType: null,\n\n                defaultQuality: 80,\n\n                failureText: \"Failed to scale\",\n\n                includeExif: false,\n\n                // metadata about each requested scaled version\n                sizes: []\n            },\n\n            workarounds: {\n                iosEmptyVideos: true,\n                ios8SafariUploads: true,\n                ios8BrowserCrash: false\n            }\n        };\n\n        // Replace any default options with user defined ones\n        qq.extend(this._options, o, true);\n\n        this._buttons = [];\n        this._extraButtonSpecs = {};\n        this._buttonIdsForFileIds = [];\n\n        this._wrapCallbacks();\n        this._disposeSupport =  new qq.DisposeSupport();\n\n        this._storedIds = [];\n        this._autoRetries = [];\n        this._retryTimeouts = [];\n        this._preventRetries = [];\n        this._thumbnailUrls = [];\n\n        this._netUploadedOrQueued = 0;\n        this._netUploaded = 0;\n        this._uploadData = this._createUploadDataTracker();\n\n        this._initFormSupportAndParams();\n\n        this._customHeadersStore = this._createStore(this._options.request.customHeaders);\n        this._deleteFileCustomHeadersStore = this._createStore(this._options.deleteFile.customHeaders);\n\n        this._deleteFileParamsStore = this._createStore(this._options.deleteFile.params);\n\n        this._endpointStore = this._createStore(this._options.request.endpoint);\n        this._deleteFileEndpointStore = this._createStore(this._options.deleteFile.endpoint);\n\n        this._handler = this._createUploadHandler();\n\n        this._deleteHandler = qq.DeleteFileAjaxRequester && this._createDeleteHandler();\n\n        if (this._options.button) {\n            this._defaultButtonId = this._createUploadButton({\n                element: this._options.button,\n                title: this._options.text.fileInputTitle\n            }).getButtonId();\n        }\n\n        this._generateExtraButtonSpecs();\n\n        this._handleCameraAccess();\n\n        if (this._options.paste.targetElement) {\n            if (qq.PasteSupport) {\n                this._pasteHandler = this._createPasteHandler();\n            }\n            else {\n                this.log(\"Paste support module not found\", \"error\");\n            }\n        }\n\n        this._options.warnBeforeUnload && this._preventLeaveInProgress();\n\n        this._imageGenerator = qq.ImageGenerator && new qq.ImageGenerator(qq.bind(this.log, this));\n        this._refreshSessionData();\n\n        this._succeededSinceLastAllComplete = [];\n        this._failedSinceLastAllComplete = [];\n\n        this._scaler = (qq.Scaler && new qq.Scaler(this._options.scaling, qq.bind(this.log, this))) || {};\n        if (this._scaler.enabled) {\n            this._customNewFileHandler = qq.bind(this._scaler.handleNewFile, this._scaler);\n        }\n\n        if (qq.TotalProgress && qq.supportedFeatures.progressBar) {\n            this._totalProgress = new qq.TotalProgress(\n                qq.bind(this._onTotalProgress, this),\n\n                function(id) {\n                    var entry = self._uploadData.retrieve({id: id});\n                    return (entry && entry.size) || 0;\n                }\n            );\n        }\n\n        this._currentItemLimit = this._options.validation.itemLimit;\n\n        this._customResumeDataStore = this._createStore();\n    };\n\n    // Define the private & public API methods.\n    qq.FineUploaderBasic.prototype = qq.basePublicApi;\n    qq.extend(qq.FineUploaderBasic.prototype, qq.basePrivateApi);\n}());\n","/*globals qq, XDomainRequest*/\n/** Generic class for sending non-upload ajax requests and handling the associated responses **/\nqq.AjaxRequester = function(o) {\n    \"use strict\";\n\n    var log, shouldParamsBeInQueryString,\n        queue = [],\n        requestData = {},\n        options = {\n            acceptHeader: null,\n            validMethods: [\"PATCH\", \"POST\", \"PUT\"],\n            method: \"POST\",\n            contentType: \"application/x-www-form-urlencoded\",\n            maxConnections: 3,\n            customHeaders: {},\n            endpointStore: {},\n            paramsStore: {},\n            mandatedParams: {},\n            allowXRequestedWithAndCacheControl: true,\n            successfulResponseCodes: {\n                DELETE: [200, 202, 204],\n                PATCH: [200, 201, 202, 203, 204],\n                POST: [200, 201, 202, 203, 204],\n                PUT: [200, 201, 202, 203, 204],\n                GET: [200]\n            },\n            cors: {\n                expected: false,\n                sendCredentials: false\n            },\n            log: function(str, level) {},\n            onSend: function(id) {},\n            onComplete: function(id, xhrOrXdr, isError) {},\n            onProgress: null\n        };\n\n    qq.extend(options, o);\n    log = options.log;\n\n    if (qq.indexOf(options.validMethods, options.method) < 0) {\n        throw new Error(\"'\" + options.method + \"' is not a supported method for this type of request!\");\n    }\n\n    // [Simple methods](http://www.w3.org/TR/cors/#simple-method)\n    // are defined by the W3C in the CORS spec as a list of methods that, in part,\n    // make a CORS request eligible to be exempt from preflighting.\n    function isSimpleMethod() {\n        return qq.indexOf([\"GET\", \"POST\", \"HEAD\"], options.method) >= 0;\n    }\n\n    // [Simple headers](http://www.w3.org/TR/cors/#simple-header)\n    // are defined by the W3C in the CORS spec as a list of headers that, in part,\n    // make a CORS request eligible to be exempt from preflighting.\n    function containsNonSimpleHeaders(headers) {\n        var containsNonSimple = false;\n\n        qq.each(containsNonSimple, function(idx, header) {\n            if (qq.indexOf([\"Accept\", \"Accept-Language\", \"Content-Language\", \"Content-Type\"], header) < 0) {\n                containsNonSimple = true;\n                return false;\n            }\n        });\n\n        return containsNonSimple;\n    }\n\n    function isXdr(xhr) {\n        //The `withCredentials` test is a commonly accepted way to determine if XHR supports CORS.\n        return options.cors.expected && xhr.withCredentials === undefined;\n    }\n\n    // Returns either a new `XMLHttpRequest` or `XDomainRequest` instance.\n    function getCorsAjaxTransport() {\n        var xhrOrXdr;\n\n        if (window.XMLHttpRequest || window.ActiveXObject) {\n            xhrOrXdr = qq.createXhrInstance();\n\n            if (xhrOrXdr.withCredentials === undefined) {\n                xhrOrXdr = new XDomainRequest();\n                // Workaround for XDR bug in IE9 - https://social.msdn.microsoft.com/Forums/ie/en-US/30ef3add-767c-4436-b8a9-f1ca19b4812e/ie9-rtm-xdomainrequest-issued-requests-may-abort-if-all-event-handlers-not-specified?forum=iewebdevelopment\n                xhrOrXdr.onload = function() {};\n                xhrOrXdr.onerror = function() {};\n                xhrOrXdr.ontimeout = function() {};\n                xhrOrXdr.onprogress = function() {};\n            }\n        }\n\n        return xhrOrXdr;\n    }\n\n    // Returns either a new XHR/XDR instance, or an existing one for the associated `File` or `Blob`.\n    function getXhrOrXdr(id, suppliedXhr) {\n        var xhrOrXdr = requestData[id] && requestData[id].xhr;\n\n        if (!xhrOrXdr) {\n            if (suppliedXhr) {\n                xhrOrXdr = suppliedXhr;\n            }\n            else {\n                if (options.cors.expected) {\n                    xhrOrXdr = getCorsAjaxTransport();\n                }\n                else {\n                    xhrOrXdr = qq.createXhrInstance();\n                }\n            }\n\n            requestData[id].xhr = xhrOrXdr;\n        }\n\n        return xhrOrXdr;\n    }\n\n    // Removes element from queue, sends next request\n    function dequeue(id) {\n        var i = qq.indexOf(queue, id),\n            max = options.maxConnections,\n            nextId;\n\n        delete requestData[id];\n        queue.splice(i, 1);\n\n        if (queue.length >= max && i < max) {\n            nextId = queue[max - 1];\n            sendRequest(nextId);\n        }\n    }\n\n    function onComplete(id, xdrError) {\n        var xhr = getXhrOrXdr(id),\n            method = options.method,\n            isError = xdrError === true;\n\n        dequeue(id);\n\n        if (isError) {\n            log(method + \" request for \" + id + \" has failed\", \"error\");\n        }\n        else if (!isXdr(xhr) && !isResponseSuccessful(xhr.status)) {\n            isError = true;\n            log(method + \" request for \" + id + \" has failed - response code \" + xhr.status, \"error\");\n        }\n\n        options.onComplete(id, xhr, isError);\n    }\n\n    function getParams(id) {\n        var onDemandParams = requestData[id].additionalParams,\n            mandatedParams = options.mandatedParams,\n            params;\n\n        if (options.paramsStore.get) {\n            params = options.paramsStore.get(id);\n        }\n\n        if (onDemandParams) {\n            qq.each(onDemandParams, function(name, val) {\n                params = params || {};\n                params[name] = val;\n            });\n        }\n\n        if (mandatedParams) {\n            qq.each(mandatedParams, function(name, val) {\n                params = params || {};\n                params[name] = val;\n            });\n        }\n\n        return params;\n    }\n\n    function sendRequest(id, optXhr) {\n        var xhr = getXhrOrXdr(id, optXhr),\n            method = options.method,\n            params = getParams(id),\n            payload = requestData[id].payload,\n            url;\n\n        options.onSend(id);\n\n        url = createUrl(id, params, requestData[id].additionalQueryParams);\n\n        // XDR and XHR status detection APIs differ a bit.\n        if (isXdr(xhr)) {\n            xhr.onload = getXdrLoadHandler(id);\n            xhr.onerror = getXdrErrorHandler(id);\n        }\n        else {\n            xhr.onreadystatechange = getXhrReadyStateChangeHandler(id);\n        }\n\n        registerForUploadProgress(id);\n\n        // The last parameter is assumed to be ignored if we are actually using `XDomainRequest`.\n        xhr.open(method, url, true);\n\n        // Instruct the transport to send cookies along with the CORS request,\n        // unless we are using `XDomainRequest`, which is not capable of this.\n        if (options.cors.expected && options.cors.sendCredentials && !isXdr(xhr)) {\n            xhr.withCredentials = true;\n        }\n\n        setHeaders(id);\n\n        log(\"Sending \" + method + \" request for \" + id);\n\n        if (payload) {\n            xhr.send(payload);\n        }\n        else if (shouldParamsBeInQueryString || !params) {\n            xhr.send();\n        }\n        else if (params && options.contentType && options.contentType.toLowerCase().indexOf(\"application/x-www-form-urlencoded\") >= 0) {\n            xhr.send(qq.obj2url(params, \"\"));\n        }\n        else if (params && options.contentType && options.contentType.toLowerCase().indexOf(\"application/json\") >= 0) {\n            xhr.send(JSON.stringify(params));\n        }\n        else {\n            xhr.send(params);\n        }\n\n        return xhr;\n    }\n\n    function createUrl(id, params, additionalQueryParams) {\n        var endpoint = options.endpointStore.get(id),\n            addToPath = requestData[id].addToPath;\n\n        /*jshint -W116,-W041 */\n        if (addToPath != undefined) {\n            endpoint += \"/\" + addToPath;\n        }\n\n        if (shouldParamsBeInQueryString && params) {\n            endpoint = qq.obj2url(params, endpoint);\n        }\n\n        if (additionalQueryParams) {\n            endpoint = qq.obj2url(additionalQueryParams, endpoint);\n        }\n\n        return endpoint;\n    }\n\n    // Invoked by the UA to indicate a number of possible states that describe\n    // a live `XMLHttpRequest` transport.\n    function getXhrReadyStateChangeHandler(id) {\n        return function() {\n            if (getXhrOrXdr(id).readyState === 4) {\n                onComplete(id);\n            }\n        };\n    }\n\n    function registerForUploadProgress(id) {\n        var onProgress = options.onProgress;\n\n        if (onProgress) {\n            getXhrOrXdr(id).upload.onprogress = function(e) {\n                if (e.lengthComputable) {\n                    onProgress(id, e.loaded, e.total);\n                }\n            };\n        }\n    }\n\n    // This will be called by IE to indicate **success** for an associated\n    // `XDomainRequest` transported request.\n    function getXdrLoadHandler(id) {\n        return function() {\n            onComplete(id);\n        };\n    }\n\n    // This will be called by IE to indicate **failure** for an associated\n    // `XDomainRequest` transported request.\n    function getXdrErrorHandler(id) {\n        return function() {\n            onComplete(id, true);\n        };\n    }\n\n    function setHeaders(id) {\n        var xhr = getXhrOrXdr(id),\n            customHeaders = options.customHeaders,\n            onDemandHeaders = requestData[id].additionalHeaders || {},\n            method = options.method,\n            allHeaders = {};\n\n        // If XDomainRequest is being used, we can't set headers, so just ignore this block.\n        if (!isXdr(xhr)) {\n            options.acceptHeader && xhr.setRequestHeader(\"Accept\", options.acceptHeader);\n\n            // Only attempt to add X-Requested-With & Cache-Control if permitted\n            if (options.allowXRequestedWithAndCacheControl) {\n                // Do not add X-Requested-With & Cache-Control if this is a cross-origin request\n                // OR the cross-origin request contains a non-simple method or header.\n                // This is done to ensure a preflight is not triggered exclusively based on the\n                // addition of these 2 non-simple headers.\n                if (!options.cors.expected || (!isSimpleMethod() || containsNonSimpleHeaders(customHeaders))) {\n                    xhr.setRequestHeader(\"X-Requested-With\", \"XMLHttpRequest\");\n                    xhr.setRequestHeader(\"Cache-Control\", \"no-cache\");\n                }\n            }\n\n            if (options.contentType && (method === \"POST\" || method === \"PUT\")) {\n                xhr.setRequestHeader(\"Content-Type\", options.contentType);\n            }\n\n            qq.extend(allHeaders, qq.isFunction(customHeaders) ? customHeaders(id) : customHeaders);\n            qq.extend(allHeaders, onDemandHeaders);\n\n            qq.each(allHeaders, function(name, val) {\n                xhr.setRequestHeader(name, val);\n            });\n        }\n    }\n\n    function isResponseSuccessful(responseCode) {\n        return qq.indexOf(options.successfulResponseCodes[options.method], responseCode) >= 0;\n    }\n\n    function prepareToSend(id, optXhr, addToPath, additionalParams, additionalQueryParams, additionalHeaders, payload) {\n        requestData[id] = {\n            addToPath: addToPath,\n            additionalParams: additionalParams,\n            additionalQueryParams: additionalQueryParams,\n            additionalHeaders: additionalHeaders,\n            payload: payload\n        };\n\n        var len = queue.push(id);\n\n        // if too many active connections, wait...\n        if (len <= options.maxConnections) {\n            return sendRequest(id, optXhr);\n        }\n    }\n\n    shouldParamsBeInQueryString = options.method === \"GET\" || options.method === \"DELETE\";\n\n    qq.extend(this, {\n        // Start the process of sending the request.  The ID refers to the file associated with the request.\n        initTransport: function(id) {\n            var path, params, headers, payload, cacheBuster, additionalQueryParams;\n\n            return {\n                // Optionally specify the end of the endpoint path for the request.\n                withPath: function(appendToPath) {\n                    path = appendToPath;\n                    return this;\n                },\n\n                // Optionally specify additional parameters to send along with the request.\n                // These will be added to the query string for GET/DELETE requests or the payload\n                // for POST/PUT requests.  The Content-Type of the request will be used to determine\n                // how these parameters should be formatted as well.\n                withParams: function(additionalParams) {\n                    params = additionalParams;\n                    return this;\n                },\n\n                withQueryParams: function(_additionalQueryParams_) {\n                    additionalQueryParams = _additionalQueryParams_;\n                    return this;\n                },\n\n                // Optionally specify additional headers to send along with the request.\n                withHeaders: function(additionalHeaders) {\n                    headers = additionalHeaders;\n                    return this;\n                },\n\n                // Optionally specify a payload/body for the request.\n                withPayload: function(thePayload) {\n                    payload = thePayload;\n                    return this;\n                },\n\n                // Appends a cache buster (timestamp) to the request URL as a query parameter (only if GET or DELETE)\n                withCacheBuster: function() {\n                    cacheBuster = true;\n                    return this;\n                },\n\n                // Send the constructed request.\n                send: function(optXhr) {\n                    if (cacheBuster && qq.indexOf([\"GET\", \"DELETE\"], options.method) >= 0) {\n                        params.qqtimestamp = new Date().getTime();\n                    }\n\n                    return prepareToSend(id, optXhr, path, params, additionalQueryParams, headers, payload);\n                }\n            };\n        },\n\n        canceled: function(id) {\n            dequeue(id);\n        }\n    });\n};\n","/* globals qq */\n/**\n * Common upload handler functions.\n *\n * @constructor\n */\nqq.UploadHandler = function(spec) {\n    \"use strict\";\n\n    var proxy = spec.proxy,\n        fileState = {},\n        onCancel = proxy.onCancel,\n        getName = proxy.getName;\n\n    qq.extend(this, {\n        add: function(id, fileItem) {\n            fileState[id] = fileItem;\n            fileState[id].temp = {};\n        },\n\n        cancel: function(id) {\n            var self = this,\n                cancelFinalizationEffort = new qq.Promise(),\n                onCancelRetVal = onCancel(id, getName(id), cancelFinalizationEffort);\n\n            onCancelRetVal.then(function() {\n                if (self.isValid(id)) {\n                    fileState[id].canceled = true;\n                    self.expunge(id);\n                }\n                cancelFinalizationEffort.success();\n            });\n        },\n\n        expunge: function(id) {\n            delete fileState[id];\n        },\n\n        getThirdPartyFileId: function(id) {\n            return fileState[id].key;\n        },\n\n        isValid: function(id) {\n            return fileState[id] !== undefined;\n        },\n\n        reset: function() {\n            fileState = {};\n        },\n\n        _getFileState: function(id) {\n            return fileState[id];\n        },\n\n        _setThirdPartyFileId: function(id, thirdPartyFileId) {\n            fileState[id].key = thirdPartyFileId;\n        },\n\n        _wasCanceled: function(id) {\n            return !!fileState[id].canceled;\n        }\n    });\n};\n","/*globals qq*/\n/**\n * Base upload handler module.  Controls more specific handlers.\n *\n * @param o Options.  Passed along to the specific handler submodule as well.\n * @param namespace [optional] Namespace for the specific handler.\n */\nqq.UploadHandlerController = function(o, namespace) {\n    \"use strict\";\n\n    var controller = this,\n        chunkingPossible = false,\n        concurrentChunkingPossible = false,\n        chunking, preventRetryResponse, log, handler,\n\n    options = {\n        paramsStore: {},\n        maxConnections: 3, // maximum number of concurrent uploads\n        chunking: {\n            enabled: false,\n            multiple: {\n                enabled: false\n            }\n        },\n        log: function(str, level) {},\n        onProgress: function(id, fileName, loaded, total) {},\n        onComplete: function(id, fileName, response, xhr) {},\n        onCancel: function(id, fileName) {},\n        onUploadPrep: function(id) {}, // Called if non-trivial operations will be performed before onUpload\n        onUpload: function(id, fileName) {},\n        onUploadChunk: function(id, fileName, chunkData) {},\n        onUploadChunkSuccess: function(id, chunkData, response, xhr) {},\n        onAutoRetry: function(id, fileName, response, xhr) {},\n        onResume: function(id, fileName, chunkData, customResumeData) {},\n        onUuidChanged: function(id, newUuid) {},\n        getName: function(id) {},\n        setSize: function(id, newSize) {},\n        isQueued: function(id) {},\n        getIdsInProxyGroup: function(id) {},\n        getIdsInBatch: function(id) {},\n        isInProgress: function(id) {}\n    },\n\n    chunked = {\n        // Called when each chunk has uploaded successfully\n        done: function(id, chunkIdx, response, xhr) {\n            var chunkData = handler._getChunkData(id, chunkIdx);\n\n            handler._getFileState(id).attemptingResume = false;\n\n            delete handler._getFileState(id).temp.chunkProgress[chunkIdx];\n            handler._getFileState(id).loaded += chunkData.size;\n\n            options.onUploadChunkSuccess(id, handler._getChunkDataForCallback(chunkData), response, xhr);\n        },\n\n        // Called when all chunks have been successfully uploaded and we want to ask the handler to perform any\n        // logic associated with closing out the file, such as combining the chunks.\n        finalize: function(id) {\n            var size = options.getSize(id),\n                name = options.getName(id);\n\n            log(\"All chunks have been uploaded for \" + id + \" - finalizing....\");\n            handler.finalizeChunks(id).then(\n                function(response, xhr) {\n                    log(\"Finalize successful for \" + id);\n\n                    var normaizedResponse = upload.normalizeResponse(response, true);\n\n                    options.onProgress(id, name, size, size);\n                    handler._maybeDeletePersistedChunkData(id);\n                    upload.cleanup(id, normaizedResponse, xhr);\n                },\n                function(response, xhr) {\n                    var normalizedResponse = upload.normalizeResponse(response, false);\n\n                    log(\"Problem finalizing chunks for file ID \" + id + \" - \" + normalizedResponse.error, \"error\");\n\n                    if (\n                        normalizedResponse.reset ||\n                        (xhr && options.chunking.success.resetOnStatus.indexOf(xhr.status) >= 0)\n                    ) {\n                        chunked.reset(id);\n                    }\n\n                    if (!options.onAutoRetry(id, name, normalizedResponse, xhr)) {\n                        upload.cleanup(id, normalizedResponse, xhr);\n                    }\n                }\n            );\n        },\n\n        handleFailure: function(chunkIdx, id, response, xhr) {\n            var name = options.getName(id);\n\n            log(\"Chunked upload request failed for \" + id + \", chunk \" + chunkIdx);\n\n            handler.clearCachedChunk(id, chunkIdx);\n\n            var responseToReport = upload.normalizeResponse(response, false),\n                inProgressIdx;\n\n            if (responseToReport.reset) {\n                chunked.reset(id);\n            }\n            else {\n                var inProgressChunksArray = handler._getFileState(id).chunking.inProgress;\n\n                inProgressIdx = inProgressChunksArray ? qq.indexOf(inProgressChunksArray, chunkIdx) : -1;\n                if (inProgressIdx >= 0) {\n                    handler._getFileState(id).chunking.inProgress.splice(inProgressIdx, 1);\n                    handler._getFileState(id).chunking.remaining.unshift(chunkIdx);\n                }\n            }\n\n            // We may have aborted all other in-progress chunks for this file due to a failure.\n            // If so, ignore the failures associated with those aborts.\n            if (!handler._getFileState(id).temp.ignoreFailure) {\n                // If this chunk has failed, we want to ignore all other failures of currently in-progress\n                // chunks since they will be explicitly aborted\n                if (concurrentChunkingPossible) {\n                    handler._getFileState(id).temp.ignoreFailure = true;\n\n                    log(qq.format(\"Going to attempt to abort these chunks: {}. These are currently in-progress: {}.\", JSON.stringify(Object.keys(handler._getXhrs(id))), JSON.stringify(handler._getFileState(id).chunking.inProgress)));\n                    qq.each(handler._getXhrs(id), function(ckid, ckXhr) {\n                        log(qq.format(\"Attempting to abort file {}.{}. XHR readyState {}. \", id, ckid, ckXhr.readyState));\n                        ckXhr.abort();\n                        // Flag the transport, in case we are waiting for some other async operation\n                        // to complete before attempting to upload the chunk\n                        ckXhr._cancelled = true;\n                    });\n\n                    // We must indicate that all aborted chunks are no longer in progress\n                    handler.moveInProgressToRemaining(id);\n\n                    // Free up any connections used by these chunks, but don't allow any\n                    // other files to take up the connections (until we have exhausted all auto-retries)\n                    connectionManager.free(id, true);\n                }\n\n                if (!options.onAutoRetry(id, name, responseToReport, xhr)) {\n                    // If one chunk fails, abort all of the others to avoid odd race conditions that occur\n                    // if a chunk succeeds immediately after one fails before we have determined if the upload\n                    // is a failure or not.\n                    upload.cleanup(id, responseToReport, xhr);\n                }\n            }\n        },\n\n        hasMoreParts: function(id) {\n            return !!handler._getFileState(id).chunking.remaining.length;\n        },\n\n        nextPart: function(id) {\n            var nextIdx = handler._getFileState(id).chunking.remaining.shift();\n\n            if (nextIdx >= handler._getTotalChunks(id)) {\n                nextIdx = null;\n            }\n\n            return nextIdx;\n        },\n\n        reset: function(id) {\n            log(\"Server or callback has ordered chunking effort to be restarted on next attempt for item ID \" + id, \"error\");\n\n            handler._maybeDeletePersistedChunkData(id);\n            handler.reevaluateChunking(id);\n            handler._getFileState(id).loaded = 0;\n            handler._getFileState(id).attemptingResume = false;\n        },\n\n        sendNext: function(id) {\n            var size = options.getSize(id),\n                name = options.getName(id),\n                chunkIdx = chunked.nextPart(id),\n                chunkData = handler._getChunkData(id, chunkIdx),\n                fileState = handler._getFileState(id),\n                resuming = fileState.attemptingResume,\n                inProgressChunks = fileState.chunking.inProgress || [];\n\n            if (fileState.loaded == null) {\n                fileState.loaded = 0;\n            }\n\n            // Don't follow-through with the resume attempt if the integrator returns false from onResume\n            if (resuming && options.onResume(id, name, chunkData, fileState.customResumeData) === false) {\n                chunked.reset(id);\n                chunkIdx = chunked.nextPart(id);\n                chunkData = handler._getChunkData(id, chunkIdx);\n                resuming = false;\n            }\n\n            // If all chunks have already uploaded successfully, we must be re-attempting the finalize step.\n            if (chunkIdx == null && inProgressChunks.length === 0) {\n                chunked.finalize(id);\n            }\n\n            // Send the next chunk\n            else {\n                inProgressChunks.push(chunkIdx);\n                handler._getFileState(id).chunking.inProgress = inProgressChunks;\n\n                if (concurrentChunkingPossible) {\n                    connectionManager.open(id, chunkIdx);\n                }\n\n                if (concurrentChunkingPossible && connectionManager.available() && handler._getFileState(id).chunking.remaining.length) {\n                    chunked.sendNext(id);\n                }\n\n                if (chunkData.blob.size === 0) {\n                    log(qq.format(\"Chunk {} for file {} will not be uploaded, zero sized chunk.\", chunkIdx, id), \"error\");\n                    chunked.handleFailure(chunkIdx, id, \"File is no longer available\", null);\n                }\n\n                var onUploadChunkPromise = options.onUploadChunk(id, name, handler._getChunkDataForCallback(chunkData));\n\n                onUploadChunkPromise.then(\n                    function(requestOverrides) {\n                        if (!options.isInProgress(id)) {\n                            log(qq.format(\"Not sending chunked upload request for item {}.{} - no longer in progress.\", id, chunkIdx));\n                        }\n                        else {\n                            log(qq.format(\"Sending chunked upload request for item {}.{}, bytes {}-{} of {}.\", id, chunkIdx, chunkData.start + 1, chunkData.end, size));\n\n                            var uploadChunkData = {\n                                chunkIdx: chunkIdx,\n                                id: id,\n                                overrides: requestOverrides,\n                                resuming: resuming\n                            };\n\n                            handler.uploadChunk(uploadChunkData).then(\n                                // upload chunk success\n                                function success(response, xhr) {\n                                    log(\"Chunked upload request succeeded for \" + id + \", chunk \" + chunkIdx);\n\n                                    handler.clearCachedChunk(id, chunkIdx);\n\n                                    var inProgressChunks = handler._getFileState(id).chunking.inProgress || [],\n                                        responseToReport = upload.normalizeResponse(response, true),\n                                        inProgressChunkIdx = qq.indexOf(inProgressChunks, chunkIdx);\n\n                                    log(qq.format(\"Chunk {} for file {} uploaded successfully.\", chunkIdx, id));\n\n                                    chunked.done(id, chunkIdx, responseToReport, xhr);\n\n                                    if (inProgressChunkIdx >= 0) {\n                                        inProgressChunks.splice(inProgressChunkIdx, 1);\n                                    }\n\n                                    handler._maybePersistChunkedState(id);\n\n                                    if (!chunked.hasMoreParts(id) && inProgressChunks.length === 0) {\n                                        chunked.finalize(id);\n                                    }\n                                    else if (chunked.hasMoreParts(id)) {\n                                        chunked.sendNext(id);\n                                    }\n                                    else {\n                                        log(qq.format(\"File ID {} has no more chunks to send and these chunk indexes are still marked as in-progress: {}\", id, JSON.stringify(inProgressChunks)));\n                                    }\n                                },\n\n                                // upload chunk failure\n                                function failure(response, xhr) {\n                                    chunked.handleFailure(chunkIdx, id, response, xhr);\n                                }\n                            )\n                                .done(function () {\n                                    handler.clearXhr(id, chunkIdx);\n                                });\n                        }\n                    },\n\n                    function(error) {\n                        chunked.handleFailure(chunkIdx, id, error, null);\n                    }\n                );\n            }\n        }\n    },\n\n    connectionManager = {\n        _open: [],\n        _openChunks: {},\n        _waiting: [],\n\n        available: function() {\n            var max = options.maxConnections,\n                openChunkEntriesCount = 0,\n                openChunksCount = 0;\n\n            qq.each(connectionManager._openChunks, function(fileId, openChunkIndexes) {\n                openChunkEntriesCount++;\n                openChunksCount += openChunkIndexes.length;\n            });\n\n            return max - (connectionManager._open.length - openChunkEntriesCount + openChunksCount);\n        },\n\n        /**\n         * Removes element from queue, starts upload of next\n         */\n        free: function(id, dontAllowNext) {\n            var allowNext = !dontAllowNext,\n                waitingIndex = qq.indexOf(connectionManager._waiting, id),\n                connectionsIndex = qq.indexOf(connectionManager._open, id),\n                nextId;\n\n            delete connectionManager._openChunks[id];\n\n            if (upload.getProxyOrBlob(id) instanceof qq.BlobProxy) {\n                log(\"Generated blob upload has ended for \" + id + \", disposing generated blob.\");\n                delete handler._getFileState(id).file;\n            }\n\n            // If this file was not consuming a connection, it was just waiting, so remove it from the waiting array\n            if (waitingIndex >= 0) {\n                connectionManager._waiting.splice(waitingIndex, 1);\n            }\n            // If this file was consuming a connection, allow the next file to be uploaded\n            else if (allowNext && connectionsIndex >= 0) {\n                connectionManager._open.splice(connectionsIndex, 1);\n\n                nextId = connectionManager._waiting.shift();\n                if (nextId >= 0) {\n                    connectionManager._open.push(nextId);\n                    upload.start(nextId);\n                }\n            }\n        },\n\n        getWaitingOrConnected: function() {\n            var waitingOrConnected = [];\n\n            // Chunked files may have multiple connections open per chunk (if concurrent chunking is enabled)\n            // We need to grab the file ID of any file that has at least one chunk consuming a connection.\n            qq.each(connectionManager._openChunks, function(fileId, chunks) {\n                if (chunks && chunks.length) {\n                    waitingOrConnected.push(parseInt(fileId));\n                }\n            });\n\n            // For non-chunked files, only one connection will be consumed per file.\n            // This is where we aggregate those file IDs.\n            qq.each(connectionManager._open, function(idx, fileId) {\n                if (!connectionManager._openChunks[fileId]) {\n                    waitingOrConnected.push(parseInt(fileId));\n                }\n            });\n\n            // There may be files waiting for a connection.\n            waitingOrConnected = waitingOrConnected.concat(connectionManager._waiting);\n\n            return waitingOrConnected;\n        },\n\n        isUsingConnection: function(id) {\n            return qq.indexOf(connectionManager._open, id) >= 0;\n        },\n\n        open: function(id, chunkIdx) {\n            if (chunkIdx == null) {\n                connectionManager._waiting.push(id);\n            }\n\n            if (connectionManager.available()) {\n                if (chunkIdx == null) {\n                    connectionManager._waiting.pop();\n                    connectionManager._open.push(id);\n                }\n                else {\n                    (function() {\n                        var openChunksEntry = connectionManager._openChunks[id] || [];\n                        openChunksEntry.push(chunkIdx);\n                        connectionManager._openChunks[id] = openChunksEntry;\n                    }());\n                }\n\n                return true;\n            }\n\n            return false;\n        },\n\n        reset: function() {\n            connectionManager._waiting = [];\n            connectionManager._open = [];\n        }\n    },\n\n    simple = {\n        send: function(id, name) {\n            var fileState = handler._getFileState(id);\n\n            if (!fileState) {\n                log(\"Ignoring send request as this upload may have been cancelled, File ID \" + id, \"warn\");\n                return;\n            }\n\n            fileState.loaded = 0;\n\n            log(\"Sending simple upload request for \" + id);\n            handler.uploadFile(id).then(\n                function(response, optXhr) {\n                    log(\"Simple upload request succeeded for \" + id);\n\n                    var responseToReport = upload.normalizeResponse(response, true),\n                        size = options.getSize(id);\n\n                    options.onProgress(id, name, size, size);\n                    upload.maybeNewUuid(id, responseToReport);\n                    upload.cleanup(id, responseToReport, optXhr);\n                },\n\n                function(response, optXhr) {\n                    log(\"Simple upload request failed for \" + id);\n\n                    var responseToReport = upload.normalizeResponse(response, false);\n\n                    if (!options.onAutoRetry(id, name, responseToReport, optXhr)) {\n                        upload.cleanup(id, responseToReport, optXhr);\n                    }\n                }\n            );\n        }\n    },\n\n    upload = {\n        cancel: function(id) {\n            log(\"Cancelling \" + id);\n            options.paramsStore.remove(id);\n            connectionManager.free(id);\n        },\n\n        cleanup: function(id, response, optXhr) {\n            var name = options.getName(id);\n\n            options.onComplete(id, name, response, optXhr);\n\n            if (handler._getFileState(id)) {\n                handler._clearXhrs && handler._clearXhrs(id);\n            }\n\n            connectionManager.free(id);\n        },\n\n        // Returns a qq.BlobProxy, or an actual File/Blob if no proxy is involved, or undefined\n        // if none of these are available for the ID\n        getProxyOrBlob: function(id) {\n            return (handler.getProxy && handler.getProxy(id)) ||\n                (handler.getFile && handler.getFile(id));\n        },\n\n        initHandler: function() {\n            var handlerType = namespace ? qq[namespace] : qq.traditional,\n                handlerModuleSubtype = qq.supportedFeatures.ajaxUploading ? \"Xhr\" : \"Form\";\n\n            handler = new handlerType[handlerModuleSubtype + \"UploadHandler\"](\n                options,\n                {\n                    getCustomResumeData: options.getCustomResumeData,\n                    getDataByUuid: options.getDataByUuid,\n                    getName: options.getName,\n                    getSize: options.getSize,\n                    getUuid: options.getUuid,\n                    log: log,\n                    onCancel: options.onCancel,\n                    onProgress: options.onProgress,\n                    onUuidChanged: options.onUuidChanged,\n                    onFinalizing: function(id) {\n                        options.setStatus(id, qq.status.UPLOAD_FINALIZING);\n                    }\n                }\n            );\n\n            if (handler._removeExpiredChunkingRecords) {\n                handler._removeExpiredChunkingRecords();\n            }\n        },\n\n        isDeferredEligibleForUpload: function(id) {\n            return options.isQueued(id);\n        },\n\n        // For Blobs that are part of a group of generated images, along with a reference image,\n        // this will ensure the blobs in the group are uploaded in the order they were triggered,\n        // even if some async processing must be completed on one or more Blobs first.\n        maybeDefer: function(id, blob) {\n            // If we don't have a file/blob yet & no file/blob exists for this item, request it,\n            // and then submit the upload to the specific handler once the blob is available.\n            // ASSUMPTION: This condition will only ever be true if XHR uploading is supported.\n            if (blob && !handler.getFile(id) && blob instanceof qq.BlobProxy) {\n\n                // Blob creation may take some time, so the caller may want to update the\n                // UI to indicate that an operation is in progress, even before the actual\n                // upload begins and an onUpload callback is invoked.\n                options.onUploadPrep(id);\n\n                log(\"Attempting to generate a blob on-demand for \" + id);\n                blob.create().then(function(generatedBlob) {\n                    log(\"Generated an on-demand blob for \" + id);\n\n                    // Update record associated with this file by providing the generated Blob\n                    handler.updateBlob(id, generatedBlob);\n\n                    // Propagate the size for this generated Blob\n                    options.setSize(id, generatedBlob.size);\n\n                    // Order handler to recalculate chunking possibility, if applicable\n                    handler.reevaluateChunking(id);\n\n                    upload.maybeSendDeferredFiles(id);\n                },\n\n                // Blob could not be generated.  Fail the upload & attempt to prevent retries.  Also bubble error message.\n                function(errorMessage) {\n                    var errorResponse = {};\n\n                    if (errorMessage) {\n                        errorResponse.error = errorMessage;\n                    }\n\n                    log(qq.format(\"Failed to generate blob for ID {}.  Error message: {}.\", id, errorMessage), \"error\");\n\n                    options.onComplete(id, options.getName(id), qq.extend(errorResponse, preventRetryResponse), null);\n                    upload.maybeSendDeferredFiles(id);\n                    connectionManager.free(id);\n                });\n            }\n            else {\n                return upload.maybeSendDeferredFiles(id);\n            }\n\n            return false;\n        },\n\n        // Upload any grouped blobs, in the proper order, that are ready to be uploaded\n        maybeSendDeferredFiles: function(id) {\n            var idsInGroup = options.getIdsInProxyGroup(id),\n                uploadedThisId = false;\n\n            if (idsInGroup && idsInGroup.length) {\n                log(\"Maybe ready to upload proxy group file \" + id);\n\n                qq.each(idsInGroup, function(idx, idInGroup) {\n                    if (upload.isDeferredEligibleForUpload(idInGroup) && !!handler.getFile(idInGroup)) {\n                        uploadedThisId = idInGroup === id;\n                        upload.now(idInGroup);\n                    }\n                    else if (upload.isDeferredEligibleForUpload(idInGroup)) {\n                        return false;\n                    }\n                });\n            }\n            else {\n                uploadedThisId = true;\n                upload.now(id);\n            }\n\n            return uploadedThisId;\n        },\n\n        maybeNewUuid: function(id, response) {\n            if (response.newUuid !== undefined) {\n                options.onUuidChanged(id, response.newUuid);\n            }\n        },\n\n        // The response coming from handler implementations may be in various formats.\n        // Instead of hoping a promise nested 5 levels deep will always return an object\n        // as its first param, let's just normalize the response here.\n        normalizeResponse: function(originalResponse, successful) {\n            var response = originalResponse;\n\n            // The passed \"response\" param may not be a response at all.\n            // It could be a string, detailing the error, for example.\n            if (!qq.isObject(originalResponse)) {\n                response = {};\n\n                if (qq.isString(originalResponse) && !successful) {\n                    response.error = originalResponse;\n                }\n            }\n\n            response.success = successful;\n\n            return response;\n        },\n\n        now: function(id) {\n            var name = options.getName(id);\n\n            if (!controller.isValid(id)) {\n                throw new qq.Error(id + \" is not a valid file ID to upload!\");\n            }\n\n            options.onUpload(id, name).then(\n                function(response) {\n                    if (response && response.pause) {\n                        options.setStatus(id, qq.status.PAUSED);\n                        handler.pause(id);\n                        connectionManager.free(id);\n                    }\n                    else {\n                        if (chunkingPossible && handler._shouldChunkThisFile(id)) {\n                            chunked.sendNext(id);\n                        }\n                        else {\n                            simple.send(id, name);\n                        }\n                    }\n                },\n\n                function(error) {\n                    error = error || {};\n\n                    log(id + \" upload start aborted due to rejected onUpload Promise - details: \" + error, \"error\");\n\n                    if (!options.onAutoRetry(id, name, error.responseJSON || {})) {\n                        var response = upload.normalizeResponse(error.responseJSON, false);\n                        upload.cleanup(id, response);\n                    }\n                }\n            );\n        },\n\n        start: function(id) {\n            var blobToUpload = upload.getProxyOrBlob(id);\n\n            if (blobToUpload) {\n                return upload.maybeDefer(id, blobToUpload);\n            }\n            else {\n                upload.now(id);\n                return true;\n            }\n        }\n    };\n\n    qq.extend(this, {\n        /**\n         * Adds file or file input to the queue\n         **/\n        add: function(id, file) {\n            handler.add.apply(this, arguments);\n        },\n\n        /**\n         * Sends the file identified by id\n         */\n        upload: function(id) {\n            if (connectionManager.open(id)) {\n                return upload.start(id);\n            }\n            return false;\n        },\n\n        retry: function(id) {\n            // On retry, if concurrent chunking has been enabled, we may have aborted all other in-progress chunks\n            // for a file when encountering a failed chunk upload.  We then signaled the controller to ignore\n            // all failures associated with these aborts.  We are now retrying, so we don't want to ignore\n            // any more failures at this point.\n            if (concurrentChunkingPossible) {\n                handler._getFileState(id).temp.ignoreFailure = false;\n            }\n\n            // If we are attempting to retry a file that is already consuming a connection, this is likely an auto-retry.\n            // Just go ahead and ask the handler to upload again.\n            if (connectionManager.isUsingConnection(id)) {\n                return upload.start(id);\n            }\n\n            // If we are attempting to retry a file that is not currently consuming a connection,\n            // this is likely a manual retry attempt.  We will need to ensure a connection is available\n            // before the retry commences.\n            else {\n                return controller.upload(id);\n            }\n        },\n\n        /**\n         * Cancels file upload by id\n         */\n        cancel: function(id) {\n            var cancelRetVal = handler.cancel(id);\n\n            if (qq.isGenericPromise(cancelRetVal)) {\n                cancelRetVal.then(function() {\n                    upload.cancel(id);\n                });\n            }\n            else if (cancelRetVal !== false) {\n                upload.cancel(id);\n            }\n        },\n\n        /**\n         * Cancels all queued or in-progress uploads\n         */\n        cancelAll: function() {\n            var waitingOrConnected = connectionManager.getWaitingOrConnected(),\n                i;\n\n            // ensure files are cancelled in reverse order which they were added\n            // to avoid a flash of time where a queued file begins to upload before it is canceled\n            if (waitingOrConnected.length) {\n                for (i = waitingOrConnected.length - 1; i >= 0; i--) {\n                    controller.cancel(waitingOrConnected[i]);\n                }\n            }\n\n            connectionManager.reset();\n        },\n\n        // Returns a File, Blob, or the Blob/File for the reference/parent file if the targeted blob is a proxy.\n        // Undefined if no file record is available.\n        getFile: function(id) {\n            if (handler.getProxy && handler.getProxy(id)) {\n                return handler.getProxy(id).referenceBlob;\n            }\n\n            return handler.getFile && handler.getFile(id);\n        },\n\n        // Returns true if the Blob associated with the ID is related to a proxy s\n        isProxied: function(id) {\n            return !!(handler.getProxy && handler.getProxy(id));\n        },\n\n        getInput: function(id) {\n            if (handler.getInput) {\n                return handler.getInput(id);\n            }\n        },\n\n        reset: function() {\n            log(\"Resetting upload handler\");\n            controller.cancelAll();\n            connectionManager.reset();\n            handler.reset();\n        },\n\n        expunge: function(id) {\n            if (controller.isValid(id)) {\n                return handler.expunge(id);\n            }\n        },\n\n        /**\n         * Determine if the file exists.\n         */\n        isValid: function(id) {\n            return handler.isValid(id);\n        },\n\n        hasResumeRecord: function(id) {\n            var key = handler.isValid(id) &&\n                handler._getLocalStorageId &&\n                handler._getLocalStorageId(id);\n\n            if (key) {\n                return !!localStorage.getItem(key);\n            }\n\n            return false;\n        },\n\n        getResumableFilesData: function() {\n            if (handler.getResumableFilesData) {\n                return handler.getResumableFilesData();\n            }\n            return [];\n        },\n\n        /**\n         * This may or may not be implemented, depending on the handler.  For handlers where a third-party ID is\n         * available (such as the \"key\" for Amazon S3), this will return that value.  Otherwise, the return value\n         * will be undefined.\n         *\n         * @param id Internal file ID\n         * @returns {*} Some identifier used by a 3rd-party service involved in the upload process\n         */\n        getThirdPartyFileId: function(id) {\n            if (controller.isValid(id)) {\n                return handler.getThirdPartyFileId(id);\n            }\n        },\n\n        /**\n         * Attempts to pause the associated upload if the specific handler supports this and the file is \"valid\".\n         * @param id ID of the upload/file to pause\n         * @returns {boolean} true if the upload was paused\n         */\n        pause: function(id) {\n            if (controller.isResumable(id) && handler.pause && controller.isValid(id) && handler.pause(id)) {\n                connectionManager.free(id);\n                handler.moveInProgressToRemaining(id);\n                return true;\n            }\n            return false;\n        },\n\n        isAttemptingResume: function(id) {\n            return !!handler.isAttemptingResume && handler.isAttemptingResume(id);\n        },\n\n        // True if the file is eligible for pause/resume.\n        isResumable: function(id) {\n            return !!handler.isResumable && handler.isResumable(id);\n        }\n    });\n\n    qq.extend(options, o);\n    log = options.log;\n    chunkingPossible = options.chunking.enabled && qq.supportedFeatures.chunking;\n    concurrentChunkingPossible = chunkingPossible && options.chunking.concurrent.enabled;\n\n    preventRetryResponse = (function() {\n        var response = {};\n\n        response[options.preventRetryParam] = true;\n\n        return response;\n    }());\n\n    upload.initHandler();\n};\n","/*globals qq */\n/*jshint -W117 */\nqq.WindowReceiveMessage = function(o) {\n    \"use strict\";\n\n    var options = {\n            log: function(message, level) {}\n        },\n        callbackWrapperDetachers = {};\n\n    qq.extend(options, o);\n\n    qq.extend(this, {\n        receiveMessage: function(id, callback) {\n            var onMessageCallbackWrapper = function(event) {\n                    callback(event.data);\n                };\n\n            if (window.postMessage) {\n                callbackWrapperDetachers[id] = qq(window).attach(\"message\", onMessageCallbackWrapper);\n            }\n            else {\n                log(\"iframe message passing not supported in this browser!\", \"error\");\n            }\n        },\n\n        stopReceivingMessages: function(id) {\n            if (window.postMessage) {\n                var detacher = callbackWrapperDetachers[id];\n                if (detacher) {\n                    detacher();\n                }\n            }\n        }\n    });\n};\n","/* globals qq */\n/**\n * Common APIs exposed to creators of upload via form/iframe handlers.  This is reused and possibly overridden\n * in some cases by specific form upload handlers.\n *\n * @constructor\n */\nqq.FormUploadHandler = function(spec) {\n    \"use strict\";\n\n    var options = spec.options,\n        handler = this,\n        proxy = spec.proxy,\n        formHandlerInstanceId = qq.getUniqueId(),\n        onloadCallbacks = {},\n        detachLoadEvents = {},\n        postMessageCallbackTimers = {},\n        isCors = options.isCors,\n        inputName = options.inputName,\n        getUuid = proxy.getUuid,\n        log = proxy.log,\n        corsMessageReceiver = new qq.WindowReceiveMessage({log: log});\n\n    /**\n     * Remove any trace of the file from the handler.\n     *\n     * @param id ID of the associated file\n     */\n    function expungeFile(id) {\n        delete detachLoadEvents[id];\n\n        // If we are dealing with CORS, we might still be waiting for a response from a loaded iframe.\n        // In that case, terminate the timer waiting for a message from the loaded iframe\n        // and stop listening for any more messages coming from this iframe.\n        if (isCors) {\n            clearTimeout(postMessageCallbackTimers[id]);\n            delete postMessageCallbackTimers[id];\n            corsMessageReceiver.stopReceivingMessages(id);\n        }\n\n        var iframe = document.getElementById(handler._getIframeName(id));\n        if (iframe) {\n            // To cancel request set src to something else.  We use src=\"javascript:false;\"\n            // because it doesn't trigger ie6 prompt on https\n            /* jshint scripturl:true */\n            iframe.setAttribute(\"src\", \"javascript:false;\");\n\n            qq(iframe).remove();\n        }\n    }\n\n    /**\n     * @param iframeName `document`-unique Name of the associated iframe\n     * @returns {*} ID of the associated file\n     */\n    function getFileIdForIframeName(iframeName) {\n        return iframeName.split(\"_\")[0];\n    }\n\n    /**\n     * Generates an iframe to be used as a target for upload-related form submits.  This also adds the iframe\n     * to the current `document`.  Note that the iframe is hidden from view.\n     *\n     * @param name Name of the iframe.\n     * @returns {HTMLIFrameElement} The created iframe\n     */\n    function initIframeForUpload(name) {\n        var iframe = qq.toElement(\"<iframe src='javascript:false;' name='\" + name + \"' />\");\n\n        iframe.setAttribute(\"id\", name);\n\n        iframe.style.display = \"none\";\n        document.body.appendChild(iframe);\n\n        return iframe;\n    }\n\n    /**\n     * If we are in CORS mode, we must listen for messages (containing the server response) from the associated\n     * iframe, since we cannot directly parse the content of the iframe due to cross-origin restrictions.\n     *\n     * @param iframe Listen for messages on this iframe.\n     * @param callback Invoke this callback with the message from the iframe.\n     */\n    function registerPostMessageCallback(iframe, callback) {\n        var iframeName = iframe.id,\n            fileId = getFileIdForIframeName(iframeName),\n            uuid = getUuid(fileId);\n\n        onloadCallbacks[uuid] = callback;\n\n        // When the iframe has loaded (after the server responds to an upload request)\n        // declare the attempt a failure if we don't receive a valid message shortly after the response comes in.\n        detachLoadEvents[fileId] = qq(iframe).attach(\"load\", function() {\n            if (handler.getInput(fileId)) {\n                log(\"Received iframe load event for CORS upload request (iframe name \" + iframeName + \")\");\n\n                postMessageCallbackTimers[iframeName] = setTimeout(function() {\n                    var errorMessage = \"No valid message received from loaded iframe for iframe name \" + iframeName;\n                    log(errorMessage, \"error\");\n                    callback({\n                        error: errorMessage\n                    });\n                }, 1000);\n            }\n        });\n\n        // Listen for messages coming from this iframe.  When a message has been received, cancel the timer\n        // that declares the upload a failure if a message is not received within a reasonable amount of time.\n        corsMessageReceiver.receiveMessage(iframeName, function(message) {\n            log(\"Received the following window message: '\" + message + \"'\");\n            var fileId = getFileIdForIframeName(iframeName),\n                response = handler._parseJsonResponse(message),\n                uuid = response.uuid,\n                onloadCallback;\n\n            if (uuid && onloadCallbacks[uuid]) {\n                log(\"Handling response for iframe name \" + iframeName);\n                clearTimeout(postMessageCallbackTimers[iframeName]);\n                delete postMessageCallbackTimers[iframeName];\n\n                handler._detachLoadEvent(iframeName);\n\n                onloadCallback = onloadCallbacks[uuid];\n\n                delete onloadCallbacks[uuid];\n                corsMessageReceiver.stopReceivingMessages(iframeName);\n                onloadCallback(response);\n            }\n            else if (!uuid) {\n                log(\"'\" + message + \"' does not contain a UUID - ignoring.\");\n            }\n        });\n    }\n\n    qq.extend(this, new qq.UploadHandler(spec));\n\n    qq.override(this, function(super_) {\n        return {\n            /**\n             * Adds File or Blob to the queue\n             **/\n            add: function(id, fileInput) {\n                super_.add(id, {input: fileInput});\n\n                fileInput.setAttribute(\"name\", inputName);\n\n                // remove file input from DOM\n                if (fileInput.parentNode) {\n                    qq(fileInput).remove();\n                }\n            },\n\n            expunge: function(id) {\n                expungeFile(id);\n                super_.expunge(id);\n            },\n\n            isValid: function(id) {\n                return super_.isValid(id) &&\n                    handler._getFileState(id).input !== undefined;\n            }\n        };\n    });\n\n    qq.extend(this, {\n        getInput: function(id) {\n            return handler._getFileState(id).input;\n        },\n\n        /**\n         * This function either delegates to a more specific message handler if CORS is involved,\n         * or simply registers a callback when the iframe has been loaded that invokes the passed callback\n         * after determining if the content of the iframe is accessible.\n         *\n         * @param iframe Associated iframe\n         * @param callback Callback to invoke after we have determined if the iframe content is accessible.\n         */\n        _attachLoadEvent: function(iframe, callback) {\n            /*jslint eqeq: true*/\n            var responseDescriptor;\n\n            if (isCors) {\n                registerPostMessageCallback(iframe, callback);\n            }\n            else {\n                detachLoadEvents[iframe.id] = qq(iframe).attach(\"load\", function() {\n                    log(\"Received response for \" + iframe.id);\n\n                    // when we remove iframe from dom\n                    // the request stops, but in IE load\n                    // event fires\n                    if (!iframe.parentNode) {\n                        return;\n                    }\n\n                    try {\n                        // fixing Opera 10.53\n                        if (iframe.contentDocument &&\n                            iframe.contentDocument.body &&\n                            iframe.contentDocument.body.innerHTML == \"false\") {\n                            // In Opera event is fired second time\n                            // when body.innerHTML changed from false\n                            // to server response approx. after 1 sec\n                            // when we upload file with iframe\n                            return;\n                        }\n                    }\n                    catch (error) {\n                        //IE may throw an \"access is denied\" error when attempting to access contentDocument on the iframe in some cases\n                        log(\"Error when attempting to access iframe during handling of upload response (\" + error.message + \")\", \"error\");\n                        responseDescriptor = {success: false};\n                    }\n\n                    callback(responseDescriptor);\n                });\n            }\n        },\n\n        /**\n         * Creates an iframe with a specific document-unique name.\n         *\n         * @param id ID of the associated file\n         * @returns {HTMLIFrameElement}\n         */\n        _createIframe: function(id) {\n            var iframeName = handler._getIframeName(id);\n\n            return initIframeForUpload(iframeName);\n        },\n\n        /**\n         * Called when we are no longer interested in being notified when an iframe has loaded.\n         *\n         * @param id Associated file ID\n         */\n        _detachLoadEvent: function(id) {\n            if (detachLoadEvents[id] !== undefined) {\n                detachLoadEvents[id]();\n                delete detachLoadEvents[id];\n            }\n        },\n\n        /**\n         * @param fileId ID of the associated file\n         * @returns {string} The `document`-unique name of the iframe\n         */\n        _getIframeName: function(fileId) {\n            return fileId + \"_\" + formHandlerInstanceId;\n        },\n\n        /**\n         * Generates a form element and appends it to the `document`.  When the form is submitted, a specific iframe is targeted.\n         * The name of the iframe is passed in as a property of the spec parameter, and must be unique in the `document`.  Note\n         * that the form is hidden from view.\n         *\n         * @param spec An object containing various properties to be used when constructing the form.  Required properties are\n         * currently: `method`, `endpoint`, `params`, `paramsInBody`, and `targetName`.\n         * @returns {HTMLFormElement} The created form\n         */\n        _initFormForUpload: function(spec) {\n            var method = spec.method,\n                endpoint = spec.endpoint,\n                params = spec.params,\n                paramsInBody = spec.paramsInBody,\n                targetName = spec.targetName,\n                form = qq.toElement(\"<form method='\" + method + \"' enctype='multipart/form-data'></form>\"),\n                url = endpoint;\n\n            if (paramsInBody) {\n                qq.obj2Inputs(params, form);\n            }\n            else {\n                url = qq.obj2url(params, endpoint);\n            }\n\n            form.setAttribute(\"action\", url);\n            form.setAttribute(\"target\", targetName);\n            form.style.display = \"none\";\n            document.body.appendChild(form);\n\n            return form;\n        },\n\n        /**\n         * @param innerHtmlOrMessage JSON message\n         * @returns {*} The parsed response, or an empty object if the response could not be parsed\n         */\n        _parseJsonResponse: function(innerHtmlOrMessage) {\n            var response = {};\n\n            try {\n                response = qq.parseJson(innerHtmlOrMessage);\n            }\n            catch (error) {\n                log(\"Error when attempting to parse iframe upload response (\" + error.message + \")\", \"error\");\n            }\n\n            return response;\n        }\n    });\n};\n","/* globals qq */\n/**\n * Common API exposed to creators of XHR handlers.  This is reused and possibly overriding in some cases by specific\n * XHR upload handlers.\n *\n * @constructor\n */\nqq.XhrUploadHandler = function(spec) {\n    \"use strict\";\n\n    var handler = this,\n        namespace = spec.options.namespace,\n        proxy = spec.proxy,\n        chunking = spec.options.chunking,\n        getChunkSize = function(id) {\n            var fileState = handler._getFileState(id);\n\n            if (fileState.chunkSize) {\n                return fileState.chunkSize;\n            }\n\n            else {\n                var chunkSize = chunking.partSize;\n\n                if (qq.isFunction(chunkSize)) {\n                    chunkSize = chunkSize(id, getSize(id));\n                }\n\n                fileState.chunkSize = chunkSize;\n                return chunkSize;\n            }\n        },\n        resume = spec.options.resume,\n        chunkFiles = chunking && spec.options.chunking.enabled && qq.supportedFeatures.chunking,\n        resumeEnabled = resume && spec.options.resume.enabled && chunkFiles && qq.supportedFeatures.resume,\n        getName = proxy.getName,\n        getSize = proxy.getSize,\n        getUuid = proxy.getUuid,\n        getEndpoint = proxy.getEndpoint,\n        getDataByUuid = proxy.getDataByUuid,\n        onUuidChanged = proxy.onUuidChanged,\n        onProgress = proxy.onProgress,\n        log = proxy.log,\n        getCustomResumeData = proxy.getCustomResumeData;\n\n    function abort(id) {\n        qq.each(handler._getXhrs(id), function(xhrId, xhr) {\n            var ajaxRequester = handler._getAjaxRequester(id, xhrId);\n\n            xhr.onreadystatechange = null;\n            xhr.upload.onprogress = null;\n            xhr.abort();\n            ajaxRequester && ajaxRequester.canceled && ajaxRequester.canceled(id);\n        });\n    }\n\n    qq.extend(this, new qq.UploadHandler(spec));\n\n    qq.override(this, function(super_) {\n        return {\n            /**\n             * Adds File or Blob to the queue\n             **/\n            add: function(id, blobOrProxy) {\n                if (qq.isFile(blobOrProxy) || qq.isBlob(blobOrProxy)) {\n                    super_.add(id, {file: blobOrProxy});\n                }\n                else if (blobOrProxy instanceof qq.BlobProxy) {\n                    super_.add(id, {proxy: blobOrProxy});\n                }\n                else {\n                    throw new Error(\"Passed obj is not a File, Blob, or proxy\");\n                }\n\n                handler._initTempState(id);\n                resumeEnabled && handler._maybePrepareForResume(id);\n            },\n\n            expunge: function(id) {\n                abort(id);\n                handler._maybeDeletePersistedChunkData(id);\n                handler._clearXhrs(id);\n                super_.expunge(id);\n            }\n        };\n    });\n\n    qq.extend(this, {\n        // Clear the cached chunk `Blob` after we are done with it, just in case the `Blob` bytes are stored in memory.\n        clearCachedChunk: function(id, chunkIdx) {\n            var fileState = handler._getFileState(id);\n\n            if (fileState) {\n                delete fileState.temp.cachedChunks[chunkIdx];\n            }\n        },\n\n        clearXhr: function(id, chunkIdx) {\n            var tempState = handler._getFileState(id).temp;\n\n            if (tempState.xhrs) {\n                delete tempState.xhrs[chunkIdx];\n            }\n            if (tempState.ajaxRequesters) {\n                delete tempState.ajaxRequesters[chunkIdx];\n            }\n        },\n\n        // Called when all chunks have been successfully uploaded.  Expected promissory return type.\n        // This defines the default behavior if nothing further is required when all chunks have been uploaded.\n        finalizeChunks: function(id, responseParser) {\n            var lastChunkIdx = handler._getTotalChunks(id) - 1,\n                xhr = handler._getXhr(id, lastChunkIdx);\n\n            if (responseParser) {\n                return new qq.Promise().success(responseParser(xhr), xhr);\n            }\n\n            return new qq.Promise().success({}, xhr);\n        },\n\n        getFile: function(id) {\n            return handler.isValid(id) && handler._getFileState(id).file;\n        },\n\n        getProxy: function(id) {\n            return handler.isValid(id) && handler._getFileState(id).proxy;\n        },\n\n        /**\n         * @returns {Array} Array of objects containing properties useful to integrators\n         * when it is important to determine which files are potentially resumable.\n         */\n        getResumableFilesData: function() {\n            var resumableFilesData = [];\n\n            handler._iterateResumeRecords(function(key, uploadData) {\n                handler.moveInProgressToRemaining(null, uploadData.chunking.inProgress,  uploadData.chunking.remaining);\n\n                var data = {\n                    name: uploadData.name,\n                    remaining: uploadData.chunking.remaining,\n                    size: uploadData.size,\n                    uuid: uploadData.uuid\n                };\n\n                if (uploadData.key) {\n                    data.key = uploadData.key;\n                }\n\n                if (uploadData.customResumeData) {\n                    data.customResumeData = uploadData.customResumeData;\n                }\n\n                resumableFilesData.push(data);\n            });\n\n            return resumableFilesData;\n        },\n\n        isAttemptingResume: function(id) {\n            return handler._getFileState(id).attemptingResume;\n        },\n\n        isResumable: function(id) {\n            return !!chunking && handler.isValid(id) &&\n                !handler._getFileState(id).notResumable;\n        },\n\n        moveInProgressToRemaining: function(id, optInProgress, optRemaining) {\n            var fileState = handler._getFileState(id) || {},\n                chunkingState =  fileState.chunking || {},\n                inProgress = optInProgress || chunkingState.inProgress,\n                remaining = optRemaining || chunkingState.remaining;\n\n            if (inProgress) {\n                log(qq.format(\"Moving these chunks from in-progress {}, to remaining.\", JSON.stringify(inProgress)));\n                inProgress.reverse();\n                qq.each(inProgress, function(idx, chunkIdx) {\n                    remaining.unshift(chunkIdx);\n                });\n                inProgress.length = 0;\n            }\n        },\n\n        pause: function(id) {\n            if (handler.isValid(id)) {\n                log(qq.format(\"Aborting XHR upload for {} '{}' due to pause instruction.\", id, getName(id)));\n                handler._getFileState(id).paused = true;\n                abort(id);\n                return true;\n            }\n        },\n\n        reevaluateChunking: function(id) {\n            if (chunking && handler.isValid(id)) {\n                var state = handler._getFileState(id),\n                    totalChunks,\n                    i;\n\n                delete state.chunking;\n\n                state.chunking = {};\n                totalChunks = handler._getTotalChunks(id);\n                if (totalChunks > 1 || chunking.mandatory) {\n                    state.chunking.enabled = true;\n                    state.chunking.parts = totalChunks;\n                    state.chunking.remaining = [];\n\n                    for (i = 0; i < totalChunks; i++) {\n                        state.chunking.remaining.push(i);\n                    }\n\n                    handler._initTempState(id);\n                }\n                else {\n                    state.chunking.enabled = false;\n                }\n            }\n        },\n\n        updateBlob: function(id, newBlob) {\n            if (handler.isValid(id)) {\n                handler._getFileState(id).file = newBlob;\n            }\n        },\n\n        _clearXhrs: function(id) {\n            var tempState = handler._getFileState(id).temp;\n\n            qq.each(tempState.ajaxRequesters, function(chunkId) {\n                delete tempState.ajaxRequesters[chunkId];\n            });\n\n            qq.each(tempState.xhrs, function(chunkId) {\n                delete tempState.xhrs[chunkId];\n            });\n        },\n\n        /**\n         * Creates an XHR instance for this file and stores it in the fileState.\n         *\n         * @param id File ID\n         * @param optChunkIdx The chunk index associated with this XHR, if applicable\n         * @returns {XMLHttpRequest}\n         */\n        _createXhr: function(id, optChunkIdx) {\n            return handler._registerXhr(id, optChunkIdx, qq.createXhrInstance());\n        },\n\n        _getAjaxRequester: function(id, optChunkIdx) {\n            var chunkIdx = optChunkIdx == null ? -1 : optChunkIdx;\n            return handler._getFileState(id).temp.ajaxRequesters[chunkIdx];\n        },\n\n        _getChunkData: function(id, chunkIndex) {\n            var chunkSize = getChunkSize(id),\n                fileSize = getSize(id),\n                fileOrBlob = handler.getFile(id),\n                startBytes = chunkSize * chunkIndex,\n                endBytes = startBytes + chunkSize >= fileSize ? fileSize : startBytes + chunkSize,\n                totalChunks = handler._getTotalChunks(id),\n                cachedChunks = this._getFileState(id).temp.cachedChunks,\n\n            // To work around a Webkit GC bug, we must keep each chunk `Blob` in scope until we are done with it.\n            // See https://github.com/FineUploader/fine-uploader/issues/937#issuecomment-41418760\n                blob = cachedChunks[chunkIndex] || qq.sliceBlob(fileOrBlob, startBytes, endBytes);\n\n            cachedChunks[chunkIndex] = blob;\n\n            return {\n                part: chunkIndex,\n                start: startBytes,\n                end: endBytes,\n                count: totalChunks,\n                blob: blob,\n                size: endBytes - startBytes\n            };\n        },\n\n        _getChunkDataForCallback: function(chunkData) {\n            return {\n                partIndex: chunkData.part,\n                startByte: chunkData.start + 1,\n                endByte: chunkData.end,\n                totalParts: chunkData.count\n            };\n        },\n\n        /**\n         * @param id File ID\n         * @returns {string} Identifier for this item that may appear in the browser's local storage\n         */\n        _getLocalStorageId: function(id) {\n            var formatVersion = \"5.0\",\n                name = getName(id),\n                size = getSize(id),\n                chunkSize = getChunkSize(id),\n                endpoint = getEndpoint(id),\n                customKeys = resume.customKeys(id),\n                localStorageId = qq.format(\"qq{}resume{}-{}-{}-{}-{}\", namespace, formatVersion, name, size, chunkSize, endpoint);\n\n            customKeys.forEach(function(key) {\n                localStorageId += \"-\" + key;\n            });\n\n            return localStorageId;\n        },\n\n        _getMimeType: function(id) {\n            return handler.getFile(id).type;\n        },\n\n        _getPersistableData: function(id) {\n            return handler._getFileState(id).chunking;\n        },\n\n        /**\n         * @param id ID of the associated file\n         * @returns {number} Number of parts this file can be divided into, or undefined if chunking is not supported in this UA\n         */\n        _getTotalChunks: function(id) {\n            if (chunking) {\n                var fileSize = getSize(id),\n                    chunkSize = getChunkSize(id);\n\n                return Math.ceil(fileSize / chunkSize);\n            }\n        },\n\n        _getXhr: function(id, optChunkIdx) {\n            var chunkIdx = optChunkIdx == null ? -1 : optChunkIdx;\n            return handler._getFileState(id).temp.xhrs[chunkIdx];\n        },\n\n        _getXhrs: function(id) {\n            return handler._getFileState(id).temp.xhrs;\n        },\n\n        // Iterates through all XHR handler-created resume records (in local storage),\n        // invoking the passed callback and passing in the key and value of each local storage record.\n        _iterateResumeRecords: function(callback) {\n            if (resumeEnabled) {\n                qq.each(localStorage, function(key, item) {\n                    if (key.indexOf(qq.format(\"qq{}resume\", namespace)) === 0) {\n                        var uploadData = JSON.parse(item);\n                        callback(key, uploadData);\n                    }\n                });\n            }\n        },\n\n        _initTempState: function(id) {\n            handler._getFileState(id).temp = {\n                ajaxRequesters: {},\n                chunkProgress: {},\n                xhrs: {},\n                cachedChunks: {}\n            };\n        },\n\n        _markNotResumable: function(id) {\n            handler._getFileState(id).notResumable = true;\n        },\n\n        // Removes a chunked upload record from local storage, if possible.\n        // Returns true if the item was removed, false otherwise.\n        _maybeDeletePersistedChunkData: function(id) {\n            var localStorageId;\n\n            if (resumeEnabled && handler.isResumable(id)) {\n                localStorageId = handler._getLocalStorageId(id);\n\n                if (localStorageId && localStorage.getItem(localStorageId)) {\n                    localStorage.removeItem(localStorageId);\n                    return true;\n                }\n            }\n\n            return false;\n        },\n\n        // If this is a resumable upload, grab the relevant data from storage and items in memory that track this upload\n        // so we can pick up from where we left off.\n        _maybePrepareForResume: function(id) {\n            var state = handler._getFileState(id),\n                localStorageId, persistedData;\n\n            // Resume is enabled and possible and this is the first time we've tried to upload this file in this session,\n            // so prepare for a resume attempt.\n            if (resumeEnabled && state.key === undefined) {\n                localStorageId = handler._getLocalStorageId(id);\n                persistedData = localStorage.getItem(localStorageId);\n\n                // If we found this item in local storage, maybe we should resume it.\n                if (persistedData) {\n                    persistedData = JSON.parse(persistedData);\n\n                    // If we found a resume record but we have already handled this file in this session,\n                    // don't try to resume it & ensure we don't persist future check data\n                    if (getDataByUuid(persistedData.uuid)) {\n                        handler._markNotResumable(id);\n                    }\n                    else {\n                        log(qq.format(\"Identified file with ID {} and name of {} as resumable.\", id, getName(id)));\n\n                        onUuidChanged(id, persistedData.uuid);\n\n                        state.key = persistedData.key;\n                        state.chunking = persistedData.chunking;\n                        state.loaded = persistedData.loaded;\n                        state.customResumeData = persistedData.customResumeData;\n                        state.attemptingResume = true;\n\n                        handler.moveInProgressToRemaining(id);\n                    }\n                }\n            }\n        },\n\n        // Persist any data needed to resume this upload in a new session.\n        _maybePersistChunkedState: function(id) {\n            var state = handler._getFileState(id),\n                localStorageId, persistedData;\n\n            // If local storage isn't supported by the browser, or if resume isn't enabled or possible, give up\n            if (resumeEnabled && handler.isResumable(id)) {\n                var customResumeData = getCustomResumeData(id);\n\n                localStorageId = handler._getLocalStorageId(id);\n\n                persistedData = {\n                    name: getName(id),\n                    size: getSize(id),\n                    uuid: getUuid(id),\n                    key: state.key,\n                    chunking: state.chunking,\n                    loaded: state.loaded,\n                    lastUpdated: Date.now(),\n                };\n\n                if (customResumeData) {\n                    persistedData.customResumeData = customResumeData;\n                }\n\n                try {\n                    localStorage.setItem(localStorageId, JSON.stringify(persistedData));\n                }\n                catch (error) {\n                    log(qq.format(\"Unable to save resume data for '{}' due to error: '{}'.\", id, error.toString()), \"warn\");\n                }\n            }\n        },\n\n        _registerProgressHandler: function(id, chunkIdx, chunkSize) {\n            var xhr = handler._getXhr(id, chunkIdx),\n                name = getName(id),\n                progressCalculator = {\n                    simple: function(loaded, total) {\n                        var fileSize = getSize(id);\n\n                        if (loaded === total) {\n                            onProgress(id, name, fileSize, fileSize);\n                        }\n                        else {\n                            onProgress(id, name, (loaded >= fileSize ? fileSize - 1 : loaded), fileSize);\n                        }\n                    },\n\n                    chunked: function(loaded, total) {\n                        var chunkProgress = handler._getFileState(id).temp.chunkProgress,\n                            totalSuccessfullyLoadedForFile = handler._getFileState(id).loaded,\n                            loadedForRequest = loaded,\n                            totalForRequest = total,\n                            totalFileSize = getSize(id),\n                            estActualChunkLoaded = loadedForRequest - (totalForRequest - chunkSize),\n                            totalLoadedForFile = totalSuccessfullyLoadedForFile;\n\n                        chunkProgress[chunkIdx] = estActualChunkLoaded;\n\n                        qq.each(chunkProgress, function(chunkIdx, chunkLoaded) {\n                            totalLoadedForFile += chunkLoaded;\n                        });\n\n                        onProgress(id, name, totalLoadedForFile, totalFileSize);\n                    }\n                };\n\n            xhr.upload.onprogress = function(e) {\n                if (e.lengthComputable) {\n                    /* jshint eqnull: true */\n                    var type = chunkSize == null ? \"simple\" : \"chunked\";\n                    progressCalculator[type](e.loaded, e.total);\n                }\n            };\n        },\n\n        /**\n         * Registers an XHR transport instance created elsewhere.\n         *\n         * @param id ID of the associated file\n         * @param optChunkIdx The chunk index associated with this XHR, if applicable\n         * @param xhr XMLHttpRequest object instance\n         * @param optAjaxRequester `qq.AjaxRequester` associated with this request, if applicable.\n         * @returns {XMLHttpRequest}\n         */\n        _registerXhr: function(id, optChunkIdx, xhr, optAjaxRequester) {\n            var xhrsId = optChunkIdx == null ? -1 : optChunkIdx,\n                tempState = handler._getFileState(id).temp;\n\n            tempState.xhrs = tempState.xhrs || {};\n            tempState.ajaxRequesters = tempState.ajaxRequesters || {};\n\n            tempState.xhrs[xhrsId] = xhr;\n\n            if (optAjaxRequester) {\n                tempState.ajaxRequesters[xhrsId] = optAjaxRequester;\n            }\n\n            return xhr;\n        },\n\n        // Deletes any local storage records that are \"expired\".\n        _removeExpiredChunkingRecords: function() {\n            var expirationDays = resume.recordsExpireIn;\n\n            handler._iterateResumeRecords(function(key, uploadData) {\n                var expirationDate = new Date(uploadData.lastUpdated);\n\n                // transform updated date into expiration date\n                expirationDate.setDate(expirationDate.getDate() + expirationDays);\n\n                if (expirationDate.getTime() <= Date.now()) {\n                    log(\"Removing expired resume record with key \" + key);\n                    localStorage.removeItem(key);\n                }\n            });\n        },\n\n        /**\n         * Determine if the associated file should be chunked.\n         *\n         * @param id ID of the associated file\n         * @returns {*} true if chunking is enabled, possible, and the file can be split into more than 1 part\n         */\n        _shouldChunkThisFile: function(id) {\n            var state = handler._getFileState(id);\n\n            // file may no longer be available if it was recently cancelled\n            if (state) {\n                if (!state.chunking) {\n                    handler.reevaluateChunking(id);\n                }\n\n                return state.chunking.enabled;\n            }\n        }\n    });\n};\n","/*globals qq, XMLHttpRequest*/\nqq.DeleteFileAjaxRequester = function(o) {\n    \"use strict\";\n\n    var requester,\n        options = {\n            method: \"DELETE\",\n            uuidParamName: \"qquuid\",\n            endpointStore: {},\n            maxConnections: 3,\n            customHeaders: function(id) {return {};},\n            paramsStore: {},\n            cors: {\n                expected: false,\n                sendCredentials: false\n            },\n            log: function(str, level) {},\n            onDelete: function(id) {},\n            onDeleteComplete: function(id, xhrOrXdr, isError) {}\n        };\n\n    qq.extend(options, o);\n\n    function getMandatedParams() {\n        if (options.method.toUpperCase() === \"POST\") {\n            return {\n                _method: \"DELETE\"\n            };\n        }\n\n        return {};\n    }\n\n    requester = qq.extend(this, new qq.AjaxRequester({\n        acceptHeader: \"application/json\",\n        validMethods: [\"POST\", \"DELETE\"],\n        method: options.method,\n        endpointStore: options.endpointStore,\n        paramsStore: options.paramsStore,\n        mandatedParams: getMandatedParams(),\n        maxConnections: options.maxConnections,\n        customHeaders: function(id) {\n            return options.customHeaders.get(id);\n        },\n        log: options.log,\n        onSend: options.onDelete,\n        onComplete: options.onDeleteComplete,\n        cors: options.cors\n    }));\n\n    qq.extend(this, {\n        sendDelete: function(id, uuid, additionalMandatedParams) {\n            var additionalOptions = additionalMandatedParams || {};\n\n            options.log(\"Submitting delete file request for \" + id);\n\n            if (options.method === \"DELETE\") {\n                requester.initTransport(id)\n                    .withPath(uuid)\n                    .withParams(additionalOptions)\n                    .send();\n            }\n            else {\n                additionalOptions[options.uuidParamName] = uuid;\n                requester.initTransport(id)\n                    .withParams(additionalOptions)\n                    .send();\n            }\n        }\n    });\n};\n","/*global qq, define */\n/*jshint strict:false,bitwise:false,nonew:false,asi:true,-W064,-W116,-W089 */\n/**\n * Mega pixel image rendering library for iOS6+\n *\n * Fixes iOS6+'s image file rendering issue for large size image (over mega-pixel),\n * which causes unexpected subsampling when drawing it in canvas.\n * By using this library, you can safely render the image with proper stretching.\n *\n * Copyright (c) 2012 Shinichi Tomita <shinichi.tomita@gmail.com>\n * Released under the MIT license\n *\n * Heavily modified by Widen for Fine Uploader\n */\n(function() {\n\n    /**\n     * Detect subsampling in loaded image.\n     * In iOS, larger images than 2M pixels may be subsampled in rendering.\n     */\n    function detectSubsampling(img) {\n        var iw = img.naturalWidth,\n            ih = img.naturalHeight,\n            canvas = document.createElement(\"canvas\"),\n            ctx;\n\n        if (iw * ih > 1024 * 1024) { // subsampling may happen over megapixel image\n            canvas.width = canvas.height = 1;\n            ctx = canvas.getContext(\"2d\");\n            ctx.drawImage(img, -iw + 1, 0);\n            // subsampled image becomes half smaller in rendering size.\n            // check alpha channel value to confirm image is covering edge pixel or not.\n            // if alpha value is 0 image is not covering, hence subsampled.\n            return ctx.getImageData(0, 0, 1, 1).data[3] === 0;\n        } else {\n            return false;\n        }\n    }\n\n    /**\n     * Detecting vertical squash in loaded image.\n     * Fixes a bug which squash image vertically while drawing into canvas for some images.\n     */\n    function detectVerticalSquash(img, iw, ih) {\n        var canvas = document.createElement(\"canvas\"),\n            sy = 0,\n            ey = ih,\n            py = ih,\n            ctx, data, alpha, ratio;\n\n        canvas.width = 1;\n        canvas.height = ih;\n        ctx = canvas.getContext(\"2d\");\n        ctx.drawImage(img, 0, 0);\n        data = ctx.getImageData(0, 0, 1, ih).data;\n\n        // search image edge pixel position in case it is squashed vertically.\n        while (py > sy) {\n            alpha = data[(py - 1) * 4 + 3];\n            if (alpha === 0) {\n                ey = py;\n            } else {\n                sy = py;\n            }\n            py = (ey + sy) >> 1;\n        }\n\n        ratio = (py / ih);\n        return (ratio === 0) ? 1 : ratio;\n    }\n\n    /**\n     * Rendering image element (with resizing) and get its data URL\n     */\n    function renderImageToDataURL(img, blob, options, doSquash) {\n        var canvas = document.createElement(\"canvas\"),\n            mime = options.mime || \"image/jpeg\",\n            promise = new qq.Promise();\n\n        renderImageToCanvas(img, blob, canvas, options, doSquash)\n            .then(function() {\n                promise.success(\n                    canvas.toDataURL(mime, options.quality || 0.8)\n                );\n            });\n\n        return promise;\n    }\n\n    function maybeCalculateDownsampledDimensions(spec) {\n        var maxPixels = 5241000; //iOS specific value\n\n        if (!qq.ios()) {\n            throw new qq.Error(\"Downsampled dimensions can only be reliably calculated for iOS!\");\n        }\n\n        if (spec.origHeight * spec.origWidth > maxPixels) {\n            return {\n                newHeight: Math.round(Math.sqrt(maxPixels * (spec.origHeight / spec.origWidth))),\n                newWidth: Math.round(Math.sqrt(maxPixels * (spec.origWidth / spec.origHeight)))\n            };\n        }\n    }\n\n    /**\n     * Rendering image element (with resizing) into the canvas element\n     */\n    function renderImageToCanvas(img, blob, canvas, options, doSquash) {\n        var iw = img.naturalWidth,\n            ih = img.naturalHeight,\n            width = options.width,\n            height = options.height,\n            ctx = canvas.getContext(\"2d\"),\n            promise = new qq.Promise(),\n            modifiedDimensions;\n\n        ctx.save();\n\n        if (options.resize) {\n            return renderImageToCanvasWithCustomResizer({\n                blob: blob,\n                canvas: canvas,\n                image: img,\n                imageHeight: ih,\n                imageWidth: iw,\n                orientation: options.orientation,\n                resize: options.resize,\n                targetHeight: height,\n                targetWidth: width\n            });\n        }\n\n        if (!qq.supportedFeatures.unlimitedScaledImageSize) {\n            modifiedDimensions = maybeCalculateDownsampledDimensions({\n                origWidth: width,\n                origHeight: height\n            });\n\n            if (modifiedDimensions) {\n                qq.log(qq.format(\"Had to reduce dimensions due to device limitations from {}w / {}h to {}w / {}h\",\n                    width, height, modifiedDimensions.newWidth, modifiedDimensions.newHeight),\n                    \"warn\");\n\n                width = modifiedDimensions.newWidth;\n                height = modifiedDimensions.newHeight;\n            }\n        }\n\n        transformCoordinate(canvas, width, height, options.orientation);\n\n        // Fine Uploader specific: Save some CPU cycles if not using iOS\n        // Assumption: This logic is only needed to overcome iOS image sampling issues\n        if (qq.ios()) {\n            (function() {\n                if (detectSubsampling(img)) {\n                    iw /= 2;\n                    ih /= 2;\n                }\n\n                var d = 1024, // size of tiling canvas\n                    tmpCanvas = document.createElement(\"canvas\"),\n                    vertSquashRatio = doSquash ? detectVerticalSquash(img, iw, ih) : 1,\n                    dw = Math.ceil(d * width / iw),\n                    dh = Math.ceil(d * height / ih / vertSquashRatio),\n                    sy = 0,\n                    dy = 0,\n                    tmpCtx, sx, dx;\n\n                tmpCanvas.width = tmpCanvas.height = d;\n                tmpCtx = tmpCanvas.getContext(\"2d\");\n\n                while (sy < ih) {\n                    sx = 0;\n                    dx = 0;\n                    while (sx < iw) {\n                        tmpCtx.clearRect(0, 0, d, d);\n                        tmpCtx.drawImage(img, -sx, -sy);\n                        ctx.drawImage(tmpCanvas, 0, 0, d, d, dx, dy, dw, dh);\n                        sx += d;\n                        dx += dw;\n                    }\n                    sy += d;\n                    dy += dh;\n                }\n                ctx.restore();\n                tmpCanvas = tmpCtx = null;\n            }());\n        }\n        else {\n            ctx.drawImage(img, 0, 0, width, height);\n        }\n\n        canvas.qqImageRendered && canvas.qqImageRendered();\n        promise.success();\n\n        return promise;\n    }\n\n    function renderImageToCanvasWithCustomResizer(resizeInfo) {\n        var blob = resizeInfo.blob,\n            image = resizeInfo.image,\n            imageHeight = resizeInfo.imageHeight,\n            imageWidth = resizeInfo.imageWidth,\n            orientation = resizeInfo.orientation,\n            promise = new qq.Promise(),\n            resize = resizeInfo.resize,\n            sourceCanvas = document.createElement(\"canvas\"),\n            sourceCanvasContext = sourceCanvas.getContext(\"2d\"),\n            targetCanvas = resizeInfo.canvas,\n            targetHeight = resizeInfo.targetHeight,\n            targetWidth = resizeInfo.targetWidth;\n\n        transformCoordinate(sourceCanvas, imageWidth, imageHeight, orientation);\n\n        targetCanvas.height = targetHeight;\n        targetCanvas.width = targetWidth;\n\n        sourceCanvasContext.drawImage(image, 0, 0);\n\n        resize({\n            blob: blob,\n            height: targetHeight,\n            image: image,\n            sourceCanvas: sourceCanvas,\n            targetCanvas: targetCanvas,\n            width: targetWidth\n        })\n            .then(\n                function success() {\n                    targetCanvas.qqImageRendered && targetCanvas.qqImageRendered();\n                    promise.success();\n                },\n                promise.failure\n            );\n\n        return promise;\n    }\n\n    /**\n     * Transform canvas coordination according to specified frame size and orientation\n     * Orientation value is from EXIF tag\n     */\n    function transformCoordinate(canvas, width, height, orientation) {\n        switch (orientation) {\n            case 5:\n            case 6:\n            case 7:\n            case 8:\n                canvas.width = height;\n                canvas.height = width;\n                break;\n            default:\n                canvas.width = width;\n                canvas.height = height;\n        }\n        var ctx = canvas.getContext(\"2d\");\n        switch (orientation) {\n            case 2:\n                // horizontal flip\n                ctx.translate(width, 0);\n                ctx.scale(-1, 1);\n                break;\n            case 3:\n                // 180 rotate left\n                ctx.translate(width, height);\n                ctx.rotate(Math.PI);\n                break;\n            case 4:\n                // vertical flip\n                ctx.translate(0, height);\n                ctx.scale(1, -1);\n                break;\n            case 5:\n                // vertical flip + 90 rotate right\n                ctx.rotate(0.5 * Math.PI);\n                ctx.scale(1, -1);\n                break;\n            case 6:\n                // 90 rotate right\n                ctx.rotate(0.5 * Math.PI);\n                ctx.translate(0, -height);\n                break;\n            case 7:\n                // horizontal flip + 90 rotate right\n                ctx.rotate(0.5 * Math.PI);\n                ctx.translate(width, -height);\n                ctx.scale(-1, 1);\n                break;\n            case 8:\n                // 90 rotate left\n                ctx.rotate(-0.5 * Math.PI);\n                ctx.translate(-width, 0);\n                break;\n            default:\n                break;\n        }\n    }\n\n    /**\n     * MegaPixImage class\n     */\n    function MegaPixImage(srcImage, errorCallback) {\n        var self = this;\n\n        if (window.Blob && srcImage instanceof Blob) {\n            (function() {\n                var img = new Image(),\n                    URL = window.URL && window.URL.createObjectURL ? window.URL :\n                        window.webkitURL && window.webkitURL.createObjectURL ? window.webkitURL : null;\n                if (!URL) { throw Error(\"No createObjectURL function found to create blob url\"); }\n                img.src = URL.createObjectURL(srcImage);\n                self.blob = srcImage;\n                srcImage = img;\n            }());\n        }\n        if (!srcImage.naturalWidth && !srcImage.naturalHeight) {\n            srcImage.onload = function() {\n                var listeners = self.imageLoadListeners;\n                if (listeners) {\n                    self.imageLoadListeners = null;\n                    // IE11 doesn't reliably report actual image dimensions immediately after onload for small files,\n                    // so let's push this to the end of the UI thread queue.\n                    setTimeout(function() {\n                        for (var i = 0, len = listeners.length; i < len; i++) {\n                            listeners[i]();\n                        }\n                    }, 0);\n                }\n            };\n            srcImage.onerror = errorCallback;\n            this.imageLoadListeners = [];\n        }\n        this.srcImage = srcImage;\n    }\n\n    /**\n     * Rendering megapix image into specified target element\n     */\n    MegaPixImage.prototype.render = function(target, options) {\n        options = options || {};\n\n        var self = this,\n            imgWidth = this.srcImage.naturalWidth,\n            imgHeight = this.srcImage.naturalHeight,\n            width = options.width,\n            height = options.height,\n            maxWidth = options.maxWidth,\n            maxHeight = options.maxHeight,\n            doSquash = !this.blob || this.blob.type === \"image/jpeg\",\n            tagName = target.tagName.toLowerCase(),\n            opt;\n\n        if (this.imageLoadListeners) {\n            this.imageLoadListeners.push(function() { self.render(target, options); });\n            return;\n        }\n\n        if (width && !height) {\n            height = (imgHeight * width / imgWidth) << 0;\n        } else if (height && !width) {\n            width = (imgWidth * height / imgHeight) << 0;\n        } else {\n            width = imgWidth;\n            height = imgHeight;\n        }\n        if (maxWidth && width > maxWidth) {\n            width = maxWidth;\n            height = (imgHeight * width / imgWidth) << 0;\n        }\n        if (maxHeight && height > maxHeight) {\n            height = maxHeight;\n            width = (imgWidth * height / imgHeight) << 0;\n        }\n\n        opt = { width: width, height: height },\n        qq.each(options, function(optionsKey, optionsValue) {\n            opt[optionsKey] = optionsValue;\n        });\n\n        if (tagName === \"img\") {\n            (function() {\n                var oldTargetSrc = target.src;\n                renderImageToDataURL(self.srcImage, self.blob, opt, doSquash)\n                    .then(function(dataUri) {\n                        target.src = dataUri;\n                        oldTargetSrc === target.src && target.onload();\n                    });\n            }());\n        } else if (tagName === \"canvas\") {\n            renderImageToCanvas(this.srcImage, this.blob, target, opt, doSquash);\n        }\n        if (typeof this.onrender === \"function\") {\n            this.onrender(target);\n        }\n    };\n\n    qq.MegaPixImage = MegaPixImage;\n})();\n","/*globals qq */\n/**\n * Draws a thumbnail of a Blob/File/URL onto an <img> or <canvas>.\n *\n * @constructor\n */\nqq.ImageGenerator = function(log) {\n    \"use strict\";\n\n    function isImg(el) {\n        return el.tagName.toLowerCase() === \"img\";\n    }\n\n    function isCanvas(el) {\n        return el.tagName.toLowerCase() === \"canvas\";\n    }\n\n    function isImgCorsSupported() {\n        return new Image().crossOrigin !== undefined;\n    }\n\n    function isCanvasSupported() {\n        var canvas = document.createElement(\"canvas\");\n\n        return canvas.getContext && canvas.getContext(\"2d\");\n    }\n\n    // This is only meant to determine the MIME type of a renderable image file.\n    // It is used to ensure images drawn from a URL that have transparent backgrounds\n    // are rendered correctly, among other things.\n    function determineMimeOfFileName(nameWithPath) {\n        /*jshint -W015 */\n        var pathSegments = nameWithPath.split(\"/\"),\n            name = pathSegments[pathSegments.length - 1].split(\"?\")[0],\n            extension = qq.getExtension(name);\n\n        extension = extension && extension.toLowerCase();\n\n        switch (extension) {\n            case \"jpeg\":\n            case \"jpg\":\n                return \"image/jpeg\";\n            case \"png\":\n                return \"image/png\";\n            case \"bmp\":\n                return \"image/bmp\";\n            case \"gif\":\n                return \"image/gif\";\n            case \"tiff\":\n            case \"tif\":\n                return \"image/tiff\";\n        }\n    }\n\n    // This will likely not work correctly in IE8 and older.\n    // It's only used as part of a formula to determine\n    // if a canvas can be used to scale a server-hosted thumbnail.\n    // If canvas isn't supported by the UA (IE8 and older)\n    // this method should not even be called.\n    function isCrossOrigin(url) {\n        var targetAnchor = document.createElement(\"a\"),\n            targetProtocol, targetHostname, targetPort;\n\n        targetAnchor.href = url;\n\n        targetProtocol = targetAnchor.protocol;\n        targetPort = targetAnchor.port;\n        targetHostname = targetAnchor.hostname;\n\n        if (targetProtocol.toLowerCase() !== window.location.protocol.toLowerCase()) {\n            return true;\n        }\n\n        if (targetHostname.toLowerCase() !== window.location.hostname.toLowerCase()) {\n            return true;\n        }\n\n        // IE doesn't take ports into consideration when determining if two endpoints are same origin.\n        if (targetPort !== window.location.port && !qq.ie()) {\n            return true;\n        }\n\n        return false;\n    }\n\n    function registerImgLoadListeners(img, promise) {\n        img.onload = function() {\n            img.onload = null;\n            img.onerror = null;\n            promise.success(img);\n        };\n\n        img.onerror = function() {\n            img.onload = null;\n            img.onerror = null;\n            log(\"Problem drawing thumbnail!\", \"error\");\n            promise.failure(img, \"Problem drawing thumbnail!\");\n        };\n    }\n\n    function registerCanvasDrawImageListener(canvas, promise) {\n        // The image is drawn on the canvas by a third-party library,\n        // and we want to know when this is completed.  Since the library\n        // may invoke drawImage many times in a loop, we need to be called\n        // back when the image is fully rendered.  So, we are expecting the\n        // code that draws this image to follow a convention that involves a\n        // function attached to the canvas instance be invoked when it is done.\n        canvas.qqImageRendered = function() {\n            promise.success(canvas);\n        };\n    }\n\n    // Fulfills a `qq.Promise` when an image has been drawn onto the target,\n    // whether that is a <canvas> or an <img>.  The attempt is considered a\n    // failure if the target is not an <img> or a <canvas>, or if the drawing\n    // attempt was not successful.\n    function registerThumbnailRenderedListener(imgOrCanvas, promise) {\n        var registered = isImg(imgOrCanvas) || isCanvas(imgOrCanvas);\n\n        if (isImg(imgOrCanvas)) {\n            registerImgLoadListeners(imgOrCanvas, promise);\n        }\n        else if (isCanvas(imgOrCanvas)) {\n            registerCanvasDrawImageListener(imgOrCanvas, promise);\n        }\n        else {\n            promise.failure(imgOrCanvas);\n            log(qq.format(\"Element container of type {} is not supported!\", imgOrCanvas.tagName), \"error\");\n        }\n\n        return registered;\n    }\n\n    // Draw a preview iff the current UA can natively display it.\n    // Also rotate the image if necessary.\n    function draw(fileOrBlob, container, options) {\n        var drawPreview = new qq.Promise(),\n            identifier = new qq.Identify(fileOrBlob, log),\n            maxSize = options.maxSize,\n            // jshint eqnull:true\n            orient = options.orient == null ? true : options.orient,\n            megapixErrorHandler = function() {\n                container.onerror = null;\n                container.onload = null;\n                log(\"Could not render preview, file may be too large!\", \"error\");\n                drawPreview.failure(container, \"Browser cannot render image!\");\n            };\n\n        identifier.isPreviewable().then(\n            function(mime) {\n                // If options explicitly specify that Orientation is not desired,\n                // replace the orient task with a dummy promise that \"succeeds\" immediately.\n                var dummyExif = {\n                        parse: function() {\n                            return new qq.Promise().success();\n                        }\n                    },\n                    exif = orient ? new qq.Exif(fileOrBlob, log) : dummyExif,\n                    mpImg = new qq.MegaPixImage(fileOrBlob, megapixErrorHandler);\n\n                if (registerThumbnailRenderedListener(container, drawPreview)) {\n                    exif.parse().then(\n                        function(exif) {\n                            var orientation = exif && exif.Orientation;\n\n                            mpImg.render(container, {\n                                maxWidth: maxSize,\n                                maxHeight: maxSize,\n                                orientation: orientation,\n                                mime: mime,\n                                resize: options.customResizeFunction\n                            });\n                        },\n\n                        function(failureMsg) {\n                            log(qq.format(\"EXIF data could not be parsed ({}).  Assuming orientation = 1.\", failureMsg));\n\n                            mpImg.render(container, {\n                                maxWidth: maxSize,\n                                maxHeight: maxSize,\n                                mime: mime,\n                                resize: options.customResizeFunction\n                            });\n                        }\n                    );\n                }\n            },\n\n            function() {\n                log(\"Not previewable\");\n                drawPreview.failure(container, \"Not previewable\");\n            }\n        );\n\n        return drawPreview;\n    }\n\n    function drawOnCanvasOrImgFromUrl(url, canvasOrImg, draw, maxSize, customResizeFunction) {\n        var tempImg = new Image(),\n            tempImgRender = new qq.Promise();\n\n        registerThumbnailRenderedListener(tempImg, tempImgRender);\n\n        if (isCrossOrigin(url)) {\n            tempImg.crossOrigin = \"anonymous\";\n        }\n\n        tempImg.src = url;\n\n        tempImgRender.then(\n            function rendered() {\n                registerThumbnailRenderedListener(canvasOrImg, draw);\n\n                var mpImg = new qq.MegaPixImage(tempImg);\n                mpImg.render(canvasOrImg, {\n                    maxWidth: maxSize,\n                    maxHeight: maxSize,\n                    mime: determineMimeOfFileName(url),\n                    resize: customResizeFunction\n                });\n            },\n\n            draw.failure\n        );\n    }\n\n    function drawOnImgFromUrlWithCssScaling(url, img, draw, maxSize) {\n        registerThumbnailRenderedListener(img, draw);\n        // NOTE: The fact that maxWidth/height is set on the thumbnail for scaled images\n        // that must drop back to CSS is known and exploited by the templating module.\n        // In this module, we pre-render \"waiting\" thumbs for all files immediately after they\n        // are submitted, and we must be sure to pass any style associated with the \"waiting\" preview.\n        qq(img).css({\n            maxWidth: maxSize + \"px\",\n            maxHeight: maxSize + \"px\"\n        });\n\n        img.src = url;\n    }\n\n    // Draw a (server-hosted) thumbnail given a URL.\n    // This will optionally scale the thumbnail as well.\n    // It attempts to use <canvas> to scale, but will fall back\n    // to max-width and max-height style properties if the UA\n    // doesn't support canvas or if the images is cross-domain and\n    // the UA doesn't support the crossorigin attribute on img tags,\n    // which is required to scale a cross-origin image using <canvas> &\n    // then export it back to an <img>.\n    function drawFromUrl(url, container, options) {\n        var draw = new qq.Promise(),\n            scale = options.scale,\n            maxSize = scale ? options.maxSize : null;\n\n        // container is an img, scaling needed\n        if (scale && isImg(container)) {\n            // Iff canvas is available in this UA, try to use it for scaling.\n            // Otherwise, fall back to CSS scaling\n            if (isCanvasSupported()) {\n                // Attempt to use <canvas> for image scaling,\n                // but we must fall back to scaling via CSS/styles\n                // if this is a cross-origin image and the UA doesn't support <img> CORS.\n                if (isCrossOrigin(url) && !isImgCorsSupported()) {\n                    drawOnImgFromUrlWithCssScaling(url, container, draw, maxSize);\n                }\n                else {\n                    drawOnCanvasOrImgFromUrl(url, container, draw, maxSize);\n                }\n            }\n            else {\n                drawOnImgFromUrlWithCssScaling(url, container, draw, maxSize);\n            }\n        }\n        // container is a canvas, scaling optional\n        else if (isCanvas(container)) {\n            drawOnCanvasOrImgFromUrl(url, container, draw, maxSize);\n        }\n        // container is an img & no scaling: just set the src attr to the passed url\n        else if (registerThumbnailRenderedListener(container, draw)) {\n            container.src = url;\n        }\n\n        return draw;\n    }\n\n    qq.extend(this, {\n        /**\n         * Generate a thumbnail.  Depending on the arguments, this may either result in\n         * a client-side rendering of an image (if a `Blob` is supplied) or a server-generated\n         * image that may optionally be scaled client-side using <canvas> or CSS/styles (as a fallback).\n         *\n         * @param fileBlobOrUrl a `File`, `Blob`, or a URL pointing to the image\n         * @param container <img> or <canvas> to contain the preview\n         * @param options possible properties include `maxSize` (int), `orient` (bool - default true), resize` (bool - default true), and `customResizeFunction`.\n         * @returns qq.Promise fulfilled when the preview has been drawn, or the attempt has failed\n         */\n        generate: function(fileBlobOrUrl, container, options) {\n            if (qq.isString(fileBlobOrUrl)) {\n                log(\"Attempting to update thumbnail based on server response.\");\n                return drawFromUrl(fileBlobOrUrl, container, options || {});\n            }\n            else {\n                log(\"Attempting to draw client-side image preview.\");\n                return draw(fileBlobOrUrl, container, options || {});\n            }\n        }\n    });\n\n    /*<testing>*/\n    this._testing = {};\n    this._testing.isImg = isImg;\n    this._testing.isCanvas = isCanvas;\n    this._testing.isCrossOrigin = isCrossOrigin;\n    this._testing.determineMimeOfFileName = determineMimeOfFileName;\n    /*</testing>*/\n};\n","/*globals qq */\n/**\n * EXIF image data parser.  Currently only parses the Orientation tag value,\n * but this may be expanded to other tags in the future.\n *\n * @param fileOrBlob Attempt to parse EXIF data in this `Blob`\n * @constructor\n */\nqq.Exif = function(fileOrBlob, log) {\n    \"use strict\";\n\n    // Orientation is the only tag parsed here at this time.\n    var TAG_IDS = [274],\n        TAG_INFO = {\n            274: {\n                name: \"Orientation\",\n                bytes: 2\n            }\n        };\n\n    // Convert a little endian (hex string) to big endian (decimal).\n    function parseLittleEndian(hex) {\n        var result = 0,\n            pow = 0;\n\n        while (hex.length > 0) {\n            result += parseInt(hex.substring(0, 2), 16) * Math.pow(2, pow);\n            hex = hex.substring(2, hex.length);\n            pow += 8;\n        }\n\n        return result;\n    }\n\n    // Find the byte offset, of Application Segment 1 (EXIF).\n    // External callers need not supply any arguments.\n    function seekToApp1(offset, promise) {\n        var theOffset = offset,\n            thePromise = promise;\n        if (theOffset === undefined) {\n            theOffset = 2;\n            thePromise = new qq.Promise();\n        }\n\n        qq.readBlobToHex(fileOrBlob, theOffset, 4).then(function(hex) {\n            var match = /^ffe([0-9])/.exec(hex),\n                segmentLength;\n\n            if (match) {\n                if (match[1] !== \"1\") {\n                    segmentLength = parseInt(hex.slice(4, 8), 16);\n                    seekToApp1(theOffset + segmentLength + 2, thePromise);\n                }\n                else {\n                    thePromise.success(theOffset);\n                }\n            }\n            else {\n                thePromise.failure(\"No EXIF header to be found!\");\n            }\n        });\n\n        return thePromise;\n    }\n\n    // Find the byte offset of Application Segment 1 (EXIF) for valid JPEGs only.\n    function getApp1Offset() {\n        var promise = new qq.Promise();\n\n        qq.readBlobToHex(fileOrBlob, 0, 6).then(function(hex) {\n            if (hex.indexOf(\"ffd8\") !== 0) {\n                promise.failure(\"Not a valid JPEG!\");\n            }\n            else {\n                seekToApp1().then(function(offset) {\n                    promise.success(offset);\n                },\n                function(error) {\n                    promise.failure(error);\n                });\n            }\n        });\n\n        return promise;\n    }\n\n    // Determine the byte ordering of the EXIF header.\n    function isLittleEndian(app1Start) {\n        var promise = new qq.Promise();\n\n        qq.readBlobToHex(fileOrBlob, app1Start + 10, 2).then(function(hex) {\n            promise.success(hex === \"4949\");\n        });\n\n        return promise;\n    }\n\n    // Determine the number of directory entries in the EXIF header.\n    function getDirEntryCount(app1Start, littleEndian) {\n        var promise = new qq.Promise();\n\n        qq.readBlobToHex(fileOrBlob, app1Start + 18, 2).then(function(hex) {\n            if (littleEndian) {\n                return promise.success(parseLittleEndian(hex));\n            }\n            else {\n                promise.success(parseInt(hex, 16));\n            }\n        });\n\n        return promise;\n    }\n\n    // Get the IFD portion of the EXIF header as a hex string.\n    function getIfd(app1Start, dirEntries) {\n        var offset = app1Start + 20,\n            bytes = dirEntries * 12;\n\n        return qq.readBlobToHex(fileOrBlob, offset, bytes);\n    }\n\n    // Obtain an array of all directory entries (as hex strings) in the EXIF header.\n    function getDirEntries(ifdHex) {\n        var entries = [],\n            offset = 0;\n\n        while (offset + 24 <= ifdHex.length) {\n            entries.push(ifdHex.slice(offset, offset + 24));\n            offset += 24;\n        }\n\n        return entries;\n    }\n\n    // Obtain values for all relevant tags and return them.\n    function getTagValues(littleEndian, dirEntries) {\n        var TAG_VAL_OFFSET = 16,\n            tagsToFind = qq.extend([], TAG_IDS),\n            vals = {};\n\n        qq.each(dirEntries, function(idx, entry) {\n            var idHex = entry.slice(0, 4),\n                id = littleEndian ? parseLittleEndian(idHex) : parseInt(idHex, 16),\n                tagsToFindIdx = tagsToFind.indexOf(id),\n                tagValHex, tagName, tagValLength;\n\n            if (tagsToFindIdx >= 0) {\n                tagName = TAG_INFO[id].name;\n                tagValLength = TAG_INFO[id].bytes;\n                tagValHex = entry.slice(TAG_VAL_OFFSET, TAG_VAL_OFFSET + (tagValLength * 2));\n                vals[tagName] = littleEndian ? parseLittleEndian(tagValHex) : parseInt(tagValHex, 16);\n\n                tagsToFind.splice(tagsToFindIdx, 1);\n            }\n\n            if (tagsToFind.length === 0) {\n                return false;\n            }\n        });\n\n        return vals;\n    }\n\n    qq.extend(this, {\n        /**\n         * Attempt to parse the EXIF header for the `Blob` associated with this instance.\n         *\n         * @returns {qq.Promise} To be fulfilled when the parsing is complete.\n         * If successful, the parsed EXIF header as an object will be included.\n         */\n        parse: function() {\n            var parser = new qq.Promise(),\n                onParseFailure = function(message) {\n                    log(qq.format(\"EXIF header parse failed: '{}' \", message));\n                    parser.failure(message);\n                };\n\n            getApp1Offset().then(function(app1Offset) {\n                log(qq.format(\"Moving forward with EXIF header parsing for '{}'\", fileOrBlob.name === undefined ? \"blob\" : fileOrBlob.name));\n\n                isLittleEndian(app1Offset).then(function(littleEndian) {\n\n                    log(qq.format(\"EXIF Byte order is {} endian\", littleEndian ? \"little\" : \"big\"));\n\n                    getDirEntryCount(app1Offset, littleEndian).then(function(dirEntryCount) {\n\n                        log(qq.format(\"Found {} APP1 directory entries\", dirEntryCount));\n\n                        getIfd(app1Offset, dirEntryCount).then(function(ifdHex) {\n                            var dirEntries = getDirEntries(ifdHex),\n                                tagValues = getTagValues(littleEndian, dirEntries);\n\n                            log(\"Successfully parsed some EXIF tags\");\n\n                            parser.success(tagValues);\n                        }, onParseFailure);\n                    }, onParseFailure);\n                }, onParseFailure);\n            }, onParseFailure);\n\n            return parser;\n        }\n    });\n\n    /*<testing>*/\n    this._testing = {};\n    this._testing.parseLittleEndian = parseLittleEndian;\n    /*</testing>*/\n};\n","/*globals qq */\nqq.Identify = function(fileOrBlob, log) {\n    \"use strict\";\n\n    function isIdentifiable(magicBytes, questionableBytes) {\n        var identifiable = false,\n            magicBytesEntries = [].concat(magicBytes);\n\n        qq.each(magicBytesEntries, function(idx, magicBytesArrayEntry) {\n            if (questionableBytes.indexOf(magicBytesArrayEntry) === 0) {\n                identifiable = true;\n                return false;\n            }\n        });\n\n        return identifiable;\n    }\n\n    qq.extend(this, {\n        /**\n         * Determines if a Blob can be displayed natively in the current browser.  This is done by reading magic\n         * bytes in the beginning of the file, so this is an asynchronous operation.  Before we attempt to read the\n         * file, we will examine the blob's type attribute to save CPU cycles.\n         *\n         * @returns {qq.Promise} Promise that is fulfilled when identification is complete.\n         * If successful, the MIME string is passed to the success handler.\n         */\n        isPreviewable: function() {\n            var self = this,\n                identifier = new qq.Promise(),\n                previewable = false,\n                name = fileOrBlob.name === undefined ? \"blob\" : fileOrBlob.name;\n\n            log(qq.format(\"Attempting to determine if {} can be rendered in this browser\", name));\n\n            log(\"First pass: check type attribute of blob object.\");\n\n            if (this.isPreviewableSync()) {\n                log(\"Second pass: check for magic bytes in file header.\");\n\n                qq.readBlobToHex(fileOrBlob, 0, 4).then(function(hex) {\n                    qq.each(self.PREVIEWABLE_MIME_TYPES, function(mime, bytes) {\n                        if (isIdentifiable(bytes, hex)) {\n                            // Safari is the only supported browser that can deal with TIFFs natively,\n                            // so, if this is a TIFF and the UA isn't Safari, declare this file \"non-previewable\".\n                            if (mime !== \"image/tiff\" || qq.supportedFeatures.tiffPreviews) {\n                                previewable = true;\n                                identifier.success(mime);\n                            }\n\n                            return false;\n                        }\n                    });\n\n                    log(qq.format(\"'{}' is {} able to be rendered in this browser\", name, previewable ? \"\" : \"NOT\"));\n\n                    if (!previewable) {\n                        identifier.failure();\n                    }\n                },\n                function() {\n                    log(\"Error reading file w/ name '\" + name + \"'.  Not able to be rendered in this browser.\");\n                    identifier.failure();\n                });\n            }\n            else {\n                identifier.failure();\n            }\n\n            return identifier;\n        },\n\n        /**\n         * Determines if a Blob can be displayed natively in the current browser.  This is done by checking the\n         * blob's type attribute.  This is a synchronous operation, useful for situations where an asynchronous operation\n         * would be challenging to support.  Note that the blob's type property is not as accurate as reading the\n         * file's magic bytes.\n         *\n         * @returns {Boolean} true if the blob can be rendered in the current browser\n         */\n        isPreviewableSync: function() {\n            var fileMime = fileOrBlob.type,\n                // Assumption: This will only ever be executed in browsers that support `Object.keys`.\n                isRecognizedImage = qq.indexOf(Object.keys(this.PREVIEWABLE_MIME_TYPES), fileMime) >= 0,\n                previewable = false,\n                name = fileOrBlob.name === undefined ? \"blob\" : fileOrBlob.name;\n\n            if (isRecognizedImage) {\n                if (fileMime === \"image/tiff\") {\n                    previewable = qq.supportedFeatures.tiffPreviews;\n                }\n                else {\n                    previewable = true;\n                }\n            }\n\n            !previewable && log(name + \" is not previewable in this browser per the blob's type attr\");\n\n            return previewable;\n        }\n    });\n};\n\nqq.Identify.prototype.PREVIEWABLE_MIME_TYPES = {\n    \"image/jpeg\": \"ffd8ff\",\n    \"image/gif\": \"474946\",\n    \"image/png\": \"89504e\",\n    \"image/bmp\": \"424d\",\n    \"image/tiff\": [\"49492a00\", \"4d4d002a\"]\n};\n","/*globals qq*/\n/**\n * Attempts to validate an image, wherever possible.\n *\n * @param blob File or Blob representing a user-selecting image.\n * @param log Uses this to post log messages to the console.\n * @constructor\n */\nqq.ImageValidation = function(blob, log) {\n    \"use strict\";\n\n    /**\n     * @param limits Object with possible image-related limits to enforce.\n     * @returns {boolean} true if at least one of the limits has a non-zero value\n     */\n    function hasNonZeroLimits(limits) {\n        var atLeastOne = false;\n\n        qq.each(limits, function(limit, value) {\n            if (value > 0) {\n                atLeastOne = true;\n                return false;\n            }\n        });\n\n        return atLeastOne;\n    }\n\n    /**\n     * @returns {qq.Promise} The promise is a failure if we can't obtain the width & height.\n     * Otherwise, `success` is called on the returned promise with an object containing\n     * `width` and `height` properties.\n     */\n    function getWidthHeight() {\n        var sizeDetermination = new qq.Promise();\n\n        new qq.Identify(blob, log).isPreviewable().then(function() {\n            var image = new Image(),\n                url = window.URL && window.URL.createObjectURL ? window.URL :\n                      window.webkitURL && window.webkitURL.createObjectURL ? window.webkitURL :\n                      null;\n\n            if (url) {\n                image.onerror = function() {\n                    log(\"Cannot determine dimensions for image.  May be too large.\", \"error\");\n                    sizeDetermination.failure();\n                };\n\n                image.onload = function() {\n                    sizeDetermination.success({\n                        width: this.width,\n                        height: this.height\n                    });\n                };\n\n                image.src = url.createObjectURL(blob);\n            }\n            else {\n                log(\"No createObjectURL function available to generate image URL!\", \"error\");\n                sizeDetermination.failure();\n            }\n        }, sizeDetermination.failure);\n\n        return sizeDetermination;\n    }\n\n    /**\n     *\n     * @param limits Object with possible image-related limits to enforce.\n     * @param dimensions Object containing `width` & `height` properties for the image to test.\n     * @returns {String || undefined} The name of the failing limit.  Undefined if no failing limits.\n     */\n    function getFailingLimit(limits, dimensions) {\n        var failingLimit;\n\n        qq.each(limits, function(limitName, limitValue) {\n            if (limitValue > 0) {\n                var limitMatcher = /(max|min)(Width|Height)/.exec(limitName),\n                    dimensionPropName = limitMatcher[2].charAt(0).toLowerCase() + limitMatcher[2].slice(1),\n                    actualValue = dimensions[dimensionPropName];\n\n                /*jshint -W015*/\n                switch (limitMatcher[1]) {\n                    case \"min\":\n                        if (actualValue < limitValue) {\n                            failingLimit = limitName;\n                            return false;\n                        }\n                        break;\n                    case \"max\":\n                        if (actualValue > limitValue) {\n                            failingLimit = limitName;\n                            return false;\n                        }\n                        break;\n                }\n            }\n        });\n\n        return failingLimit;\n    }\n\n    /**\n     * Validate the associated blob.\n     *\n     * @param limits\n     * @returns {qq.Promise} `success` is called on the promise is the image is valid or\n     * if the blob is not an image, or if the image is not verifiable.\n     * Otherwise, `failure` with the name of the failing limit.\n     */\n    this.validate = function(limits) {\n        var validationEffort = new qq.Promise();\n\n        log(\"Attempting to validate image.\");\n\n        if (hasNonZeroLimits(limits)) {\n            getWidthHeight().then(function(dimensions) {\n                var failingLimit = getFailingLimit(limits, dimensions);\n\n                if (failingLimit) {\n                    validationEffort.failure(failingLimit);\n                }\n                else {\n                    validationEffort.success();\n                }\n            }, validationEffort.success);\n        }\n        else {\n            validationEffort.success();\n        }\n\n        return validationEffort;\n    };\n};\n","/* globals qq */\n/**\n * Module used to control populating the initial list of files.\n *\n * @constructor\n */\nqq.Session = function(spec) {\n    \"use strict\";\n\n    var options = {\n        endpoint: null,\n        params: {},\n        customHeaders: {},\n        cors: {},\n        addFileRecord: function(sessionData) {},\n        log: function(message, level) {}\n    };\n\n    qq.extend(options, spec, true);\n\n    function isJsonResponseValid(response) {\n        if (qq.isArray(response)) {\n            return true;\n        }\n\n        options.log(\"Session response is not an array.\", \"error\");\n    }\n\n    function handleFileItems(fileItems, success, xhrOrXdr, promise) {\n        var someItemsIgnored = false;\n\n        success = success && isJsonResponseValid(fileItems);\n\n        if (success) {\n            qq.each(fileItems, function(idx, fileItem) {\n                /* jshint eqnull:true */\n                if (fileItem.uuid == null) {\n                    someItemsIgnored = true;\n                    options.log(qq.format(\"Session response item {} did not include a valid UUID - ignoring.\", idx), \"error\");\n                }\n                else if (fileItem.name == null) {\n                    someItemsIgnored = true;\n                    options.log(qq.format(\"Session response item {} did not include a valid name - ignoring.\", idx), \"error\");\n                }\n                else {\n                    try {\n                        options.addFileRecord(fileItem);\n                        return true;\n                    }\n                    catch (err) {\n                        someItemsIgnored = true;\n                        options.log(err.message, \"error\");\n                    }\n                }\n\n                return false;\n            });\n        }\n\n        promise[success && !someItemsIgnored ? \"success\" : \"failure\"](fileItems, xhrOrXdr);\n    }\n\n    // Initiate a call to the server that will be used to populate the initial file list.\n    // Returns a `qq.Promise`.\n    this.refresh = function() {\n        /*jshint indent:false */\n        var refreshEffort = new qq.Promise(),\n            refreshCompleteCallback = function(response, success, xhrOrXdr) {\n                handleFileItems(response, success, xhrOrXdr, refreshEffort);\n            },\n            requesterOptions = qq.extend({}, options),\n            requester = new qq.SessionAjaxRequester(\n                qq.extend(requesterOptions, {onComplete: refreshCompleteCallback})\n            );\n\n        requester.queryServer();\n\n        return refreshEffort;\n    };\n};\n","/*globals qq, XMLHttpRequest*/\n/**\n * Thin module used to send GET requests to the server, expecting information about session\n * data used to initialize an uploader instance.\n *\n * @param spec Various options used to influence the associated request.\n * @constructor\n */\nqq.SessionAjaxRequester = function(spec) {\n    \"use strict\";\n\n    var requester,\n        options = {\n            endpoint: null,\n            customHeaders: {},\n            params: {},\n            cors: {\n                expected: false,\n                sendCredentials: false\n            },\n            onComplete: function(response, success, xhrOrXdr) {},\n            log: function(str, level) {}\n        };\n\n    qq.extend(options, spec);\n\n    function onComplete(id, xhrOrXdr, isError) {\n        var response = null;\n\n        /* jshint eqnull:true */\n        if (xhrOrXdr.responseText != null) {\n            try {\n                response = qq.parseJson(xhrOrXdr.responseText);\n            }\n            catch (err) {\n                options.log(\"Problem parsing session response: \" + err.message, \"error\");\n                isError = true;\n            }\n        }\n\n        options.onComplete(response, !isError, xhrOrXdr);\n    }\n\n    requester = qq.extend(this, new qq.AjaxRequester({\n        acceptHeader: \"application/json\",\n        validMethods: [\"GET\"],\n        method: \"GET\",\n        endpointStore: {\n            get: function() {\n                return options.endpoint;\n            }\n        },\n        customHeaders: options.customHeaders,\n        log: options.log,\n        onComplete: onComplete,\n        cors: options.cors\n    }));\n\n    qq.extend(this, {\n        queryServer: function() {\n            var params = qq.extend({}, options.params);\n\n            options.log(\"Session query request.\");\n\n            requester.initTransport(\"sessionRefresh\")\n                .withParams(params)\n                .withCacheBuster()\n                .send();\n        }\n    });\n};\n","/* globals qq, ExifRestorer */\n/**\n * Controls generation of scaled images based on a reference image encapsulated in a `File` or `Blob`.\n * Scaled images are generated and converted to blobs on-demand.\n * Multiple scaled images per reference image with varying sizes and other properties are supported.\n *\n * @param spec Information about the scaled images to generate.\n * @param log Logger instance\n * @constructor\n */\nqq.Scaler = function(spec, log) {\n    \"use strict\";\n\n    var self = this,\n        customResizeFunction = spec.customResizer,\n        includeOriginal = spec.sendOriginal,\n        orient = spec.orient,\n        defaultType = spec.defaultType,\n        defaultQuality = spec.defaultQuality / 100,\n        failedToScaleText = spec.failureText,\n        includeExif = spec.includeExif,\n        sizes = this._getSortedSizes(spec.sizes);\n\n    // Revealed API for instances of this module\n    qq.extend(this, {\n        // If no targeted sizes have been declared or if this browser doesn't support\n        // client-side image preview generation, there is no scaling to do.\n        enabled: qq.supportedFeatures.scaling && sizes.length > 0,\n\n        getFileRecords: function(originalFileUuid, originalFileName, originalBlobOrBlobData) {\n            var self = this,\n                records = [],\n                originalBlob = originalBlobOrBlobData.blob ? originalBlobOrBlobData.blob : originalBlobOrBlobData,\n                identifier = new qq.Identify(originalBlob, log);\n\n            // If the reference file cannot be rendered natively, we can't create scaled versions.\n            if (identifier.isPreviewableSync()) {\n                // Create records for each scaled version & add them to the records array, smallest first.\n                qq.each(sizes, function(idx, sizeRecord) {\n                    var outputType = self._determineOutputType({\n                        defaultType: defaultType,\n                        requestedType: sizeRecord.type,\n                        refType: originalBlob.type\n                    });\n\n                    records.push({\n                        uuid: qq.getUniqueId(),\n                        name: self._getName(originalFileName, {\n                            name: sizeRecord.name,\n                            type: outputType,\n                            refType: originalBlob.type\n                        }),\n                        blob: new qq.BlobProxy(originalBlob,\n                        qq.bind(self._generateScaledImage, self, {\n                            customResizeFunction: customResizeFunction,\n                            maxSize: sizeRecord.maxSize,\n                            orient: orient,\n                            type: outputType,\n                            quality: defaultQuality,\n                            failedText: failedToScaleText,\n                            includeExif: includeExif,\n                            log: log\n                        }))\n                    });\n                });\n\n                records.push({\n                    uuid: originalFileUuid,\n                    name: originalFileName,\n                    size: originalBlob.size,\n                    blob: includeOriginal ? originalBlob : null\n                });\n            }\n            else {\n                records.push({\n                    uuid: originalFileUuid,\n                    name: originalFileName,\n                    size: originalBlob.size,\n                    blob: originalBlob\n                });\n            }\n\n            return records;\n        },\n\n        handleNewFile: function(file, name, uuid, size, fileList, batchId, uuidParamName, api) {\n            var self = this,\n                buttonId = file.qqButtonId || (file.blob && file.blob.qqButtonId),\n                scaledIds = [],\n                originalId = null,\n                addFileToHandler = api.addFileToHandler,\n                uploadData = api.uploadData,\n                paramsStore = api.paramsStore,\n                proxyGroupId = qq.getUniqueId();\n\n            qq.each(self.getFileRecords(uuid, name, file), function(idx, record) {\n                var blobSize = record.size,\n                    id;\n\n                if (record.blob instanceof qq.BlobProxy) {\n                    blobSize = -1;\n                }\n\n                id = uploadData.addFile({\n                    uuid: record.uuid,\n                    name: record.name,\n                    size: blobSize,\n                    batchId: batchId,\n                    proxyGroupId: proxyGroupId\n                });\n\n                if (record.blob instanceof qq.BlobProxy) {\n                    scaledIds.push(id);\n                }\n                else {\n                    originalId = id;\n                }\n\n                if (record.blob) {\n                    addFileToHandler(id, record.blob);\n                    fileList.push({id: id, file: record.blob});\n                }\n                else {\n                    uploadData.setStatus(id, qq.status.REJECTED);\n                }\n            });\n\n            // If we are potentially uploading an original file and some scaled versions,\n            // ensure the scaled versions include reference's to the parent's UUID and size\n            // in their associated upload requests.\n            if (originalId !== null) {\n                qq.each(scaledIds, function(idx, scaledId) {\n                    var params = {\n                        qqparentuuid: uploadData.retrieve({id: originalId}).uuid,\n                        qqparentsize: uploadData.retrieve({id: originalId}).size\n                    };\n\n                    // Make sure the UUID for each scaled image is sent with the upload request,\n                    // to be consistent (since we may need to ensure it is sent for the original file as well).\n                    params[uuidParamName] = uploadData.retrieve({id: scaledId}).uuid;\n\n                    uploadData.setParentId(scaledId, originalId);\n                    paramsStore.addReadOnly(scaledId, params);\n                });\n\n                // If any scaled images are tied to this parent image, be SURE we send its UUID as an upload request\n                // parameter as well.\n                if (scaledIds.length) {\n                    (function() {\n                        var param = {};\n                        param[uuidParamName] = uploadData.retrieve({id: originalId}).uuid;\n                        paramsStore.addReadOnly(originalId, param);\n                    }());\n                }\n            }\n        }\n    });\n};\n\nqq.extend(qq.Scaler.prototype, {\n    scaleImage: function(id, specs, api) {\n        \"use strict\";\n\n        if (!qq.supportedFeatures.scaling) {\n            throw new qq.Error(\"Scaling is not supported in this browser!\");\n        }\n\n        var scalingEffort = new qq.Promise(),\n            log = api.log,\n            file = api.getFile(id),\n            uploadData = api.uploadData.retrieve({id: id}),\n            name = uploadData && uploadData.name,\n            uuid = uploadData && uploadData.uuid,\n            scalingOptions = {\n                customResizer: specs.customResizer,\n                sendOriginal: false,\n                orient: specs.orient,\n                defaultType: specs.type || null,\n                defaultQuality: specs.quality,\n                failedToScaleText: \"Unable to scale\",\n                sizes: [{name: \"\", maxSize: specs.maxSize}]\n            },\n            scaler = new qq.Scaler(scalingOptions, log);\n\n        if (!qq.Scaler || !qq.supportedFeatures.imagePreviews || !file) {\n            scalingEffort.failure();\n\n            log(\"Could not generate requested scaled image for \" + id + \".  \" +\n                \"Scaling is either not possible in this browser, or the file could not be located.\", \"error\");\n        }\n        else {\n            (qq.bind(function() {\n                // Assumption: There will never be more than one record\n                var record = scaler.getFileRecords(uuid, name, file)[0];\n\n                if (record && record.blob instanceof qq.BlobProxy) {\n                    record.blob.create().then(scalingEffort.success, scalingEffort.failure);\n                }\n                else {\n                    log(id + \" is not a scalable image!\", \"error\");\n                    scalingEffort.failure();\n                }\n            }, this)());\n        }\n\n        return scalingEffort;\n    },\n\n    // NOTE: We cannot reliably determine at this time if the UA supports a specific MIME type for the target format.\n    // image/jpeg and image/png are the only safe choices at this time.\n    _determineOutputType: function(spec) {\n        \"use strict\";\n\n        var requestedType = spec.requestedType,\n            defaultType = spec.defaultType,\n            referenceType = spec.refType;\n\n        // If a default type and requested type have not been specified, this should be a\n        // JPEG if the original type is a JPEG, otherwise, a PNG.\n        if (!defaultType && !requestedType) {\n            if (referenceType !== \"image/jpeg\") {\n                return \"image/png\";\n            }\n            return referenceType;\n        }\n\n        // A specified default type is used when a requested type is not specified.\n        if (!requestedType) {\n            return defaultType;\n        }\n\n        // If requested type is specified, use it, as long as this recognized type is supported by the current UA\n        if (qq.indexOf(Object.keys(qq.Identify.prototype.PREVIEWABLE_MIME_TYPES), requestedType) >= 0) {\n            if (requestedType === \"image/tiff\") {\n                return qq.supportedFeatures.tiffPreviews ? requestedType : defaultType;\n            }\n\n            return requestedType;\n        }\n\n        return defaultType;\n    },\n\n    // Get a file name for a generated scaled file record, based on the provided scaled image description\n    _getName: function(originalName, scaledVersionProperties) {\n        \"use strict\";\n\n        var startOfExt = originalName.lastIndexOf(\".\"),\n            versionType = scaledVersionProperties.type || \"image/png\",\n            referenceType = scaledVersionProperties.refType,\n            scaledName = \"\",\n            scaledExt = qq.getExtension(originalName),\n            nameAppendage = \"\";\n\n        if (scaledVersionProperties.name && scaledVersionProperties.name.trim().length) {\n            nameAppendage = \" (\" + scaledVersionProperties.name + \")\";\n        }\n\n        if (startOfExt >= 0) {\n            scaledName = originalName.substr(0, startOfExt);\n\n            if (referenceType !== versionType) {\n                scaledExt = versionType.split(\"/\")[1];\n            }\n\n            scaledName += nameAppendage + \".\" + scaledExt;\n        }\n        else {\n            scaledName = originalName + nameAppendage;\n        }\n\n        return scaledName;\n    },\n\n    // We want the smallest scaled file to be uploaded first\n    _getSortedSizes: function(sizes) {\n        \"use strict\";\n\n        sizes = qq.extend([], sizes);\n\n        return sizes.sort(function(a, b) {\n            if (a.maxSize > b.maxSize) {\n                return 1;\n            }\n            if (a.maxSize < b.maxSize) {\n                return -1;\n            }\n            return 0;\n        });\n    },\n\n    _generateScaledImage: function(spec, sourceFile) {\n        \"use strict\";\n\n        var self = this,\n            customResizeFunction = spec.customResizeFunction,\n            log = spec.log,\n            maxSize = spec.maxSize,\n            orient = spec.orient,\n            type = spec.type,\n            quality = spec.quality,\n            failedText = spec.failedText,\n            includeExif = spec.includeExif && sourceFile.type === \"image/jpeg\" && type === \"image/jpeg\",\n            scalingEffort = new qq.Promise(),\n            imageGenerator = new qq.ImageGenerator(log),\n            canvas = document.createElement(\"canvas\");\n\n        log(\"Attempting to generate scaled version for \" + sourceFile.name);\n\n        imageGenerator.generate(sourceFile, canvas, {maxSize: maxSize, orient: orient, customResizeFunction: customResizeFunction}).then(function() {\n            var scaledImageDataUri = canvas.toDataURL(type, quality),\n                signalSuccess = function() {\n                    log(\"Success generating scaled version for \" + sourceFile.name);\n                    var blob = qq.dataUriToBlob(scaledImageDataUri);\n                    scalingEffort.success(blob);\n                };\n\n            if (includeExif) {\n                self._insertExifHeader(sourceFile, scaledImageDataUri, log).then(function(scaledImageDataUriWithExif) {\n                    scaledImageDataUri = scaledImageDataUriWithExif;\n                    signalSuccess();\n                },\n                function() {\n                    log(\"Problem inserting EXIF header into scaled image.  Using scaled image w/out EXIF data.\", \"error\");\n                    signalSuccess();\n                });\n            }\n            else {\n                signalSuccess();\n            }\n        }, function() {\n            log(\"Failed attempt to generate scaled version for \" + sourceFile.name, \"error\");\n            scalingEffort.failure(failedText);\n        });\n\n        return scalingEffort;\n    },\n\n    // Attempt to insert the original image's EXIF header into a scaled version.\n    _insertExifHeader: function(originalImage, scaledImageDataUri, log) {\n        \"use strict\";\n\n        var reader = new FileReader(),\n            insertionEffort = new qq.Promise(),\n            originalImageDataUri = \"\";\n\n        reader.onload = function() {\n            originalImageDataUri = reader.result;\n            insertionEffort.success(qq.ExifRestorer.restore(originalImageDataUri, scaledImageDataUri));\n        };\n\n        reader.onerror = function() {\n            log(\"Problem reading \" + originalImage.name + \" during attempt to transfer EXIF data to scaled version.\", \"error\");\n            insertionEffort.failure();\n        };\n\n        reader.readAsDataURL(originalImage);\n\n        return insertionEffort;\n    },\n\n    _dataUriToBlob: function(dataUri) {\n        \"use strict\";\n\n        var byteString, mimeString, arrayBuffer, intArray;\n\n        // convert base64 to raw binary data held in a string\n        if (dataUri.split(\",\")[0].indexOf(\"base64\") >= 0) {\n            byteString = atob(dataUri.split(\",\")[1]);\n        }\n        else {\n            byteString = decodeURI(dataUri.split(\",\")[1]);\n        }\n\n        // extract the MIME\n        mimeString = dataUri.split(\",\")[0]\n            .split(\":\")[1]\n            .split(\";\")[0];\n\n        // write the bytes of the binary string to an ArrayBuffer\n        arrayBuffer = new ArrayBuffer(byteString.length);\n        intArray = new Uint8Array(arrayBuffer);\n        qq.each(byteString, function(idx, character) {\n            intArray[idx] = character.charCodeAt(0);\n        });\n\n        return this._createBlob(arrayBuffer, mimeString);\n    },\n\n    _createBlob: function(data, mime) {\n        \"use strict\";\n\n        var BlobBuilder = window.BlobBuilder ||\n                window.WebKitBlobBuilder ||\n                window.MozBlobBuilder ||\n                window.MSBlobBuilder,\n            blobBuilder = BlobBuilder && new BlobBuilder();\n\n        if (blobBuilder) {\n            blobBuilder.append(data);\n            return blobBuilder.getBlob(mime);\n        }\n        else {\n            return new Blob([data], {type: mime});\n        }\n    }\n});\n","//Based on MinifyJpeg\n//http://elicon.blog57.fc2.com/blog-entry-206.html\n\nqq.ExifRestorer = (function()\n{\n   \n\tvar ExifRestorer = {};\n\t \n    ExifRestorer.KEY_STR = \"ABCDEFGHIJKLMNOP\" +\n                         \"QRSTUVWXYZabcdef\" +\n                         \"ghijklmnopqrstuv\" +\n                         \"wxyz0123456789+/\" +\n                         \"=\";\n\n    ExifRestorer.encode64 = function(input)\n    {\n        var output = \"\",\n            chr1, chr2, chr3 = \"\",\n            enc1, enc2, enc3, enc4 = \"\",\n            i = 0;\n\n        do {\n            chr1 = input[i++];\n            chr2 = input[i++];\n            chr3 = input[i++];\n\n            enc1 = chr1 >> 2;\n            enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);\n            enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);\n            enc4 = chr3 & 63;\n\n            if (isNaN(chr2)) {\n               enc3 = enc4 = 64;\n            } else if (isNaN(chr3)) {\n               enc4 = 64;\n            }\n\n            output = output +\n               this.KEY_STR.charAt(enc1) +\n               this.KEY_STR.charAt(enc2) +\n               this.KEY_STR.charAt(enc3) +\n               this.KEY_STR.charAt(enc4);\n            chr1 = chr2 = chr3 = \"\";\n            enc1 = enc2 = enc3 = enc4 = \"\";\n        } while (i < input.length);\n\n        return output;\n    };\n    \n    ExifRestorer.restore = function(origFileBase64, resizedFileBase64)\n    {\n        var expectedBase64Header = \"data:image/jpeg;base64,\";\n\n        if (!origFileBase64.match(expectedBase64Header))\n        {\n        \treturn resizedFileBase64;\n        }       \n        \n        var rawImage = this.decode64(origFileBase64.replace(expectedBase64Header, \"\"));\n        var segments = this.slice2Segments(rawImage);\n                \n        var image = this.exifManipulation(resizedFileBase64, segments);\n        \n        return expectedBase64Header + this.encode64(image);\n        \n    };\n\n\n    ExifRestorer.exifManipulation = function(resizedFileBase64, segments)\n    {\n            var exifArray = this.getExifArray(segments),\n                newImageArray = this.insertExif(resizedFileBase64, exifArray),\n                aBuffer = new Uint8Array(newImageArray);\n\n            return aBuffer;\n    };\n\n\n    ExifRestorer.getExifArray = function(segments)\n    {\n            var seg;\n            for (var x = 0; x < segments.length; x++)\n            {\n                seg = segments[x];\n                if (seg[0] == 255 & seg[1] == 225) //(ff e1)\n                {\n                    return seg;\n                }\n            }\n            return [];\n    };\n\n\n    ExifRestorer.insertExif = function(resizedFileBase64, exifArray)\n    {\n            var imageData = resizedFileBase64.replace(\"data:image/jpeg;base64,\", \"\"),\n                buf = this.decode64(imageData),\n                separatePoint = buf.indexOf(255,3),\n                mae = buf.slice(0, separatePoint),\n                ato = buf.slice(separatePoint),\n                array = mae;\n\n            array = array.concat(exifArray);\n            array = array.concat(ato);\n           return array;\n    };\n\n\n    \n    ExifRestorer.slice2Segments = function(rawImageArray)\n    {\n        var head = 0,\n            segments = [];\n\n        while (1)\n        {\n            if (rawImageArray[head] == 255 & rawImageArray[head + 1] == 218){break;}\n            if (rawImageArray[head] == 255 & rawImageArray[head + 1] == 216)\n            {\n                head += 2;\n            }\n            else\n            {\n                var length = rawImageArray[head + 2] * 256 + rawImageArray[head + 3],\n                    endPoint = head + length + 2,\n                    seg = rawImageArray.slice(head, endPoint);\n                segments.push(seg);\n                head = endPoint;\n            }\n            if (head > rawImageArray.length){break;}\n        }\n\n        return segments;\n    };\n\n\n    \n    ExifRestorer.decode64 = function(input) \n    {\n        var output = \"\",\n            chr1, chr2, chr3 = \"\",\n            enc1, enc2, enc3, enc4 = \"\",\n            i = 0,\n            buf = [];\n\n        // remove all characters that are not A-Z, a-z, 0-9, +, /, or =\n        var base64test = /[^A-Za-z0-9\\+\\/\\=]/g;\n        if (base64test.exec(input)) {\n            throw new Error(\"There were invalid base64 characters in the input text.  \" +\n                \"Valid base64 characters are A-Z, a-z, 0-9, '+', '/',and '='\");\n        }\n        input = input.replace(/[^A-Za-z0-9\\+\\/\\=]/g, \"\");\n\n        do {\n            enc1 = this.KEY_STR.indexOf(input.charAt(i++));\n            enc2 = this.KEY_STR.indexOf(input.charAt(i++));\n            enc3 = this.KEY_STR.indexOf(input.charAt(i++));\n            enc4 = this.KEY_STR.indexOf(input.charAt(i++));\n\n            chr1 = (enc1 << 2) | (enc2 >> 4);\n            chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);\n            chr3 = ((enc3 & 3) << 6) | enc4;\n\n            buf.push(chr1);\n\n            if (enc3 != 64) {\n               buf.push(chr2);\n            }\n            if (enc4 != 64) {\n               buf.push(chr3);\n            }\n\n            chr1 = chr2 = chr3 = \"\";\n            enc1 = enc2 = enc3 = enc4 = \"\";\n\n        } while (i < input.length);\n\n        return buf;\n    };\n\n    \n    return ExifRestorer;\n})();\n","/* globals qq */\n/**\n * Keeps a running tally of total upload progress for a batch of files.\n *\n * @param callback Invoked when total progress changes, passing calculated total loaded & total size values.\n * @param getSize Function that returns the size of a file given its ID\n * @constructor\n */\nqq.TotalProgress = function(callback, getSize) {\n    \"use strict\";\n\n    var perFileProgress = {},\n        totalLoaded = 0,\n        totalSize = 0,\n\n        lastLoadedSent = -1,\n        lastTotalSent = -1,\n        callbackProxy = function(loaded, total) {\n            if (loaded !== lastLoadedSent || total !== lastTotalSent) {\n                callback(loaded, total);\n            }\n\n            lastLoadedSent = loaded;\n            lastTotalSent = total;\n        },\n\n        /**\n         * @param failed Array of file IDs that have failed\n         * @param retryable Array of file IDs that are retryable\n         * @returns true if none of the failed files are eligible for retry\n         */\n        noRetryableFiles = function(failed, retryable) {\n            var none = true;\n\n            qq.each(failed, function(idx, failedId) {\n                if (qq.indexOf(retryable, failedId) >= 0) {\n                    none = false;\n                    return false;\n                }\n            });\n\n            return none;\n        },\n\n        onCancel = function(id) {\n            updateTotalProgress(id, -1, -1);\n            delete perFileProgress[id];\n        },\n\n        onAllComplete = function(successful, failed, retryable) {\n            if (failed.length === 0 || noRetryableFiles(failed, retryable)) {\n                callbackProxy(totalSize, totalSize);\n                this.reset();\n            }\n        },\n\n        onNew = function(id) {\n            var size = getSize(id);\n\n            // We might not know the size yet, such as for blob proxies\n            if (size > 0) {\n                updateTotalProgress(id, 0, size);\n                perFileProgress[id] = {loaded: 0, total: size};\n            }\n        },\n\n        /**\n         * Invokes the callback with the current total progress of all files in the batch.  Called whenever it may\n         * be appropriate to re-calculate and disseminate this data.\n         *\n         * @param id ID of a file that has changed in some important way\n         * @param newLoaded New loaded value for this file.  -1 if this value should no longer be part of calculations\n         * @param newTotal New total size of the file.  -1 if this value should no longer be part of calculations\n         */\n        updateTotalProgress = function(id, newLoaded, newTotal) {\n            var oldLoaded = perFileProgress[id] ? perFileProgress[id].loaded : 0,\n                oldTotal = perFileProgress[id] ? perFileProgress[id].total : 0;\n\n            if (newLoaded === -1 && newTotal === -1) {\n                totalLoaded -= oldLoaded;\n                totalSize -= oldTotal;\n            }\n            else {\n                if (newLoaded) {\n                    totalLoaded += newLoaded - oldLoaded;\n                }\n                if (newTotal) {\n                    totalSize += newTotal - oldTotal;\n                }\n            }\n\n            callbackProxy(totalLoaded, totalSize);\n        };\n\n    qq.extend(this, {\n        // Called when a batch of files has completed uploading.\n        onAllComplete: onAllComplete,\n\n        // Called when the status of a file has changed.\n        onStatusChange: function(id, oldStatus, newStatus) {\n            if (newStatus === qq.status.CANCELED || newStatus === qq.status.REJECTED) {\n                onCancel(id);\n            }\n            else if (newStatus === qq.status.SUBMITTING) {\n                onNew(id);\n            }\n        },\n\n        // Called whenever the upload progress of an individual file has changed.\n        onIndividualProgress: function(id, loaded, total) {\n            updateTotalProgress(id, loaded, total);\n            perFileProgress[id] = {loaded: loaded, total: total};\n        },\n\n        // Called whenever the total size of a file has changed, such as when the size of a generated blob is known.\n        onNewSize: function(id) {\n            onNew(id);\n        },\n\n        reset: function() {\n            perFileProgress = {};\n            totalLoaded = 0;\n            totalSize = 0;\n        }\n    });\n};\n","/*globals qq*/\nqq.PasteSupport = function(o) {\n    \"use strict\";\n\n    var options, detachPasteHandler;\n\n    options = {\n        targetElement: null,\n        callbacks: {\n            log: function(message, level) {},\n            pasteReceived: function(blob) {}\n        }\n    };\n\n    function isImage(item) {\n        return item.type &&\n            item.type.indexOf(\"image/\") === 0;\n    }\n\n    function registerPasteHandler() {\n        detachPasteHandler = qq(options.targetElement).attach(\"paste\", function(event) {\n            var clipboardData = event.clipboardData;\n\n            if (clipboardData) {\n                qq.each(clipboardData.items, function(idx, item) {\n                    if (isImage(item)) {\n                        var blob = item.getAsFile();\n                        options.callbacks.pasteReceived(blob);\n                    }\n                });\n            }\n        });\n    }\n\n    function unregisterPasteHandler() {\n        if (detachPasteHandler) {\n            detachPasteHandler();\n        }\n    }\n\n    qq.extend(options, o);\n    registerPasteHandler();\n\n    qq.extend(this, {\n        reset: function() {\n            unregisterPasteHandler();\n        }\n    });\n};\n","/* globals qq */\n/**\n * Module that handles support for existing forms.\n *\n * @param options Options passed from the integrator-supplied options related to form support.\n * @param startUpload Callback to invoke when files \"stored\" should be uploaded.\n * @param log Proxy for the logger\n * @constructor\n */\nqq.FormSupport = function(options, startUpload, log) {\n    \"use strict\";\n    var self  = this,\n        interceptSubmit = options.interceptSubmit,\n        formEl = options.element,\n        autoUpload = options.autoUpload;\n\n    // Available on the public API associated with this module.\n    qq.extend(this, {\n        // To be used by the caller to determine if the endpoint will be determined by some processing\n        // that occurs in this module, such as if the form has an action attribute.\n        // Ignore if `attachToForm === false`.\n        newEndpoint: null,\n\n        // To be used by the caller to determine if auto uploading should be allowed.\n        // Ignore if `attachToForm === false`.\n        newAutoUpload: autoUpload,\n\n        // true if a form was detected and is being tracked by this module\n        attachedToForm: false,\n\n        // Returns an object with names and values for all valid form elements associated with the attached form.\n        getFormInputsAsObject: function() {\n            /* jshint eqnull:true */\n            if (formEl == null) {\n                return null;\n            }\n\n            return self._form2Obj(formEl);\n        }\n    });\n\n    // If the form contains an action attribute, this should be the new upload endpoint.\n    function determineNewEndpoint(formEl) {\n        if (formEl.getAttribute(\"action\")) {\n            self.newEndpoint = formEl.getAttribute(\"action\");\n        }\n    }\n\n    // Return true only if the form is valid, or if we cannot make this determination.\n    // If the form is invalid, ensure invalid field(s) are highlighted in the UI.\n    function validateForm(formEl, nativeSubmit) {\n        if (formEl.checkValidity && !formEl.checkValidity()) {\n            log(\"Form did not pass validation checks - will not upload.\", \"error\");\n            nativeSubmit();\n        }\n        else {\n            return true;\n        }\n    }\n\n    // Intercept form submit attempts, unless the integrator has told us not to do this.\n    function maybeUploadOnSubmit(formEl) {\n        var nativeSubmit = formEl.submit;\n\n        // Intercept and squelch submit events.\n        qq(formEl).attach(\"submit\", function(event) {\n            event = event || window.event;\n\n            if (event.preventDefault) {\n                event.preventDefault();\n            }\n            else {\n                event.returnValue = false;\n            }\n\n            validateForm(formEl, nativeSubmit) && startUpload();\n        });\n\n        // The form's `submit()` function may be called instead (i.e. via jQuery.submit()).\n        // Intercept that too.\n        formEl.submit = function() {\n            validateForm(formEl, nativeSubmit) && startUpload();\n        };\n    }\n\n    // If the element value passed from the uploader is a string, assume it is an element ID - select it.\n    // The rest of the code in this module depends on this being an HTMLElement.\n    function determineFormEl(formEl) {\n        if (formEl) {\n            if (qq.isString(formEl)) {\n                formEl = document.getElementById(formEl);\n            }\n\n            if (formEl) {\n                log(\"Attaching to form element.\");\n                determineNewEndpoint(formEl);\n                interceptSubmit && maybeUploadOnSubmit(formEl);\n            }\n        }\n\n        return formEl;\n    }\n\n    formEl = determineFormEl(formEl);\n    this.attachedToForm = !!formEl;\n};\n\nqq.extend(qq.FormSupport.prototype, {\n    // Converts all relevant form fields to key/value pairs.  This is meant to mimic the data a browser will\n    // construct from a given form when the form is submitted.\n    _form2Obj: function(form) {\n        \"use strict\";\n        var obj = {},\n            notIrrelevantType = function(type) {\n                var irrelevantTypes = [\n                    \"button\",\n                    \"image\",\n                    \"reset\",\n                    \"submit\"\n                ];\n\n                return qq.indexOf(irrelevantTypes, type.toLowerCase()) < 0;\n            },\n            radioOrCheckbox = function(type) {\n                return qq.indexOf([\"checkbox\", \"radio\"], type.toLowerCase()) >= 0;\n            },\n            ignoreValue = function(el) {\n                if (radioOrCheckbox(el.type) && !el.checked) {\n                    return true;\n                }\n\n                return el.disabled && el.type.toLowerCase() !== \"hidden\";\n            },\n            selectValue = function(select) {\n                var value = null;\n\n                qq.each(qq(select).children(), function(idx, child) {\n                    if (child.tagName.toLowerCase() === \"option\" && child.selected) {\n                        value = child.value;\n                        return false;\n                    }\n                });\n\n                return value;\n            };\n\n        qq.each(form.elements, function(idx, el) {\n            if ((qq.isInput(el, true) || el.tagName.toLowerCase() === \"textarea\") &&\n                notIrrelevantType(el.type) &&\n                !ignoreValue(el)) {\n\n                obj[el.name] = el.value;\n            }\n            else if (el.tagName.toLowerCase() === \"select\" && !ignoreValue(el)) {\n                var value = selectValue(el);\n\n                if (value !== null) {\n                    obj[el.name] = value;\n                }\n            }\n        });\n\n        return obj;\n    }\n});\n","/*globals qq*/\n/**\n * Upload handler used that assumes the current user agent does not have any support for the\n * File API, and, therefore, makes use of iframes and forms to submit the files directly to\n * a generic server.\n *\n * @param options Options passed from the base handler\n * @param proxy Callbacks & methods used to query for or push out data/changes\n */\nqq.traditional = qq.traditional || {};\nqq.traditional.FormUploadHandler = function(options, proxy) {\n    \"use strict\";\n\n    var handler = this,\n        getName = proxy.getName,\n        getUuid = proxy.getUuid,\n        log = proxy.log;\n\n    /**\n     * Returns json object received by iframe from server.\n     */\n    function getIframeContentJson(id, iframe) {\n        /*jshint evil: true*/\n\n        var response, doc, innerHtml;\n\n        //IE may throw an \"access is denied\" error when attempting to access contentDocument on the iframe in some cases\n        try {\n            // iframe.contentWindow.document - for IE<7\n            doc = iframe.contentDocument || iframe.contentWindow.document;\n            innerHtml = doc.body.innerHTML;\n\n            log(\"converting iframe's innerHTML to JSON\");\n            log(\"innerHTML = \" + innerHtml);\n            //plain text response may be wrapped in <pre> tag\n            if (innerHtml && innerHtml.match(/^<pre/i)) {\n                innerHtml = doc.body.firstChild.firstChild.nodeValue;\n            }\n\n            response = handler._parseJsonResponse(innerHtml);\n        }\n        catch (error) {\n            log(\"Error when attempting to parse form upload response (\" + error.message + \")\", \"error\");\n            response = {success: false};\n        }\n\n        return response;\n    }\n\n    /**\n     * Creates form, that will be submitted to iframe\n     */\n    function createForm(id, iframe) {\n        var params = options.paramsStore.get(id),\n            method = options.method.toLowerCase() === \"get\" ? \"GET\" : \"POST\",\n            endpoint = options.endpointStore.get(id),\n            name = getName(id);\n\n        params[options.uuidName] = getUuid(id);\n        params[options.filenameParam] = name;\n\n        return handler._initFormForUpload({\n            method: method,\n            endpoint: endpoint,\n            params: params,\n            paramsInBody: options.paramsInBody,\n            targetName: iframe.name\n        });\n    }\n\n    this.uploadFile = function(id) {\n        var input = handler.getInput(id),\n            iframe = handler._createIframe(id),\n            promise = new qq.Promise(),\n            form;\n\n        form = createForm(id, iframe);\n        form.appendChild(input);\n\n        handler._attachLoadEvent(iframe, function(responseFromMessage) {\n            log(\"iframe loaded\");\n\n            var response = responseFromMessage ? responseFromMessage : getIframeContentJson(id, iframe);\n\n            handler._detachLoadEvent(id);\n\n            //we can't remove an iframe if the iframe doesn't belong to the same domain\n            if (!options.cors.expected) {\n                qq(iframe).remove();\n            }\n\n            if (response.success) {\n                promise.success(response);\n            }\n            else {\n                promise.failure(response);\n            }\n        });\n\n        log(\"Sending upload request for \" + id);\n        form.submit();\n        qq(form).remove();\n\n        return promise;\n    };\n\n    qq.extend(this, new qq.FormUploadHandler({\n        options: {\n            isCors: options.cors.expected,\n            inputName: options.inputName\n        },\n\n        proxy: {\n            onCancel: options.onCancel,\n            getName: getName,\n            getUuid: getUuid,\n            log: log\n        }\n    }));\n};\n","/*globals qq*/\n/**\n * Upload handler used to upload to traditional endpoints.  It depends on File API support, and, therefore,\n * makes use of `XMLHttpRequest` level 2 to upload `File`s and `Blob`s to a generic server.\n *\n * @param spec Options passed from the base handler\n * @param proxy Callbacks & methods used to query for or push out data/changes\n */\nqq.traditional = qq.traditional || {};\nqq.traditional.XhrUploadHandler = function(spec, proxy) {\n    \"use strict\";\n\n    var handler = this,\n        getName = proxy.getName,\n        getSize = proxy.getSize,\n        getUuid = proxy.getUuid,\n        log = proxy.log,\n        multipart = spec.forceMultipart || spec.paramsInBody,\n\n        addChunkingSpecificParams = function(id, params, chunkData) {\n            var size = getSize(id),\n                name = getName(id);\n\n            if (!spec.omitDefaultParams) {\n                params[spec.chunking.paramNames.partIndex] = chunkData.part;\n                params[spec.chunking.paramNames.partByteOffset] = chunkData.start;\n                params[spec.chunking.paramNames.chunkSize] = chunkData.size;\n                params[spec.chunking.paramNames.totalParts] = chunkData.count;\n                params[spec.totalFileSizeName] = size;\n            }\n\n            /**\n             * When a Blob is sent in a multipart request, the filename value in the content-disposition header is either \"blob\"\n             * or an empty string.  So, we will need to include the actual file name as a param in this case.\n             */\n            if (multipart && !spec.omitDefaultParams) {\n                params[spec.filenameParam] = name;\n            }\n        },\n\n        allChunksDoneRequester = new qq.traditional.AllChunksDoneAjaxRequester({\n            cors: spec.cors,\n            endpoint: spec.chunking.success.endpoint,\n            headers: spec.chunking.success.headers,\n            jsonPayload: spec.chunking.success.jsonPayload,\n            log: log,\n            method: spec.chunking.success.method,\n            params: spec.chunking.success.params\n        }),\n\n        createReadyStateChangedHandler = function(id, xhr) {\n            var promise = new qq.Promise();\n\n            xhr.onreadystatechange = function() {\n                if (xhr.readyState === 4) {\n                    var result = onUploadOrChunkComplete(id, xhr);\n\n                    if (result.success) {\n                        promise.success(result.response, xhr);\n                    }\n                    else {\n                        promise.failure(result.response, xhr);\n                    }\n                }\n            };\n\n            return promise;\n        },\n\n        getChunksCompleteParams = function(id) {\n            var params = spec.paramsStore.get(id),\n                name = getName(id),\n                size = getSize(id);\n\n            params[spec.uuidName] = getUuid(id);\n            params[spec.filenameParam] = name;\n            params[spec.totalFileSizeName] = size;\n            params[spec.chunking.paramNames.totalParts] = handler._getTotalChunks(id);\n\n            return params;\n        },\n\n        isErrorUploadResponse = function(xhr, response) {\n            return qq.indexOf([200, 201, 202, 203, 204], xhr.status) < 0 ||\n                (spec.requireSuccessJson && !response.success) ||\n                response.reset;\n        },\n\n        onUploadOrChunkComplete = function(id, xhr) {\n            var response;\n\n            log(\"xhr - server response received for \" + id);\n            log(\"responseText = \" + xhr.responseText);\n\n            response = parseResponse(true, xhr);\n\n            return {\n                success: !isErrorUploadResponse(xhr, response),\n                response: response\n            };\n        },\n\n        // If this is an upload response, we require a JSON payload, otherwise, it is optional.\n        parseResponse = function(upload, xhr) {\n            var response = {};\n\n            try {\n                log(qq.format(\"Received response status {} with body: {}\", xhr.status, xhr.responseText));\n                response = qq.parseJson(xhr.responseText);\n            }\n            catch (error) {\n                upload && spec.requireSuccessJson && log(\"Error when attempting to parse xhr response text (\" + error.message + \")\", \"error\");\n            }\n\n            return response;\n        },\n\n        sendChunksCompleteRequest = function(id) {\n            var promise = new qq.Promise();\n\n            allChunksDoneRequester.complete(\n                    id,\n                    handler._createXhr(id),\n                    getChunksCompleteParams(id),\n                    spec.customHeaders.get(id)\n                )\n                .then(function(xhr) {\n                    promise.success(parseResponse(false, xhr), xhr);\n                }, function(xhr) {\n                    promise.failure(parseResponse(false, xhr), xhr);\n                });\n\n            return promise;\n        },\n\n        setParamsAndGetEntityToSend = function(entityToSendParams) {\n            var fileOrBlob = entityToSendParams.fileOrBlob;\n            var id = entityToSendParams.id;\n            var xhr = entityToSendParams.xhr;\n            var xhrOverrides = entityToSendParams.xhrOverrides || {};\n            var customParams = entityToSendParams.customParams || {};\n            var defaultParams = entityToSendParams.params || {};\n            var xhrOverrideParams = xhrOverrides.params || {};\n            var params;\n\n            var formData = multipart ? new FormData() : null,\n                method = xhrOverrides.method || spec.method,\n                endpoint = xhrOverrides.endpoint || spec.endpointStore.get(id),\n                name = getName(id),\n                size = getSize(id);\n\n            if (spec.omitDefaultParams) {\n                params = qq.extend({}, customParams);\n                qq.extend(params, xhrOverrideParams);\n            }\n            else {\n                params = qq.extend({}, customParams);\n                qq.extend(params, xhrOverrideParams);\n                qq.extend(params, defaultParams);\n\n                params[spec.uuidName] = getUuid(id);\n                params[spec.filenameParam] = name;\n\n                if (multipart) {\n                    params[spec.totalFileSizeName] = size;\n                }\n                else if (!spec.paramsInBody) {\n                    params[spec.inputName] = name;\n                }\n            }\n\n            //build query string\n            if (!spec.paramsInBody) {\n                endpoint = qq.obj2url(params, endpoint);\n            }\n\n            xhr.open(method, endpoint, true);\n\n            if (spec.cors.expected && spec.cors.sendCredentials) {\n                xhr.withCredentials = true;\n            }\n\n            if (multipart) {\n                if (spec.paramsInBody) {\n                    qq.obj2FormData(params, formData);\n                }\n\n                formData.append(spec.inputName, fileOrBlob);\n                return formData;\n            }\n\n            return fileOrBlob;\n        },\n\n        setUploadHeaders = function(headersOptions) {\n            var headerOverrides = headersOptions.headerOverrides;\n            var id = headersOptions.id;\n            var xhr = headersOptions.xhr;\n\n            if (headerOverrides) {\n                qq.each(headerOverrides, function(headerName, headerValue) {\n                    xhr.setRequestHeader(headerName, headerValue);\n                });\n            }\n            else {\n                var extraHeaders = spec.customHeaders.get(id),\n                    fileOrBlob = handler.getFile(id);\n\n                xhr.setRequestHeader(\"Accept\", \"application/json\");\n                xhr.setRequestHeader(\"X-Requested-With\", \"XMLHttpRequest\");\n                xhr.setRequestHeader(\"Cache-Control\", \"no-cache\");\n\n                if (!multipart) {\n                    xhr.setRequestHeader(\"Content-Type\", \"application/octet-stream\");\n                    //NOTE: return mime type in xhr works on chrome 16.0.9 firefox 11.0a2\n                    xhr.setRequestHeader(\"X-Mime-Type\", fileOrBlob.type);\n                }\n\n                qq.each(extraHeaders, function(name, val) {\n                    xhr.setRequestHeader(name, val);\n                });\n            }\n        };\n\n    qq.extend(this, {\n        uploadChunk: function(uploadChunkParams) {\n            var id = uploadChunkParams.id;\n            var chunkIdx = uploadChunkParams.chunkIdx;\n            var overrides = uploadChunkParams.overrides || {};\n            var resuming = uploadChunkParams.resuming;\n\n            var chunkData = handler._getChunkData(id, chunkIdx),\n                xhr = handler._createXhr(id, chunkIdx),\n                promise, toSend, customParams, params = {};\n\n            promise = createReadyStateChangedHandler(id, xhr);\n            handler._registerProgressHandler(id, chunkIdx, chunkData.size);\n            customParams = spec.paramsStore.get(id);\n            addChunkingSpecificParams(id, params, chunkData);\n\n            if (resuming) {\n                params[spec.resume.paramNames.resuming] = true;\n            }\n\n            toSend = setParamsAndGetEntityToSend({\n                fileOrBlob: chunkData.blob,\n                id: id,\n                customParams: customParams,\n                params: params,\n                xhr: xhr,\n                xhrOverrides: overrides\n            });\n\n            setUploadHeaders({\n                headerOverrides: overrides.headers,\n                id: id,\n                xhr: xhr\n            });\n\n            xhr.send(toSend);\n\n            return promise;\n        },\n\n        uploadFile: function(id) {\n            var fileOrBlob = handler.getFile(id),\n                promise, xhr, customParams, toSend;\n\n            xhr = handler._createXhr(id);\n            handler._registerProgressHandler(id);\n            promise = createReadyStateChangedHandler(id, xhr);\n            customParams = spec.paramsStore.get(id);\n\n            toSend = setParamsAndGetEntityToSend({\n                fileOrBlob: fileOrBlob,\n                id: id,\n                customParams: customParams,\n                xhr: xhr\n            });\n\n            setUploadHeaders({\n                id: id,\n                xhr: xhr\n            });\n\n            xhr.send(toSend);\n\n            return promise;\n        }\n    });\n\n    qq.extend(this, new qq.XhrUploadHandler({\n        options: qq.extend({namespace: \"traditional\"}, spec),\n        proxy: qq.extend({getEndpoint: spec.endpointStore.get}, proxy)\n    }));\n\n    qq.override(this, function(super_) {\n        return {\n            finalizeChunks: function(id) {\n                proxy.onFinalizing(id);\n\n                if (spec.chunking.success.endpoint) {\n                    return sendChunksCompleteRequest(id);\n                }\n                else {\n                    return super_.finalizeChunks(id, qq.bind(parseResponse, this, true));\n                }\n            }\n        };\n    });\n};\n","/*globals qq*/\n/**\n * Ajax requester used to send a POST to a traditional endpoint once all chunks for a specific file have uploaded\n * successfully.\n *\n * @param o Options from the caller - will override the defaults.\n * @constructor\n */\nqq.traditional.AllChunksDoneAjaxRequester = function(o) {\n    \"use strict\";\n\n    var requester,\n        options = {\n            cors: {\n                allowXdr: false,\n                expected: false,\n                sendCredentials: false\n            },\n            endpoint: null,\n            log: function(str, level) {},\n            method: \"POST\"\n        },\n        promises = {},\n        endpointHandler = {\n            get: function(id) {\n                if (qq.isFunction(options.endpoint)) {\n                    return options.endpoint(id);\n                }\n\n                return options.endpoint;\n            }\n        };\n\n    qq.extend(options, o);\n\n    requester = qq.extend(this, new qq.AjaxRequester({\n        acceptHeader: \"application/json\",\n        contentType: options.jsonPayload ? \"application/json\" : \"application/x-www-form-urlencoded\",\n        validMethods: [options.method],\n        method: options.method,\n        endpointStore: endpointHandler,\n        allowXRequestedWithAndCacheControl: false,\n        cors: options.cors,\n        log: options.log,\n        onComplete: function(id, xhr, isError) {\n            var promise = promises[id];\n\n            delete promises[id];\n\n            if (isError) {\n                promise.failure(xhr);\n            }\n            else {\n                promise.success(xhr);\n            }\n        }\n    }));\n\n    qq.extend(this, {\n        complete: function(id, xhr, params, headers) {\n            var promise = new qq.Promise();\n\n            options.log(\"Submitting All Chunks Done request for \" + id);\n\n            promises[id] = promise;\n\n            requester.initTransport(id)\n                .withParams(options.params(id) || params)\n                .withHeaders(options.headers(id) || headers)\n                .send(xhr);\n\n            return promise;\n        }\n    });\n};\n"]}