mirror of
https://github.com/the-djmaze/snappymail.git
synced 2025-01-02 21:12:02 +08:00
369 lines
7.5 KiB
JavaScript
369 lines
7.5 KiB
JavaScript
/* RainLoop Webmail (c) RainLoop Team | MIT */
|
|
(doc => {
|
|
const
|
|
defined = v => undefined !== v,
|
|
/**
|
|
* @param {*} aItems
|
|
* @param {Function} fFileCallback
|
|
* @param {number=} iLimit = 20
|
|
*/
|
|
getDataFromFiles = (aItems, fFileCallback, iLimit) =>
|
|
{
|
|
if (aItems?.length)
|
|
{
|
|
let
|
|
oFile,
|
|
iCount = 0,
|
|
bCallLimit = false
|
|
;
|
|
|
|
[...aItems].forEach(oItem => {
|
|
if (oItem) {
|
|
if (iLimit && iLimit < ++iCount) {
|
|
if (!bCallLimit) {
|
|
bCallLimit = true;
|
|
// fLimitCallback(iLimit);
|
|
}
|
|
} else {
|
|
oFile = getDataFromFile(oItem);
|
|
oFile && fFileCallback(oFile);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
},
|
|
|
|
addEventListeners = (element, obj) =>
|
|
Object.entries(obj).forEach(([key, value]) => element.addEventListener(key, value)),
|
|
|
|
/**
|
|
* @param {*} oFile
|
|
* @return {Object}
|
|
*/
|
|
getDataFromFile = oFile =>
|
|
{
|
|
return oFile.size
|
|
? {
|
|
fileName: (oFile.name || '').replace(/^.*\/([^/]*)$/, '$1'),
|
|
size: oFile.size,
|
|
file: oFile
|
|
}
|
|
: null; // Folder
|
|
},
|
|
|
|
eventContainsFiles = oEvent => oEvent.dataTransfer.types.includes('Files');
|
|
|
|
class Queue extends Array
|
|
{
|
|
push(fn, ...args) {
|
|
super.push([fn, args]);
|
|
this.call();
|
|
}
|
|
call() {
|
|
if (!this.running) {
|
|
this.running = true;
|
|
let f;
|
|
while ((f = this.shift())) f[0](...f[1]);
|
|
this.running = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @constructor
|
|
* @param {Object=} options
|
|
*/
|
|
class Jua
|
|
{
|
|
constructor(options)
|
|
{
|
|
let timer,
|
|
el = options.clickElement;
|
|
|
|
const self = this,
|
|
timerStart = fn => {
|
|
timerStop();
|
|
timer = setTimeout(fn, 200);
|
|
},
|
|
timerStop = () => {
|
|
timer && clearTimeout(timer);
|
|
timer = 0;
|
|
};
|
|
|
|
self.oEvents = {
|
|
onSelect: null,
|
|
onStart: null,
|
|
onComplete: null,
|
|
onProgress: null,
|
|
onDragEnter: null,
|
|
onDragLeave: null,
|
|
onBodyDragEnter: null,
|
|
onBodyDragLeave: null
|
|
};
|
|
|
|
self.oXhrs = {};
|
|
self.oUids = {};
|
|
self.options = Object.assign({
|
|
action: '',
|
|
name: 'uploader',
|
|
limit: 0,
|
|
// clickElement:
|
|
// dragAndDropElement:
|
|
}, options || {});
|
|
self.oQueue = new Queue();
|
|
|
|
// clickElement
|
|
if (el) {
|
|
el.style.position = 'relative';
|
|
el.style.overflow = 'hidden';
|
|
if ('inline' === el.style.display) {
|
|
el.style.display = 'inline-block';
|
|
}
|
|
|
|
self.generateNewInput(el);
|
|
}
|
|
|
|
el = options.dragAndDropElement;
|
|
if (el) {
|
|
addEventListeners(doc, {
|
|
dragover: oEvent => {
|
|
if (eventContainsFiles(oEvent)) {
|
|
timerStop();
|
|
if (el.contains(oEvent.target)) {
|
|
oEvent.dataTransfer.dropEffect = 'copy';
|
|
oEvent.stopPropagation();
|
|
} else {
|
|
oEvent.dataTransfer.dropEffect = 'none';
|
|
}
|
|
oEvent.preventDefault();
|
|
}
|
|
},
|
|
dragenter: oEvent => {
|
|
if (eventContainsFiles(oEvent)) {
|
|
timerStop();
|
|
oEvent.preventDefault();
|
|
self.runEvent('onBodyDragEnter', oEvent);
|
|
if (el.contains(oEvent.target)) {
|
|
timerStop();
|
|
self.runEvent('onDragEnter', el, oEvent);
|
|
}
|
|
}
|
|
},
|
|
dragleave: oEvent => {
|
|
if (eventContainsFiles(oEvent)) {
|
|
let oRelatedTarget = doc.elementFromPoint(oEvent.clientX, oEvent.clientY);
|
|
if (!oRelatedTarget || !el.contains(oRelatedTarget)) {
|
|
self.runEvent('onDragLeave', el, oEvent);
|
|
}
|
|
timerStart(() => self.runEvent('onBodyDragLeave', oEvent))
|
|
}
|
|
},
|
|
drop: oEvent => {
|
|
if (eventContainsFiles(oEvent)) {
|
|
timerStop();
|
|
oEvent.preventDefault();
|
|
if (el.contains(oEvent.target)) {
|
|
getDataFromFiles(
|
|
oEvent.files || oEvent.dataTransfer.files,
|
|
oFile => {
|
|
if (oFile) {
|
|
self.addFile(oFile);
|
|
}
|
|
},
|
|
self.options.limit
|
|
);
|
|
}
|
|
}
|
|
self.runEvent('onDragLeave', oEvent);
|
|
self.runEvent('onBodyDragLeave', oEvent);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} sName
|
|
* @param {Function} fFunc
|
|
*/
|
|
on(sName, fFunc)
|
|
{
|
|
this.oEvents[sName] = fFunc;
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* @param {string} sName
|
|
*/
|
|
runEvent(sName, ...aArgs)
|
|
{
|
|
this.oEvents[sName]?.apply(null, aArgs);
|
|
}
|
|
|
|
/**
|
|
* @param {string} sName
|
|
*/
|
|
getEvent(sName)
|
|
{
|
|
return this.oEvents[sName] || null;
|
|
}
|
|
|
|
/**
|
|
* @param {Object} oFileInfo
|
|
*/
|
|
addFile(oFileInfo)
|
|
{
|
|
const sUid = 'jua-uid-' + Jua.randomId(16) + '-' + (Date.now().toString()),
|
|
fOnSelect = this.getEvent('onSelect');
|
|
if (oFileInfo && (!fOnSelect || (false !== fOnSelect(sUid, oFileInfo))))
|
|
{
|
|
this.oUids[sUid] = true;
|
|
this.oQueue.push((...args) => this.uploadTask(...args), sUid, oFileInfo);
|
|
}
|
|
else
|
|
{
|
|
this.cancel(sUid);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} sUid
|
|
* @param {?} oFileInfo
|
|
*/
|
|
uploadTask(sUid, oFileInfo)
|
|
{
|
|
if (false === this.oUids[sUid] || !oFileInfo || !oFileInfo.file)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
try
|
|
{
|
|
const
|
|
self = this,
|
|
oXhr = new XMLHttpRequest(),
|
|
oFormData = new FormData(),
|
|
sAction = this.options.action,
|
|
fStartFunction = this.getEvent('onStart'),
|
|
fProgressFunction = this.getEvent('onProgress')
|
|
;
|
|
|
|
oXhr.open('POST', sAction, true);
|
|
|
|
if (fProgressFunction && oXhr.upload)
|
|
{
|
|
oXhr.upload.onprogress = oEvent => {
|
|
if (oEvent && oEvent.lengthComputable && defined(oEvent.loaded) && defined(oEvent.total))
|
|
{
|
|
fProgressFunction(sUid, oEvent.loaded, oEvent.total);
|
|
}
|
|
};
|
|
}
|
|
|
|
oXhr.onreadystatechange = () => {
|
|
if (4 === oXhr.readyState)
|
|
{
|
|
delete self.oXhrs[sUid];
|
|
let bResult = false,
|
|
oResult = null;
|
|
if (200 === oXhr.status)
|
|
{
|
|
try
|
|
{
|
|
oResult = JSON.parse(oXhr.responseText);
|
|
bResult = true;
|
|
}
|
|
catch (e)
|
|
{
|
|
console.error(e);
|
|
}
|
|
}
|
|
this.getEvent('onComplete')(sUid, bResult, oResult);
|
|
}
|
|
};
|
|
|
|
fStartFunction && fStartFunction(sUid);
|
|
|
|
oFormData.append(this.options.name, oFileInfo.file);
|
|
|
|
oXhr.send(oFormData);
|
|
|
|
this.oXhrs[sUid] = oXhr;
|
|
return true;
|
|
}
|
|
catch (oError)
|
|
{
|
|
console.error(oError)
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
generateNewInput(oClickElement)
|
|
{
|
|
if (oClickElement)
|
|
{
|
|
const self = this,
|
|
limit = self.options.limit,
|
|
oInput = doc.createElement('input'),
|
|
onClick = ()=>oInput.click();
|
|
|
|
oInput.type = 'file';
|
|
oInput.tabIndex = -1;
|
|
oInput.style.display = 'none';
|
|
oInput.multiple = 1 != limit;
|
|
|
|
oClickElement.addEventListener('click', onClick);
|
|
|
|
oInput.addEventListener('input', () => {
|
|
const fFileCallback = oFile => {
|
|
self.addFile(oFile);
|
|
setTimeout(() => {
|
|
oInput.remove();
|
|
oClickElement.removeEventListener('click', onClick);
|
|
self.generateNewInput(oClickElement);
|
|
}, 10);
|
|
};
|
|
if (oInput.files?.length) {
|
|
getDataFromFiles(oInput.files, fFileCallback, limit);
|
|
} else {
|
|
fFileCallback({
|
|
fileName: oInput.value.split(/\\\//).pop(),
|
|
size: null,
|
|
file : null
|
|
});
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string} sUid
|
|
*/
|
|
cancel(sUid)
|
|
{
|
|
this.oUids[sUid] = false;
|
|
if (this.oXhrs[sUid])
|
|
{
|
|
try
|
|
{
|
|
this.oXhrs[sUid].abort && this.oXhrs[sUid].abort();
|
|
}
|
|
catch (oError)
|
|
{
|
|
console.error(oError);
|
|
}
|
|
|
|
delete this.oXhrs[sUid];
|
|
}
|
|
}
|
|
}
|
|
|
|
Jua.randomId = len => {
|
|
let arr = new Uint8Array((len || 32) / 2);
|
|
crypto.getRandomValues(arr);
|
|
return arr.map(dec => dec.toString(16).padStart(2,'0')).join('');
|
|
}
|
|
|
|
this.Jua = Jua;
|
|
|
|
})(document);
|