|
@@ -0,0 +1,238 @@
|
|
|
|
|
+/*
|
|
|
|
|
+ * Copyright 2010-2020 Gildas Lormeau
|
|
|
|
|
+ * contact : gildas.lormeau <at> gmail.com
|
|
|
|
|
+ *
|
|
|
|
|
+ * This file is part of SingleFile.
|
|
|
|
|
+ *
|
|
|
|
|
+ * The code in this file is free software: you can redistribute it and/or
|
|
|
|
|
+ * modify it under the terms of the GNU Affero General Public License
|
|
|
|
|
+ * (GNU AGPL) as published by the Free Software Foundation, either version 3
|
|
|
|
|
+ * of the License, or (at your option) any later version.
|
|
|
|
|
+ *
|
|
|
|
|
+ * The code in this file is distributed in the hope that it will be useful,
|
|
|
|
|
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
|
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
|
|
|
|
|
+ * General Public License for more details.
|
|
|
|
|
+ *
|
|
|
|
|
+ * As additional permission under GNU AGPL version 3 section 7, you may
|
|
|
|
|
+ * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
|
|
|
|
|
+ * AGPL normally required by section 4, provided you include this license
|
|
|
|
|
+ * notice and a URL through which recipients can access the Corresponding
|
|
|
|
|
+ * Source.
|
|
|
|
|
+ */
|
|
|
|
|
+
|
|
|
|
|
+/* global browser, document, globalThis, getComputedStyle, FileReader, Image, OffscreenCanvas */
|
|
|
|
|
+
|
|
|
|
|
+const singlefile = globalThis.singlefile;
|
|
|
|
|
+
|
|
|
|
|
+const CLOSE_ICON = "";
|
|
|
|
|
+
|
|
|
|
|
+const SINGLE_FILE_UI_ELEMENT_CLASS = singlefile.helper.SINGLE_FILE_UI_ELEMENT_CLASS;
|
|
|
|
|
+const ERROR_BAR_TAGNAME = "singlefile-error-bar";
|
|
|
|
|
+const OPEN_FILE_BAR_TAGNAME = "singlefile-open-file-bar";
|
|
|
|
|
+const EMBEDDED_IMAGE_BUTTON_MESSAGE = browser.i18n.getMessage("topPanelEmbeddedImageButton");
|
|
|
|
|
+const ERROR_TITLE_MESSAGE = browser.i18n.getMessage("topPanelError");
|
|
|
|
|
+
|
|
|
|
|
+const CSS_PROPERTIES = new Set(Array.from(getComputedStyle(document.documentElement)));
|
|
|
|
|
+
|
|
|
|
|
+export {
|
|
|
|
|
+ openFile,
|
|
|
|
|
+ getOpenFileBar,
|
|
|
|
|
+ onError
|
|
|
|
|
+};
|
|
|
|
|
+
|
|
|
|
|
+function onError(message, link) {
|
|
|
|
|
+ console.error("SingleFile", message, link); // eslint-disable-line no-console
|
|
|
|
|
+ displayBar(ERROR_BAR_TAGNAME, ERROR_TITLE_MESSAGE + message, { link });
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+function getOpenFileBar() {
|
|
|
|
|
+ let resolvePromise;
|
|
|
|
|
+ return {
|
|
|
|
|
+ display: async function () {
|
|
|
|
|
+ return new Promise(resolve => {
|
|
|
|
|
+ resolvePromise = resolve;
|
|
|
|
|
+ displayBar(OPEN_FILE_BAR_TAGNAME, "", { buttonLabel: EMBEDDED_IMAGE_BUTTON_MESSAGE, buttonOnclick: resolve });
|
|
|
|
|
+ });
|
|
|
|
|
+ },
|
|
|
|
|
+ hide: function () {
|
|
|
|
|
+ const barElement = document.querySelector(OPEN_FILE_BAR_TAGNAME);
|
|
|
|
|
+ if (barElement) {
|
|
|
|
|
+ barElement.remove();
|
|
|
|
|
+ }
|
|
|
|
|
+ },
|
|
|
|
|
+ cancel: function () {
|
|
|
|
|
+ this.hide();
|
|
|
|
|
+ if (resolvePromise) {
|
|
|
|
|
+ resolvePromise(true);
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+ };
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+function openFile({ accept } = { accept: "image/*" }) {
|
|
|
|
|
+ const inputElement = document.createElement("input");
|
|
|
|
|
+ inputElement.type = "file";
|
|
|
|
|
+ inputElement.accept = accept;
|
|
|
|
|
+ inputElement.click();
|
|
|
|
|
+ return new Promise(resolve => {
|
|
|
|
|
+ inputElement.addEventListener("change", event => {
|
|
|
|
|
+ if (event.target.files.length) {
|
|
|
|
|
+ const file = event.target.files[0];
|
|
|
|
|
+ const fileReader = new FileReader();
|
|
|
|
|
+ fileReader.addEventListener("load", async () => {
|
|
|
|
|
+ let mimeType = file.type;
|
|
|
|
|
+ if (mimeType == "image/png") {
|
|
|
|
|
+ resolve(new Uint8Array(fileReader.result));
|
|
|
|
|
+ } else {
|
|
|
|
|
+ const dataURI = await new Promise(resolve => {
|
|
|
|
|
+ const fileReader = new FileReader();
|
|
|
|
|
+ fileReader.addEventListener("load", () => resolve(fileReader.result));
|
|
|
|
|
+ fileReader.addEventListener("error", () => resolve());
|
|
|
|
|
+ fileReader.readAsDataURL(file);
|
|
|
|
|
+ });
|
|
|
|
|
+ if (dataURI) {
|
|
|
|
|
+ const image = new Image();
|
|
|
|
|
+ image.src = dataURI;
|
|
|
|
|
+ image.addEventListener("error", () => resolve());
|
|
|
|
|
+ await new Promise(resolve => image.addEventListener("load", resolve));
|
|
|
|
|
+ const canvas = new OffscreenCanvas(image.width, image.height);
|
|
|
|
|
+ const context = canvas.getContext("2d");
|
|
|
|
|
+ context.drawImage(image, 0, 0);
|
|
|
|
|
+ const blob = await canvas.convertToBlob({ type: "image/png" });
|
|
|
|
|
+ const fileReader = new FileReader();
|
|
|
|
|
+ fileReader.addEventListener("load", () => resolve(new Uint8Array(fileReader.result)));
|
|
|
|
|
+ fileReader.addEventListener("error", () => resolve());
|
|
|
|
|
+ fileReader.readAsArrayBuffer(blob);
|
|
|
|
|
+ } else {
|
|
|
|
|
+ resolve();
|
|
|
|
|
+ }
|
|
|
|
|
+ }
|
|
|
|
|
+ resolve(Array.from(new Uint8Array(fileReader.result)));
|
|
|
|
|
+ });
|
|
|
|
|
+ fileReader.addEventListener("error", () => resolve());
|
|
|
|
|
+ fileReader.readAsArrayBuffer(file);
|
|
|
|
|
+ } else {
|
|
|
|
|
+ resolve();
|
|
|
|
|
+ }
|
|
|
|
|
+ });
|
|
|
|
|
+ inputElement.addEventListener("cancel", () => resolve());
|
|
|
|
|
+ });
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+function displayBar(tagName, message, { link, buttonLabel, buttonOnclick } = {}) {
|
|
|
|
|
+ try {
|
|
|
|
|
+ const barElement = document.querySelector(tagName);
|
|
|
|
|
+ if (!barElement) {
|
|
|
|
|
+ const barElement = createElement(tagName);
|
|
|
|
|
+ const shadowRoot = barElement.attachShadow({ mode: "open" });
|
|
|
|
|
+ const styleElement = document.createElement("style");
|
|
|
|
|
+ styleElement.textContent = `
|
|
|
|
|
+ .container {
|
|
|
|
|
+ background-color: #ff6c00;
|
|
|
|
|
+ color: white;
|
|
|
|
|
+ display: flex;
|
|
|
|
|
+ position: fixed;
|
|
|
|
|
+ top: 0px;
|
|
|
|
|
+ left: 0px;
|
|
|
|
|
+ right: 0px;
|
|
|
|
|
+ height: auto;
|
|
|
|
|
+ width: auto;
|
|
|
|
|
+ min-height: 24px;
|
|
|
|
|
+ min-width: 24px;
|
|
|
|
|
+ z-index: 2147483647;
|
|
|
|
|
+ margin: 0;
|
|
|
|
|
+ padding: 2px;
|
|
|
|
|
+ font-family: Arial;
|
|
|
|
|
+ }
|
|
|
|
|
+ .singlefile-open-file-bar.container {
|
|
|
|
|
+ background-color: whitesmoke;
|
|
|
|
|
+ }
|
|
|
|
|
+ .text {
|
|
|
|
|
+ flex: 1;
|
|
|
|
|
+ padding-top: 4px;
|
|
|
|
|
+ padding-bottom: 4px;
|
|
|
|
|
+ padding-left: 8px;
|
|
|
|
|
+ }
|
|
|
|
|
+ button {
|
|
|
|
|
+ background-color: grey;
|
|
|
|
|
+ color: white;
|
|
|
|
|
+ border: 1px solid darkgrey;
|
|
|
|
|
+ padding: 3px;
|
|
|
|
|
+ padding-left: 8px;
|
|
|
|
|
+ padding-right: 8px;
|
|
|
|
|
+ border-radius: 4px;
|
|
|
|
|
+ cursor: pointer;
|
|
|
|
|
+ }
|
|
|
|
|
+ .close-button {
|
|
|
|
|
+ opacity: .7;
|
|
|
|
|
+ padding-left: 8px;
|
|
|
|
|
+ padding-right: 8px;
|
|
|
|
|
+ cursor: pointer;
|
|
|
|
|
+ transition: opacity 250ms;
|
|
|
|
|
+ height: 16px;
|
|
|
|
|
+ font-size: .8rem;
|
|
|
|
|
+ align-self: center;
|
|
|
|
|
+ }
|
|
|
|
|
+ .singlefile-open-file-bar .close-button {
|
|
|
|
|
+ filter: invert(1);
|
|
|
|
|
+ }
|
|
|
|
|
+ a {
|
|
|
|
|
+ color: #303036;
|
|
|
|
|
+ }
|
|
|
|
|
+ .close-button:hover {
|
|
|
|
|
+ opacity: 1;
|
|
|
|
|
+ }
|
|
|
|
|
+ `;
|
|
|
|
|
+ shadowRoot.appendChild(styleElement);
|
|
|
|
|
+ const containerElement = document.createElement("div");
|
|
|
|
|
+ containerElement.classList.add(tagName);
|
|
|
|
|
+ containerElement.classList.add("container");
|
|
|
|
|
+ const textElement = document.createElement("span");
|
|
|
|
|
+ textElement.classList.add("text");
|
|
|
|
|
+ const content = message.split("__DOC_LINK__");
|
|
|
|
|
+ textElement.textContent = content[0];
|
|
|
|
|
+ if (link && content.length == 2) {
|
|
|
|
|
+ const linkElement = document.createElement("a");
|
|
|
|
|
+ linkElement.textContent = link;
|
|
|
|
|
+ linkElement.href = link;
|
|
|
|
|
+ linkElement.target = "_blank";
|
|
|
|
|
+ textElement.appendChild(linkElement);
|
|
|
|
|
+ textElement.appendChild(document.createTextNode(content[1]));
|
|
|
|
|
+ }
|
|
|
|
|
+ if (buttonLabel && buttonOnclick) {
|
|
|
|
|
+ const buttonElement = document.createElement("button");
|
|
|
|
|
+ buttonElement.textContent = buttonLabel;
|
|
|
|
|
+ buttonElement.onclick = () => buttonOnclick();
|
|
|
|
|
+ textElement.appendChild(buttonElement);
|
|
|
|
|
+ }
|
|
|
|
|
+ containerElement.appendChild(textElement);
|
|
|
|
|
+ const closeElement = document.createElement("img");
|
|
|
|
|
+ closeElement.classList.add("close-button");
|
|
|
|
|
+ containerElement.appendChild(closeElement);
|
|
|
|
|
+ shadowRoot.appendChild(containerElement);
|
|
|
|
|
+ closeElement.src = CLOSE_ICON;
|
|
|
|
|
+ closeElement.onclick = event => {
|
|
|
|
|
+ if (event.button === 0) {
|
|
|
|
|
+ if (buttonOnclick) {
|
|
|
|
|
+ buttonOnclick(true);
|
|
|
|
|
+ }
|
|
|
|
|
+ barElement.remove();
|
|
|
|
|
+ }
|
|
|
|
|
+ };
|
|
|
|
|
+ document.body.appendChild(barElement);
|
|
|
|
|
+ }
|
|
|
|
|
+ } catch (error) {
|
|
|
|
|
+ // iignored
|
|
|
|
|
+ }
|
|
|
|
|
+}
|
|
|
|
|
+
|
|
|
|
|
+function createElement(tagName, parentElement) {
|
|
|
|
|
+ const element = document.createElement(tagName);
|
|
|
|
|
+ element.className = SINGLE_FILE_UI_ELEMENT_CLASS;
|
|
|
|
|
+ if (parentElement) {
|
|
|
|
|
+ parentElement.appendChild(element);
|
|
|
|
|
+ }
|
|
|
|
|
+ CSS_PROPERTIES.forEach(property => element.style.setProperty(property, "initial", "important"));
|
|
|
|
|
+ return element;
|
|
|
|
|
+}
|