/* RainLoop Webmail (c) RainLoop Team | MIT */
(()=>{
'use strict';
var
iDefLimit = 20,
$ = jQuery;
const
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;
};
var Utils = {
/**
* @param {*} oParent
* @param {*} oDescendant
*
* @return {boolean}
*/
contains : (oParent, oDescendant) =>
{
if (oParent && oDescendant)
{
if (oParent === oDescendant)
{
return true;
}
if (oParent.contains)
{
return oParent.contains(oDescendant);
}
/*jshint bitwise: false*/
return oDescendant.compareDocumentPosition ?
!!(oDescendant.compareDocumentPosition(oParent) & 8) : false;
/*jshint bitwise: true*/
}
return false;
},
mainClearTimeout : iTimer =>
{
if (0 < iTimer)
{
clearTimeout(iTimer);
}
iTimer = 0;
},
/**
* @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}
*/
getNewUid : () => 'jua-uid-' + fakeMd5(16) + '-' + (new Date()).getTime().toString(),
/**
* @param {*} oFile
* @return {Object}
*/
getDataFromFile : oFile =>
{
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
;
if (sFileName.charAt(0) === '/')
{
sFileName = sFileName.substr(1);
}
if (!sType && 0 === iSize)
{
return null; // Folder
}
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) =>
{
var aFiles = oInput && 0 < oInput.files.length ? oInput.files : null;
if (aFiles)
{
getDataFromFiles(aFiles, fFileCallback, iLimit, fLimitCallback);
}
else
{
fFileCallback({
'FileName': oInput.value.split('\\').pop().split('/').pop(),
'Size': null,
'Type': null,
'Folder': '',
'File' : null
});
}
},
eventContainsFiles : oEvent =>
{
var bResult = false;
if (oEvent && oEvent.dataTransfer && oEvent.dataTransfer.types && oEvent.dataTransfer.types.length)
{
var
iIindex = 0,
iLen = oEvent.dataTransfer.types.length
;
for (; iIindex < iLen; iIindex++)
{
if (oEvent.dataTransfer.types[iIindex].toLowerCase() === 'files')
{
bResult = true;
break;
}
}
}
return bResult;
},
/**
* @param {Event} oEvent
* @param {Function} fFileCallback
* @param {number=} iLimit = 20
* @param {Function=} fLimitCallback
*/
getDataFromDragEvent : (oEvent, fFileCallback, iLimit, fLimitCallback) =>
{
var aFiles = null;
oEvent = Utils.getEvent(oEvent);
if (oEvent && Utils.eventContainsFiles(oEvent))
{
aFiles = (Utils.getValue(oEvent, 'files', null) || (oEvent.dataTransfer ?
Utils.getValue(oEvent.dataTransfer, 'files', null) : null));
getDataFromFiles(aFiles, fFileCallback, iLimit, fLimitCallback);
}
},
createNextLabel : () =>
{
return $('').css({
'opacity': 0
});
},
createNextInput : () => $(''),
/**
* @param {string=} sName
* @param {boolean=} bMultiple = true
* @return {?Object}
*/
getNewInput : (sName, bMultiple) =>
{
sName = defined(sName) ? sName.toString() : '';
var oLocal = Utils.createNextInput();
if (0 < sName.length)
{
oLocal.attr('name', sName);
}
if (defined(bMultiple) ? bMultiple : true)
{
oLocal.prop('multiple', true);
}
return oLocal;
},
/**
* @param {?} mStringOrFunction
* @param {Array=} aFunctionParams
* @return {string}
*/
getStringOrCallFunction : (mStringOrFunction, aFunctionParams) => $.isFunction(mStringOrFunction) ?
mStringOrFunction.apply(null, Array.isArray(aFunctionParams) ? aFunctionParams : []).toString() :
mStringOrFunction.toString()
};
/**
* @constructor
* @param {Jua} oJua
* @param {Object} oOptions
*/
class AjaxDriver
{
constructor(oJua, oOptions)
{
this.oXhrs = {};
this.oUids = {};
this.oJua = oJua;
this.oOptions = oOptions;
}
/**
* @param {string} sUid
*/
regTaskUid(sUid)
{
this.oUids[sUid] = true;
}
/**
* @param {string} sUid
* @param {?} oFileInfo
* @param {Function} fCallback
*/
uploadTask(sUid, oFileInfo, fCallback)
{
if (false === this.oUids[sUid] || !oFileInfo || !oFileInfo['File'])
{
fCallback(null, sUid);
return false;
}
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)
{
oXhr.upload.onprogress = function (oEvent) {
if (oEvent && oEvent.lengthComputable && defined(oEvent.loaded) && defined(oEvent.total))
{
fProgressFunction(sUid, oEvent.loaded, oEvent.total);
}
};
}
oXhr.onreadystatechange = function () {
if (4 === oXhr.readyState && 200 === oXhr.status)
{
if (fCompleteFunction)
{
var
bResult = false,
oResult = null
;
try
{
oResult = JSON.parse(oXhr.responseText);
bResult = true;
}
catch (oException)
{
oResult = null;
}
fCompleteFunction(sUid, bResult, oResult);
}
if (defined(self.oXhrs[sUid]))
{
self.oXhrs[sUid] = null;
}
fCallback(null, sUid);
}
else
{
if (4 === oXhr.readyState)
{
fCompleteFunction(sUid, false, null);
fCallback(null, sUid);
}
}
};
if (fStartFunction)
{
fStartFunction(sUid);
}
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]));
});
oXhr.send(oFormData);
this.oXhrs[sUid] = oXhr;
return true;
}
catch (oError)
{
}
fCallback(null, sUid);
return false;
}
generateNewInput(oClickElement)
{
var
self = this,
oLabel = null,
oInput = null
;
if (oClickElement)
{
oInput = Utils.getNewInput('', !Utils.getValue(this.oOptions, 'disableMultiple', false));
oLabel = Utils.createNextLabel();
oLabel.append(oInput);
$(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')
);
})
;
}
}
cancel(sUid)
{
this.oUids[sUid] = false;
if (this.oXhrs[sUid])
{
try
{
if (this.oXhrs[sUid].abort)
{
this.oXhrs[sUid].abort();
}
}
catch (oError)
{
}
this.oXhrs[sUid] = null;
}
}
}
/**
* @type {Object}
*/
AjaxDriver.prototype.oXhrs = {};
/**
* @type {Object}
*/
AjaxDriver.prototype.oUids = {};
/**
* @type {?Jua}
*/
AjaxDriver.prototype.oJua = null;
/**
* @type {Object}
*/
AjaxDriver.prototype.oOptions = {};
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)
}
}
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()
}
return c
}, c.await = function (a) {
return k = a, e || k(i, j), c
}, c
}
/**
* @constructor
* @param {Object=} oOptions
*/
class Jua
{
constructor(oOptions)
{
oOptions = defined(oOptions) ? oOptions : {};
var
self = this,
$ = jQuery
;
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);
self.oQueue = queue(parseInt(Utils.getValue(self.oOptions, 'queueSize', 10) || 0, 10));
if (self.runEvent('onCompleteAll'))
{
self.oQueue.await(function () {
self.runEvent('onCompleteAll');
});
}
self.oDriver = new AjaxDriver(self, self.oOptions);
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)
{
oEvent = Utils.getEvent(oEvent);
if (oEvent && oEvent.dataTransfer && Utils.eventContainsFiles(oEvent))
{
try
{
var sEffect = oEvent.dataTransfer.effectAllowed;
Utils.mainClearTimeout(self.iDocTimer);
oEvent.dataTransfer.dropEffect = (sEffect === 'move' || sEffect === 'linkMove') ? 'move' : 'copy';
oEvent.stopPropagation();
oEvent.preventDefault();
oBigDropZone.trigger('dragover', oEvent);
}
catch (oExc) {}
}
}
},
fHandleDrop = function (oEvent) {
if (self.bEnableDnD && oEvent)
{
oEvent = Utils.getEvent(oEvent);
if (oEvent && Utils.eventContainsFiles(oEvent))
{
oEvent.preventDefault();
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')
);
}
}
self.runEvent('onDragLeave', [oEvent]);
},
fHandleDragEnter = function (oEvent) {
if (self.bEnableDnD && oEvent)
{
oEvent = Utils.getEvent(oEvent);
if (oEvent && Utils.eventContainsFiles(oEvent))
{
Utils.mainClearTimeout(self.iDocTimer);
oEvent.preventDefault();
self.runEvent('onDragEnter', [oDragAndDropElement, oEvent]);
}
}
},
fHandleDragLeave = function (oEvent) {
if (self.bEnableDnD && oEvent)
{
oEvent = Utils.getEvent(oEvent);
if (oEvent)
{
var oRelatedTarget = document['elementFromPoint'] ? document['elementFromPoint'](oEvent['clientX'], oEvent['clientY']) : null;
if (oRelatedTarget && Utils.contains(this, oRelatedTarget))
{
return;
}
Utils.mainClearTimeout(self.iDocTimer);
self.runEvent('onDragLeave', [oDragAndDropElement, oEvent]);
}
return;
}
}
;
if (oDragAndDropElement)
{
if (!Utils.getValue(self.oOptions, 'disableDocumentDropPrevent', false))
{
$doc.on('dragover', function (oEvent) {
if (self.bEnableDnD && oEvent)
{
oEvent = Utils.getEvent(oEvent);
if (oEvent && oEvent.dataTransfer && Utils.eventContainsFiles(oEvent))
{
try
{
oEvent.dataTransfer.dropEffect = 'none';
oEvent.preventDefault();
}
catch (oExc) {}
}
}
});
}
if (oBigDropZone && oBigDropZone[0])
{
oBigDropZone
.on('dragover', function (oEvent) {
if (self.bEnableDnD && oEvent)
{
Utils.mainClearTimeout(self.iDocTimer);
}
})
.on('dragenter', function (oEvent) {
if (self.bEnableDnD && oEvent)
{
oEvent = Utils.getEvent(oEvent);
if (oEvent && Utils.eventContainsFiles(oEvent))
{
Utils.mainClearTimeout(self.iDocTimer);
oEvent.preventDefault();
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);
}
}
})
.on('drop', function (oEvent) {
if (self.bEnableDnD && oEvent)
{
oEvent = Utils.getEvent(oEvent);
if (oEvent)
{
var bFiles = Utils.eventContainsFiles(oEvent);
if (bFiles)
{
oEvent.preventDefault();
}
self.runEvent('onBodyDragLeave', [oEvent]);
return !bFiles;
}
}
return false;
})
;
}
$(oDragAndDropElement)
.on('dragenter', fHandleDragEnter)
.on('dragover', fHandleDragOver)
.on('dragleave', fHandleDragLeave)
.on('drop', fHandleDrop)
;
}
}(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;
}
/**
* @param {string} sUid
*/
cancel(sUid)
{
this.oDriver.cancel(sUid);
}
/**
* @param {boolean} bEnabled
*/
setDragAndDropEnabledStatus(bEnabled)
{
this.bEnableDnD = !!bEnabled;
}
/**
* @param {Object} oFileInfo
*/
addNewFile(oFileInfo)
{
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);
}
}
}
/**
* @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;
})();