2015-03-31 04:53:53 +08:00
|
|
|
/* RainLoop Webmail (c) RainLoop Team | MIT */
|
2020-07-27 22:58:04 +08:00
|
|
|
(()=>{
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var
|
2020-08-12 06:25:36 +08:00
|
|
|
iDefLimit = 20,
|
2020-07-27 22:58:04 +08:00
|
|
|
$ = jQuery;
|
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
const
|
2020-08-12 17:49:40 +08:00
|
|
|
defined = v => undefined !== v,
|
|
|
|
/**
|
|
|
|
* @param {*} aItems
|
|
|
|
* @param {Function} fFileCallback
|
|
|
|
* @param {number=} iLimit = 20
|
|
|
|
* @param {Function=} fLimitCallback
|
|
|
|
*/
|
|
|
|
getDataFromFiles = (aItems, fFileCallback, iLimit, fLimitCallback) =>
|
|
|
|
{
|
|
|
|
if (aItems && aItems.length)
|
|
|
|
{
|
|
|
|
var
|
|
|
|
iInputLimit = 0,
|
|
|
|
oFile = null,
|
|
|
|
bUseLimit = false,
|
|
|
|
bCallLimit = false
|
|
|
|
;
|
|
|
|
|
|
|
|
iLimit = defined(iLimit) ? parseInt(iLimit || 0, 10) : iDefLimit;
|
|
|
|
iInputLimit = iLimit;
|
|
|
|
bUseLimit = 0 < iLimit;
|
|
|
|
Array.from(aItems).forEach(oItem => {
|
|
|
|
if (oItem)
|
|
|
|
{
|
|
|
|
if (!bUseLimit) {
|
|
|
|
if (0 <= --iLimit)
|
|
|
|
{
|
|
|
|
oFile = Utils.getDataFromFile(oItem);
|
|
|
|
oFile && fFileCallback(oFile);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (!bCallLimit)
|
|
|
|
{
|
|
|
|
if (0 > iLimit && fLimitCallback)
|
|
|
|
{
|
|
|
|
bCallLimit = true;
|
|
|
|
fLimitCallback(iInputLimit);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
/**
|
|
|
|
* @param {number=} iLen
|
|
|
|
* @return {string}
|
|
|
|
*/
|
|
|
|
fakeMd5 = iLen =>
|
|
|
|
{
|
|
|
|
var
|
|
|
|
sResult = '',
|
|
|
|
sLine = '0123456789abcdefghijklmnopqrstuvwxyz'
|
|
|
|
;
|
|
|
|
|
|
|
|
iLen = defined(iLen) ? parseInt(iLen || 0, 10) : 32;
|
|
|
|
|
|
|
|
while (iLen--)
|
|
|
|
sResult += sLine.substr(Math.round(Math.random() * 36), 1);
|
|
|
|
|
|
|
|
return sResult;
|
|
|
|
};
|
2020-08-12 06:25:36 +08:00
|
|
|
|
|
|
|
var Utils = {
|
|
|
|
/**
|
|
|
|
* @param {*} oParent
|
|
|
|
* @param {*} oDescendant
|
|
|
|
*
|
|
|
|
* @return {boolean}
|
|
|
|
*/
|
|
|
|
contains : (oParent, oDescendant) =>
|
2020-07-27 22:58:04 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
if (oParent && oDescendant)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
if (oParent === oDescendant)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (oParent.contains)
|
|
|
|
{
|
|
|
|
return oParent.contains(oDescendant);
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-07-27 22:58:04 +08:00
|
|
|
/*jshint bitwise: false*/
|
2020-08-12 06:25:36 +08:00
|
|
|
return oDescendant.compareDocumentPosition ?
|
2020-07-27 22:58:04 +08:00
|
|
|
!!(oDescendant.compareDocumentPosition(oParent) & 8) : false;
|
|
|
|
/*jshint bitwise: true*/
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
return false;
|
|
|
|
},
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
mainClearTimeout : iTimer =>
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
if (0 < iTimer)
|
|
|
|
{
|
|
|
|
clearTimeout(iTimer);
|
|
|
|
}
|
2020-07-27 22:58:04 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
iTimer = 0;
|
|
|
|
},
|
2020-07-27 22:58:04 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
/**
|
|
|
|
* @param {Event} oEvent
|
|
|
|
* @return {?Event}
|
|
|
|
*/
|
|
|
|
getEvent : oEvent =>
|
|
|
|
{
|
|
|
|
oEvent = (oEvent && (oEvent.originalEvent ?
|
|
|
|
oEvent.originalEvent : oEvent)) || window.event;
|
|
|
|
|
|
|
|
return oEvent.dataTransfer ? oEvent : null;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Object} oValues
|
|
|
|
* @param {string} sKey
|
|
|
|
* @param {?} mDefault
|
|
|
|
* @return {?}
|
|
|
|
*/
|
|
|
|
getValue : (oValues, sKey, mDefault) => (!oValues || !sKey || !defined(oValues[sKey])) ? mDefault : oValues[sKey],
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Function} fFunction
|
|
|
|
* @param {Object=} oScope
|
|
|
|
* @return {Function}
|
|
|
|
*/
|
|
|
|
scopeBind : (fFunction, oScope) => (...args) => {
|
|
|
|
return fFunction.apply(defined(oScope) ? oScope : null,
|
|
|
|
Array.prototype.slice.call(args));
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return {string}
|
|
|
|
*/
|
2020-08-12 17:49:40 +08:00
|
|
|
getNewUid : () => 'jua-uid-' + fakeMd5(16) + '-' + (new Date()).getTime().toString(),
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
/**
|
|
|
|
* @param {*} oFile
|
|
|
|
* @return {Object}
|
|
|
|
*/
|
|
|
|
getDataFromFile : oFile =>
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
var
|
|
|
|
sFileName = defined(oFile.fileName) ? oFile.fileName : (defined(oFile.name) ? oFile.name : null),
|
|
|
|
iSize = defined(oFile.fileSize) ? oFile.fileSize : (defined(oFile.size) ? oFile.size : null),
|
|
|
|
sType = defined(oFile.type) ? oFile.type : null
|
|
|
|
;
|
2020-07-27 22:58:04 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
if (sFileName.charAt(0) === '/')
|
|
|
|
{
|
|
|
|
sFileName = sFileName.substr(1);
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
if (!sType && 0 === iSize)
|
|
|
|
{
|
|
|
|
return null; // Folder
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
return {
|
|
|
|
'FileName': sFileName,
|
|
|
|
'Size': iSize,
|
|
|
|
'Type': sType,
|
|
|
|
'Folder': '',
|
|
|
|
'File' : oFile
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {*} oInput
|
|
|
|
* @param {Function} fFileCallback
|
|
|
|
* @param {number=} iLimit = 20
|
|
|
|
* @param {Function=} fLimitCallback
|
|
|
|
*/
|
|
|
|
getDataFromInput : (oInput, fFileCallback, iLimit, fLimitCallback) =>
|
2020-07-27 22:58:04 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
var aFiles = oInput && 0 < oInput.files.length ? oInput.files : null;
|
2020-08-12 06:25:36 +08:00
|
|
|
if (aFiles)
|
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
getDataFromFiles(aFiles, fFileCallback, iLimit, fLimitCallback);
|
2020-08-12 06:25:36 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
fFileCallback({
|
|
|
|
'FileName': oInput.value.split('\\').pop().split('/').pop(),
|
|
|
|
'Size': null,
|
|
|
|
'Type': null,
|
|
|
|
'Folder': '',
|
|
|
|
'File' : null
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
eventContainsFiles : oEvent =>
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
var bResult = false;
|
|
|
|
if (oEvent && oEvent.dataTransfer && oEvent.dataTransfer.types && oEvent.dataTransfer.types.length)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
var
|
|
|
|
iIindex = 0,
|
|
|
|
iLen = oEvent.dataTransfer.types.length
|
|
|
|
;
|
|
|
|
|
|
|
|
for (; iIindex < iLen; iIindex++)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
if (oEvent.dataTransfer.types[iIindex].toLowerCase() === 'files')
|
|
|
|
{
|
|
|
|
bResult = true;
|
|
|
|
break;
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
return bResult;
|
|
|
|
},
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
/**
|
|
|
|
* @param {Event} oEvent
|
|
|
|
* @param {Function} fFileCallback
|
|
|
|
* @param {number=} iLimit = 20
|
|
|
|
* @param {Function=} fLimitCallback
|
|
|
|
*/
|
|
|
|
getDataFromDragEvent : (oEvent, fFileCallback, iLimit, fLimitCallback) =>
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
var aFiles = null;
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
oEvent = Utils.getEvent(oEvent);
|
|
|
|
if (oEvent && Utils.eventContainsFiles(oEvent))
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
aFiles = (Utils.getValue(oEvent, 'files', null) || (oEvent.dataTransfer ?
|
|
|
|
Utils.getValue(oEvent.dataTransfer, 'files', null) : null));
|
2020-07-27 22:58:04 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
getDataFromFiles(aFiles, fFileCallback, iLimit, fLimitCallback);
|
2020-08-12 06:25:36 +08:00
|
|
|
}
|
|
|
|
},
|
2020-07-27 22:58:04 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
createNextLabel : () =>
|
|
|
|
{
|
|
|
|
return $('<label style="' +
|
|
|
|
'position: absolute; background-color:#fff; right: 0px; top: 0px; left: 0px; bottom: 0px; margin: 0px; padding: 0px; cursor: pointer;' +
|
|
|
|
'"></label>').css({
|
|
|
|
'opacity': 0
|
|
|
|
});
|
|
|
|
},
|
2020-07-27 22:58:04 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
createNextInput : () => $('<input type="file" tabindex="-1" hidefocus="hidefocus" style="position: absolute; left: -9999px;" />'),
|
2020-07-27 22:58:04 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
/**
|
|
|
|
* @param {string=} sName
|
|
|
|
* @param {boolean=} bMultiple = true
|
|
|
|
* @return {?Object}
|
|
|
|
*/
|
|
|
|
getNewInput : (sName, bMultiple) =>
|
2020-07-27 22:58:04 +08:00
|
|
|
{
|
2020-08-12 06:25:36 +08:00
|
|
|
sName = defined(sName) ? sName.toString() : '';
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
var oLocal = Utils.createNextInput();
|
|
|
|
if (0 < sName.length)
|
|
|
|
{
|
|
|
|
oLocal.attr('name', sName);
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
if (defined(bMultiple) ? bMultiple : true)
|
|
|
|
{
|
|
|
|
oLocal.prop('multiple', true);
|
|
|
|
}
|
2020-07-27 22:58:04 +08:00
|
|
|
|
2020-08-12 06:25:36 +08:00
|
|
|
return oLocal;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {?} mStringOrFunction
|
|
|
|
* @param {Array=} aFunctionParams
|
|
|
|
* @return {string}
|
|
|
|
*/
|
|
|
|
getStringOrCallFunction : (mStringOrFunction, aFunctionParams) => $.isFunction(mStringOrFunction) ?
|
|
|
|
mStringOrFunction.apply(null, Array.isArray(aFunctionParams) ? aFunctionParams : []).toString() :
|
|
|
|
mStringOrFunction.toString()
|
2020-07-27 22:58:04 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @constructor
|
|
|
|
* @param {Jua} oJua
|
|
|
|
* @param {Object} oOptions
|
|
|
|
*/
|
2020-08-12 17:49:40 +08:00
|
|
|
class AjaxDriver
|
2020-07-27 22:58:04 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
constructor(oJua, oOptions)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
this.oXhrs = {};
|
|
|
|
this.oUids = {};
|
|
|
|
this.oJua = oJua;
|
|
|
|
this.oOptions = oOptions;
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
/**
|
|
|
|
* @param {string} sUid
|
|
|
|
*/
|
|
|
|
regTaskUid(sUid)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
this.oUids[sUid] = true;
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
/**
|
|
|
|
* @param {string} sUid
|
|
|
|
* @param {?} oFileInfo
|
|
|
|
* @param {Function} fCallback
|
|
|
|
*/
|
|
|
|
uploadTask(sUid, oFileInfo, fCallback)
|
|
|
|
{
|
|
|
|
if (false === this.oUids[sUid] || !oFileInfo || !oFileInfo['File'])
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
fCallback(null, sUid);
|
|
|
|
return false;
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
try
|
|
|
|
{
|
|
|
|
var
|
|
|
|
self = this,
|
|
|
|
oXhr = new XMLHttpRequest(),
|
|
|
|
oFormData = new FormData(),
|
|
|
|
sAction = Utils.getValue(this.oOptions, 'action', ''),
|
|
|
|
aHidden = Utils.getValue(this.oOptions, 'hidden', {}),
|
|
|
|
fStartFunction = this.oJua.getEvent('onStart'),
|
|
|
|
fCompleteFunction = this.oJua.getEvent('onComplete'),
|
|
|
|
fProgressFunction = this.oJua.getEvent('onProgress')
|
|
|
|
;
|
|
|
|
|
|
|
|
oXhr.open('POST', sAction, true);
|
|
|
|
|
|
|
|
if (fProgressFunction && oXhr.upload)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
oXhr.upload.onprogress = function (oEvent) {
|
|
|
|
if (oEvent && oEvent.lengthComputable && defined(oEvent.loaded) && defined(oEvent.total))
|
|
|
|
{
|
|
|
|
fProgressFunction(sUid, oEvent.loaded, oEvent.total);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
oXhr.onreadystatechange = function () {
|
|
|
|
if (4 === oXhr.readyState && 200 === oXhr.status)
|
|
|
|
{
|
|
|
|
if (fCompleteFunction)
|
2020-07-27 22:58:04 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
var
|
|
|
|
bResult = false,
|
|
|
|
oResult = null
|
|
|
|
;
|
|
|
|
|
|
|
|
try
|
|
|
|
{
|
|
|
|
oResult = JSON.parse(oXhr.responseText);
|
|
|
|
bResult = true;
|
|
|
|
}
|
|
|
|
catch (oException)
|
|
|
|
{
|
|
|
|
oResult = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
fCompleteFunction(sUid, bResult, oResult);
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
|
|
|
|
if (defined(self.oXhrs[sUid]))
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
self.oXhrs[sUid] = null;
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
fCallback(null, sUid);
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
else
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
if (4 === oXhr.readyState)
|
|
|
|
{
|
|
|
|
fCompleteFunction(sUid, false, null);
|
|
|
|
fCallback(null, sUid);
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
};
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
if (fStartFunction)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
fStartFunction(sUid);
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
oFormData.append('jua-post-type', 'ajax');
|
|
|
|
oFormData.append(Utils.getValue(this.oOptions, 'name', 'juaFile'), oFileInfo['File']);
|
|
|
|
$.each(aHidden, function (sKey, sValue) {
|
|
|
|
oFormData.append(sKey, Utils.getStringOrCallFunction(sValue, [oFileInfo]));
|
|
|
|
});
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
oXhr.send(oFormData);
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
this.oXhrs[sUid] = oXhr;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
catch (oError)
|
|
|
|
{
|
|
|
|
}
|
2020-03-11 21:17:52 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
fCallback(null, sUid);
|
|
|
|
return false;
|
2020-07-27 22:58:04 +08:00
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
generateNewInput(oClickElement)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
var
|
|
|
|
self = this,
|
|
|
|
oLabel = null,
|
|
|
|
oInput = null
|
|
|
|
;
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
if (oClickElement)
|
|
|
|
{
|
|
|
|
oInput = Utils.getNewInput('', !Utils.getValue(this.oOptions, 'disableMultiple', false));
|
|
|
|
oLabel = Utils.createNextLabel();
|
|
|
|
oLabel.append(oInput);
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
$(oClickElement).append(oLabel);
|
|
|
|
|
|
|
|
oInput
|
|
|
|
.on('click', function () {
|
|
|
|
var fOn = self.oJua.getEvent('onDialog');
|
|
|
|
if (fOn)
|
|
|
|
{
|
|
|
|
fOn();
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.on('change', function () {
|
|
|
|
Utils.getDataFromInput(this, function (oFile) {
|
|
|
|
self.oJua.addNewFile(oFile);
|
|
|
|
self.generateNewInput(oClickElement);
|
|
|
|
|
|
|
|
setTimeout(function () {
|
|
|
|
oLabel.remove();
|
|
|
|
}, 10);
|
|
|
|
},
|
|
|
|
Utils.getValue(self.oOptions, 'multipleSizeLimit', iDefLimit),
|
|
|
|
self.oJua.getEvent('onLimitReached')
|
|
|
|
);
|
|
|
|
})
|
|
|
|
;
|
|
|
|
}
|
2020-07-27 22:58:04 +08:00
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
cancel(sUid)
|
2020-07-27 22:58:04 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
this.oUids[sUid] = false;
|
|
|
|
if (this.oXhrs[sUid])
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
try
|
|
|
|
{
|
|
|
|
if (this.oXhrs[sUid].abort)
|
|
|
|
{
|
|
|
|
this.oXhrs[sUid].abort();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (oError)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
this.oXhrs[sUid] = null;
|
|
|
|
}
|
2020-07-27 22:58:04 +08:00
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
AjaxDriver.prototype.oXhrs = {};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
AjaxDriver.prototype.oUids = {};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {?Jua}
|
|
|
|
*/
|
|
|
|
AjaxDriver.prototype.oJua = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
AjaxDriver.prototype.oOptions = {};
|
|
|
|
|
2020-07-27 22:58:04 +08:00
|
|
|
|
|
|
|
function queue(a) {
|
|
|
|
function l() {
|
|
|
|
if (g && d < a) {
|
|
|
|
var b = g,
|
|
|
|
c = b[0],
|
|
|
|
f = Array.prototype.slice.call(b, 1),
|
|
|
|
m = b.index;
|
|
|
|
g === h ? g = h = null : g = g.next, ++d, f.push(function (a, b) {
|
|
|
|
--d;
|
|
|
|
if (i) return;
|
|
|
|
a ? e && k(i = a, e = j = g = h = null) : (j[m] = b, --e ? l() : k(null, j))
|
|
|
|
}),
|
|
|
|
c.apply(null, f)
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-07-27 22:58:04 +08:00
|
|
|
}
|
|
|
|
var c = {},
|
|
|
|
d = 0,
|
|
|
|
e = 0,
|
|
|
|
f = -1,
|
|
|
|
g, h, i = null,
|
|
|
|
j = [],
|
|
|
|
k = ()=>{};
|
|
|
|
return arguments.length < 1 && (a = Infinity), c.defer = function () {
|
|
|
|
if (!i) {
|
|
|
|
var a = arguments;
|
|
|
|
a.index = ++f, h ? (h.next = a, h = h.next) : g = h = a, ++e, l()
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-07-27 22:58:04 +08:00
|
|
|
return c
|
|
|
|
}, c.await = function (a) {
|
|
|
|
return k = a, e || k(i, j), c
|
|
|
|
}, c
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @constructor
|
|
|
|
* @param {Object=} oOptions
|
|
|
|
*/
|
2020-08-12 17:49:40 +08:00
|
|
|
class Jua
|
2020-07-27 22:58:04 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
constructor(oOptions)
|
2020-07-27 22:58:04 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
oOptions = defined(oOptions) ? oOptions : {};
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
var
|
|
|
|
self = this,
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
$ = jQuery
|
|
|
|
;
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
self.bEnableDnD = true;
|
|
|
|
|
|
|
|
self.oEvents = {
|
|
|
|
'onDialog': null,
|
|
|
|
'onSelect': null,
|
|
|
|
'onStart': null,
|
|
|
|
'onComplete': null,
|
|
|
|
'onCompleteAll': null,
|
|
|
|
'onProgress': null,
|
|
|
|
'onDragEnter': null,
|
|
|
|
'onDragLeave': null,
|
|
|
|
'onDrop': null,
|
|
|
|
'onBodyDragEnter': null,
|
|
|
|
'onBodyDragLeave': null,
|
|
|
|
'onLimitReached': null
|
|
|
|
};
|
|
|
|
|
|
|
|
self.oOptions = {
|
|
|
|
'action': '',
|
|
|
|
'name': '',
|
|
|
|
'hidden': {},
|
|
|
|
'queueSize': 10,
|
|
|
|
'clickElement': false,
|
|
|
|
'dragAndDropElement': false,
|
|
|
|
'dragAndDropBodyElement': false,
|
|
|
|
'disableDragAndDrop': false,
|
|
|
|
'disableMultiple': false,
|
|
|
|
'disableDocumentDropPrevent': false,
|
|
|
|
'multipleSizeLimit': 50
|
|
|
|
};
|
|
|
|
Object.entries(oOptions).forEach(([key, value])=>self.oOptions[key]=value);
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
self.oQueue = queue(parseInt(Utils.getValue(self.oOptions, 'queueSize', 10) || 0, 10));
|
|
|
|
if (self.runEvent('onCompleteAll'))
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
self.oQueue.await(function () {
|
|
|
|
self.runEvent('onCompleteAll');
|
|
|
|
});
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
self.oDriver = new AjaxDriver(self, self.oOptions);
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
self.oClickElement = Utils.getValue(self.oOptions, 'clickElement', null);
|
|
|
|
|
|
|
|
if (self.oClickElement)
|
|
|
|
{
|
|
|
|
$(self.oClickElement).css({
|
|
|
|
'position': 'relative',
|
|
|
|
'overflow': 'hidden'
|
|
|
|
});
|
|
|
|
|
|
|
|
if ('inline' === $(this.oClickElement).css('display'))
|
|
|
|
{
|
|
|
|
$(this.oClickElement).css('display', 'inline-block');
|
|
|
|
}
|
|
|
|
|
|
|
|
this.oDriver.generateNewInput(this.oClickElement);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Utils.getValue(this.oOptions, 'dragAndDropElement', false))
|
|
|
|
{
|
|
|
|
(function (self) {
|
|
|
|
var
|
|
|
|
$doc = $(document),
|
|
|
|
oBigDropZone = $(Utils.getValue(self.oOptions, 'dragAndDropBodyElement', false) || $doc),
|
|
|
|
oDragAndDropElement = Utils.getValue(self.oOptions, 'dragAndDropElement', false),
|
|
|
|
fHandleDragOver = function (oEvent) {
|
|
|
|
if (self.bEnableDnD && oEvent)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
oEvent = Utils.getEvent(oEvent);
|
|
|
|
if (oEvent && oEvent.dataTransfer && Utils.eventContainsFiles(oEvent))
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
try
|
|
|
|
{
|
|
|
|
var sEffect = oEvent.dataTransfer.effectAllowed;
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
Utils.mainClearTimeout(self.iDocTimer);
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
oEvent.dataTransfer.dropEffect = (sEffect === 'move' || sEffect === 'linkMove') ? 'move' : 'copy';
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
oEvent.stopPropagation();
|
|
|
|
oEvent.preventDefault();
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
oBigDropZone.trigger('dragover', oEvent);
|
|
|
|
}
|
|
|
|
catch (oExc) {}
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
},
|
|
|
|
fHandleDrop = function (oEvent) {
|
|
|
|
if (self.bEnableDnD && oEvent)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
oEvent = Utils.getEvent(oEvent);
|
|
|
|
if (oEvent && Utils.eventContainsFiles(oEvent))
|
|
|
|
{
|
|
|
|
oEvent.preventDefault();
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
Utils.getDataFromDragEvent(oEvent, function (oFile) {
|
|
|
|
if (oFile)
|
|
|
|
{
|
|
|
|
self.runEvent('onDrop', [oFile, oEvent]);
|
|
|
|
self.addNewFile(oFile);
|
|
|
|
Utils.mainClearTimeout(self.iDocTimer);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Utils.getValue(self.oOptions, 'multipleSizeLimit', iDefLimit),
|
|
|
|
self.getEvent('onLimitReached')
|
|
|
|
);
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
self.runEvent('onDragLeave', [oEvent]);
|
|
|
|
},
|
|
|
|
fHandleDragEnter = function (oEvent) {
|
|
|
|
if (self.bEnableDnD && oEvent)
|
2020-07-27 22:58:04 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
oEvent = Utils.getEvent(oEvent);
|
|
|
|
if (oEvent && Utils.eventContainsFiles(oEvent))
|
2020-07-27 22:58:04 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
Utils.mainClearTimeout(self.iDocTimer);
|
2020-07-27 22:58:04 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
oEvent.preventDefault();
|
|
|
|
self.runEvent('onDragEnter', [oDragAndDropElement, oEvent]);
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
},
|
|
|
|
fHandleDragLeave = function (oEvent) {
|
2015-03-31 04:53:53 +08:00
|
|
|
if (self.bEnableDnD && oEvent)
|
|
|
|
{
|
|
|
|
oEvent = Utils.getEvent(oEvent);
|
2020-08-12 17:49:40 +08:00
|
|
|
if (oEvent)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
var oRelatedTarget = document['elementFromPoint'] ? document['elementFromPoint'](oEvent['clientX'], oEvent['clientY']) : null;
|
|
|
|
if (oRelatedTarget && Utils.contains(this, oRelatedTarget))
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
return;
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
|
|
|
|
Utils.mainClearTimeout(self.iDocTimer);
|
|
|
|
self.runEvent('onDragLeave', [oDragAndDropElement, oEvent]);
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
|
|
|
|
return;
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
}
|
|
|
|
;
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
if (oDragAndDropElement)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
if (!Utils.getValue(self.oOptions, 'disableDocumentDropPrevent', false))
|
|
|
|
{
|
|
|
|
$doc.on('dragover', function (oEvent) {
|
2015-03-31 04:53:53 +08:00
|
|
|
if (self.bEnableDnD && oEvent)
|
|
|
|
{
|
|
|
|
oEvent = Utils.getEvent(oEvent);
|
2020-08-12 17:49:40 +08:00
|
|
|
if (oEvent && oEvent.dataTransfer && Utils.eventContainsFiles(oEvent))
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
try
|
|
|
|
{
|
|
|
|
oEvent.dataTransfer.dropEffect = 'none';
|
|
|
|
oEvent.preventDefault();
|
|
|
|
}
|
|
|
|
catch (oExc) {}
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (oBigDropZone && oBigDropZone[0])
|
|
|
|
{
|
|
|
|
oBigDropZone
|
|
|
|
.on('dragover', function (oEvent) {
|
|
|
|
if (self.bEnableDnD && oEvent)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
|
|
|
Utils.mainClearTimeout(self.iDocTimer);
|
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
})
|
|
|
|
.on('dragenter', function (oEvent) {
|
|
|
|
if (self.bEnableDnD && oEvent)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
oEvent = Utils.getEvent(oEvent);
|
|
|
|
if (oEvent && Utils.eventContainsFiles(oEvent))
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
Utils.mainClearTimeout(self.iDocTimer);
|
2015-03-31 04:53:53 +08:00
|
|
|
oEvent.preventDefault();
|
2020-08-12 17:49:40 +08:00
|
|
|
|
|
|
|
self.runEvent('onBodyDragEnter', [oEvent]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.on('dragleave', function (oEvent) {
|
|
|
|
if (self.bEnableDnD && oEvent)
|
|
|
|
{
|
|
|
|
oEvent = Utils.getEvent(oEvent);
|
|
|
|
if (oEvent)
|
|
|
|
{
|
|
|
|
Utils.mainClearTimeout(self.iDocTimer);
|
|
|
|
self.iDocTimer = setTimeout(function () {
|
|
|
|
self.runEvent('onBodyDragLeave', [oEvent]);
|
|
|
|
}, 200);
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.on('drop', function (oEvent) {
|
|
|
|
if (self.bEnableDnD && oEvent)
|
|
|
|
{
|
|
|
|
oEvent = Utils.getEvent(oEvent);
|
|
|
|
if (oEvent)
|
|
|
|
{
|
|
|
|
var bFiles = Utils.eventContainsFiles(oEvent);
|
|
|
|
if (bFiles)
|
|
|
|
{
|
|
|
|
oEvent.preventDefault();
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
self.runEvent('onBodyDragLeave', [oEvent]);
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
return !bFiles;
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
return false;
|
|
|
|
})
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
|
|
|
$(oDragAndDropElement)
|
|
|
|
.on('dragenter', fHandleDragEnter)
|
|
|
|
.on('dragover', fHandleDragOver)
|
|
|
|
.on('dragleave', fHandleDragLeave)
|
|
|
|
.on('drop', fHandleDrop)
|
2015-03-31 04:53:53 +08:00
|
|
|
;
|
|
|
|
}
|
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
}(self));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
self.bEnableDnD = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} sName
|
|
|
|
* @param {Function} fFunc
|
|
|
|
*/
|
|
|
|
on(sName, fFunc)
|
|
|
|
{
|
|
|
|
this.oEvents[sName] = fFunc;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} sName
|
|
|
|
* @param {string=} aArgs
|
|
|
|
*/
|
|
|
|
runEvent(sName, aArgs)
|
|
|
|
{
|
|
|
|
if (this.oEvents[sName])
|
|
|
|
{
|
|
|
|
this.oEvents[sName].apply(null, aArgs || []);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} sName
|
|
|
|
*/
|
|
|
|
getEvent(sName)
|
|
|
|
{
|
|
|
|
return this.oEvents[sName] || null;
|
|
|
|
}
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-08-12 17:49:40 +08:00
|
|
|
/**
|
|
|
|
* @param {string} sUid
|
|
|
|
*/
|
|
|
|
cancel(sUid)
|
|
|
|
{
|
|
|
|
this.oDriver.cancel(sUid);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {boolean} bEnabled
|
|
|
|
*/
|
|
|
|
setDragAndDropEnabledStatus(bEnabled)
|
|
|
|
{
|
|
|
|
this.bEnableDnD = !!bEnabled;
|
2015-03-31 04:53:53 +08:00
|
|
|
}
|
2020-08-12 17:49:40 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {Object} oFileInfo
|
|
|
|
*/
|
|
|
|
addNewFile(oFileInfo)
|
2015-03-31 04:53:53 +08:00
|
|
|
{
|
2020-08-12 17:49:40 +08:00
|
|
|
this.addFile(Utils.getNewUid(), oFileInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} sUid
|
|
|
|
* @param {Object} oFileInfo
|
|
|
|
*/
|
|
|
|
addFile(sUid, oFileInfo)
|
|
|
|
{
|
|
|
|
var fOnSelect = this.getEvent('onSelect');
|
|
|
|
if (oFileInfo && (!fOnSelect || (false !== fOnSelect(sUid, oFileInfo))))
|
|
|
|
{
|
|
|
|
this.oDriver.regTaskUid(sUid);
|
|
|
|
this.oQueue.defer(Utils.scopeBind(this.oDriver.uploadTask, this.oDriver), sUid, oFileInfo);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.oDriver.cancel(sUid);
|
|
|
|
}
|
2020-07-27 22:58:04 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {boolean}
|
|
|
|
*/
|
|
|
|
Jua.prototype.bEnableDnD = true;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {number}
|
|
|
|
*/
|
|
|
|
Jua.prototype.iDocTimer = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
Jua.prototype.oOptions = {};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
Jua.prototype.oEvents = {};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {?Object}
|
|
|
|
*/
|
|
|
|
Jua.prototype.oQueue = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {?Object}
|
|
|
|
*/
|
|
|
|
Jua.prototype.oDriver = null;
|
|
|
|
|
|
|
|
window.Jua = Jua;
|
2015-03-31 04:53:53 +08:00
|
|
|
|
2020-07-27 22:58:04 +08:00
|
|
|
})();
|