2020-06-06 00:03:37 +02:00
|
|
|
"use strict";
|
2016-05-20 21:35:12 +02:00
|
|
|
|
2020-06-06 00:03:37 +02:00
|
|
|
const events = require("../events.js");
|
|
|
|
const views = require("../util/views.js");
|
2016-05-20 21:35:12 +02:00
|
|
|
|
2020-06-06 00:03:37 +02:00
|
|
|
const template = views.getTemplate("file-dropper");
|
2016-08-20 22:30:08 +02:00
|
|
|
|
2017-02-21 18:29:32 +01:00
|
|
|
const KEY_RETURN = 13;
|
|
|
|
|
2016-08-20 22:30:08 +02:00
|
|
|
class FileDropperControl extends events.EventTarget {
|
2016-05-20 21:35:12 +02:00
|
|
|
constructor(target, options) {
|
2016-08-20 22:30:08 +02:00
|
|
|
super();
|
|
|
|
|
2016-05-20 21:35:12 +02:00
|
|
|
this._options = options;
|
2016-08-20 22:30:08 +02:00
|
|
|
const source = template({
|
2017-02-12 10:40:50 +01:00
|
|
|
extraText: options.extraText,
|
|
|
|
allowMultiple: options.allowMultiple,
|
|
|
|
allowUrls: options.allowUrls,
|
2017-02-21 18:29:32 +01:00
|
|
|
lock: options.lock,
|
2020-06-06 00:03:37 +02:00
|
|
|
id: "file-" + Math.random().toString(36).substring(7),
|
2017-02-21 19:09:18 +01:00
|
|
|
urlPlaceholder:
|
2020-06-06 00:03:37 +02:00
|
|
|
options.urlPlaceholder || "Alternatively, paste an URL here.",
|
2016-05-20 21:35:12 +02:00
|
|
|
});
|
|
|
|
|
2020-06-06 00:03:37 +02:00
|
|
|
this._dropperNode = source.querySelector(".file-dropper");
|
|
|
|
this._urlInputNode = source.querySelector("input[type=text]");
|
|
|
|
this._urlConfirmButtonNode = source.querySelector("button");
|
|
|
|
this._fileInputNode = source.querySelector("input[type=file]");
|
|
|
|
this._fileInputNode.style.display = "none";
|
2017-02-12 10:40:50 +01:00
|
|
|
this._fileInputNode.multiple = options.allowMultiple || false;
|
2016-05-20 21:35:12 +02:00
|
|
|
|
|
|
|
this._counter = 0;
|
2020-06-06 00:03:37 +02:00
|
|
|
this._dropperNode.addEventListener("dragenter", (e) =>
|
|
|
|
this._evtDragEnter(e)
|
|
|
|
);
|
|
|
|
this._dropperNode.addEventListener("dragleave", (e) =>
|
|
|
|
this._evtDragLeave(e)
|
|
|
|
);
|
|
|
|
this._dropperNode.addEventListener("dragover", (e) =>
|
|
|
|
this._evtDragOver(e)
|
|
|
|
);
|
|
|
|
this._dropperNode.addEventListener("drop", (e) => this._evtDrop(e));
|
|
|
|
this._fileInputNode.addEventListener("change", (e) =>
|
|
|
|
this._evtFileChange(e)
|
|
|
|
);
|
2016-05-20 21:35:12 +02:00
|
|
|
|
2016-08-20 22:31:22 +02:00
|
|
|
if (this._urlInputNode) {
|
2020-06-06 00:03:37 +02:00
|
|
|
this._urlInputNode.addEventListener("keydown", (e) =>
|
|
|
|
this._evtUrlInputKeyDown(e)
|
|
|
|
);
|
2017-02-21 18:29:32 +01:00
|
|
|
}
|
|
|
|
if (this._urlConfirmButtonNode) {
|
2020-06-06 00:03:37 +02:00
|
|
|
this._urlConfirmButtonNode.addEventListener("click", (e) =>
|
|
|
|
this._evtUrlConfirmButtonClick(e)
|
|
|
|
);
|
2016-08-20 22:31:22 +02:00
|
|
|
}
|
|
|
|
|
2016-07-27 22:27:33 +02:00
|
|
|
this._originalHtml = this._dropperNode.innerHTML;
|
2016-06-14 10:31:48 +02:00
|
|
|
views.replaceContent(target, source);
|
2016-05-20 21:35:12 +02:00
|
|
|
}
|
|
|
|
|
2016-07-27 22:27:33 +02:00
|
|
|
reset() {
|
|
|
|
this._dropperNode.innerHTML = this._originalHtml;
|
2020-06-06 00:03:37 +02:00
|
|
|
this.dispatchEvent(new CustomEvent("reset"));
|
2016-07-27 22:27:33 +02:00
|
|
|
}
|
|
|
|
|
2016-08-20 22:30:08 +02:00
|
|
|
_emitFiles(files) {
|
2016-05-20 21:35:12 +02:00
|
|
|
files = Array.from(files);
|
|
|
|
if (this._options.lock) {
|
2020-06-06 00:03:37 +02:00
|
|
|
this._dropperNode.innerText = files
|
|
|
|
.map((file) => file.name)
|
|
|
|
.join(", ");
|
2016-05-20 21:35:12 +02:00
|
|
|
}
|
2016-08-20 22:30:08 +02:00
|
|
|
this.dispatchEvent(
|
2020-06-06 00:03:37 +02:00
|
|
|
new CustomEvent("fileadd", { detail: { files: files } })
|
|
|
|
);
|
2016-06-12 22:10:20 +02:00
|
|
|
}
|
2016-05-20 21:35:12 +02:00
|
|
|
|
2016-08-20 22:31:22 +02:00
|
|
|
_emitUrls(urls) {
|
2020-06-06 00:03:37 +02:00
|
|
|
urls = Array.from(urls).map((url) => url.trim());
|
2017-02-21 18:29:32 +01:00
|
|
|
if (this._options.lock) {
|
2020-06-06 00:03:37 +02:00
|
|
|
this._dropperNode.innerText = urls
|
|
|
|
.map((url) => url.split(/\//).reverse()[0])
|
|
|
|
.join(", ");
|
2017-02-21 18:29:32 +01:00
|
|
|
}
|
2016-08-20 22:31:22 +02:00
|
|
|
for (let url of urls) {
|
|
|
|
if (!url) {
|
|
|
|
return;
|
|
|
|
}
|
2016-08-27 23:45:07 +02:00
|
|
|
if (!url.match(/^https?:\/\/[^.]+\..+$/)) {
|
2016-08-20 22:31:22 +02:00
|
|
|
window.alert(`"${url}" does not look like a valid URL.`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2020-06-06 00:03:37 +02:00
|
|
|
this.dispatchEvent(
|
|
|
|
new CustomEvent("urladd", { detail: { urls: urls } })
|
|
|
|
);
|
2016-05-20 21:35:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
_evtDragEnter(e) {
|
2020-06-06 00:03:37 +02:00
|
|
|
this._dropperNode.classList.add("active");
|
2020-04-20 18:31:15 +02:00
|
|
|
this._counter++;
|
2016-05-20 21:35:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
_evtDragLeave(e) {
|
|
|
|
this._counter--;
|
|
|
|
if (this._counter === 0) {
|
2020-06-06 00:03:37 +02:00
|
|
|
this._dropperNode.classList.remove("active");
|
2016-05-20 21:35:12 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
_evtDragOver(e) {
|
|
|
|
e.preventDefault();
|
|
|
|
}
|
|
|
|
|
2016-08-20 22:30:08 +02:00
|
|
|
_evtFileChange(e) {
|
|
|
|
this._emitFiles(e.target.files);
|
|
|
|
}
|
|
|
|
|
2016-05-20 21:35:12 +02:00
|
|
|
_evtDrop(e) {
|
|
|
|
e.preventDefault();
|
2020-06-06 00:03:37 +02:00
|
|
|
this._dropperNode.classList.remove("active");
|
2016-05-20 21:35:12 +02:00
|
|
|
if (!e.dataTransfer.files.length) {
|
2020-06-06 00:03:37 +02:00
|
|
|
window.alert("Only files are supported.");
|
2016-05-20 21:35:12 +02:00
|
|
|
}
|
|
|
|
if (!this._options.allowMultiple && e.dataTransfer.files.length > 1) {
|
2020-06-06 00:03:37 +02:00
|
|
|
window.alert("Cannot select multiple files.");
|
2016-05-20 21:35:12 +02:00
|
|
|
}
|
2016-08-20 22:30:08 +02:00
|
|
|
this._emitFiles(e.dataTransfer.files);
|
2016-05-20 21:35:12 +02:00
|
|
|
}
|
2016-08-20 22:31:22 +02:00
|
|
|
|
2017-02-21 18:29:32 +01:00
|
|
|
_evtUrlInputKeyDown(e) {
|
|
|
|
if (e.which !== KEY_RETURN) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
e.preventDefault();
|
2020-06-06 00:03:37 +02:00
|
|
|
this._dropperNode.classList.remove("active");
|
2017-02-21 18:29:32 +01:00
|
|
|
this._emitUrls(this._urlInputNode.value.split(/[\r\n]/));
|
2020-06-06 00:03:37 +02:00
|
|
|
this._urlInputNode.value = "";
|
2017-02-21 18:29:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
_evtUrlConfirmButtonClick(e) {
|
2016-08-20 22:31:22 +02:00
|
|
|
e.preventDefault();
|
2020-06-06 00:03:37 +02:00
|
|
|
this._dropperNode.classList.remove("active");
|
2016-08-20 22:31:22 +02:00
|
|
|
this._emitUrls(this._urlInputNode.value.split(/[\r\n]/));
|
2020-06-06 00:03:37 +02:00
|
|
|
this._urlInputNode.value = "";
|
2016-08-20 22:31:22 +02:00
|
|
|
}
|
2016-05-20 21:35:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = FileDropperControl;
|