| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231 |
- /*
- * 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 document, Node, window, top, getComputedStyle, setTimeout, XPathResult */
- (() => {
- const INFOBAR_TAGNAME = "singlefile-infobar";
- const LINK_ICON = "";
- const IMAGE_ICON = "";
- const SINGLEFILE_COMMENT = "SingleFile";
- const browser = this.browser;
- if (window == top) {
- if (document.readyState == "loading") {
- document.addEventListener("DOMContentLoaded", displayIcon, false);
- } else {
- displayIcon();
- }
- }
- async function displayIcon() {
- const result = document.evaluate("//comment()", document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null);
- let singleFileComment = result && result.singleNodeValue;
- if (singleFileComment && isSingleFileComment(singleFileComment)) {
- const info = singleFileComment.textContent.split("\n");
- const [, , url, saveDate, ...infoData] = info;
- if (url && saveDate) {
- let options;
- if (browser && browser.runtime && browser.runtime.sendMessage) {
- options = await browser.runtime.sendMessage({ method: "tabs.getOptions", url });
- } else {
- options = { displayInfobar: true };
- }
- if (options.displayInfobar) {
- initInfobar(url, saveDate, infoData);
- }
- }
- }
- }
- function isSingleFileComment(node) {
- return node.nodeType == Node.COMMENT_NODE && node.textContent.includes(SINGLEFILE_COMMENT);
- }
- function initInfobar(url, saveDate, infoData) {
- let infobarElement = document.querySelector(INFOBAR_TAGNAME);
- if (!infobarElement) {
- url = url.split("url: ")[1];
- saveDate = saveDate.split("saved date: ")[1];
- if (infoData && infoData.length > 1) {
- let content = infoData[0].split("info: ")[1].trim();
- for (let indexLine = 1; indexLine < infoData.length - 1; indexLine++) {
- content += "\n" + infoData[indexLine].trim();
- }
- infoData = content.trim();
- } else {
- infoData = saveDate;
- }
- infobarElement = createElement(INFOBAR_TAGNAME, document.body);
- setProperty(infobarElement, "background-color", "#f9f9f9");
- setProperty(infobarElement, "display", "flex");
- setProperty(infobarElement, "position", "fixed");
- setProperty(infobarElement, "top", "16px");
- setProperty(infobarElement, "right", "16px");
- setProperty(infobarElement, "height", "auto");
- setProperty(infobarElement, "min-height", "24px");
- setProperty(infobarElement, "min-width", "24px");
- setProperty(infobarElement, "background-position", "center");
- setProperty(infobarElement, "background-repeat", "no-repeat");
- setProperty(infobarElement, "z-index", 2147483647);
- setProperty(infobarElement, "text-align", "center");
- setProperty(infobarElement, "will-change", "opacity, padding-left, padding-right, width, background-color, color");
- setProperty(infobarElement, "margin", "0 0 0 16px");
- const closeElement = createElement("span", infobarElement);
- closeElement.textContent = "✕";
- setProperty(closeElement, "display", "none");
- setProperty(closeElement, "opacity", .7);
- setProperty(closeElement, "padding-right", "8px");
- setProperty(closeElement, "cursor", "pointer");
- setProperty(closeElement, "color", "#9aa0a6");
- setProperty(closeElement, "line-height", "24px");
- closeElement.onmouseover = () => setProperty(closeElement, "opacity", 1);
- closeElement.onmouseout = () => setProperty(closeElement, "opacity", .7);
- closeElement.onclick = event => {
- if (event.button === 0) {
- infobarElement.remove();
- }
- };
- const infoElement = createElement("span", infobarElement);
- setProperty(infoElement, "font-family", "Arial");
- setProperty(infoElement, "color", "#9aa0a6");
- setProperty(infoElement, "font-size", "14px");
- setProperty(infoElement, "line-height", "22px");
- setProperty(infoElement, "word-break", "break-word");
- setProperty(infoElement, "white-space", "pre-wrap");
- infoElement.textContent = infoData;
- const linkElement = createElement("a", infobarElement);
- setProperty(linkElement, "display", "inline-block");
- setProperty(linkElement, "padding-left", "8px");
- setProperty(linkElement, "line-height", "11px");
- setProperty(linkElement, "cursor", "pointer");
- setProperty(linkElement, "user-select", "none");
- linkElement.target = "_blank";
- linkElement.rel = "noopener noreferrer";
- linkElement.title = "Open source URL: " + url;
- linkElement.href = url;
- const imgElement = createElement("img", linkElement);
- setProperty(imgElement, "padding-top", "3px");
- setProperty(imgElement, "-webkit-padding-after", "2px");
- setProperty(imgElement, "padding-left", "2px");
- setProperty(imgElement, "-webkit-padding-start", "2px");
- setProperty(imgElement, "cursor", "pointer");
- setProperty(infobarElement, "text-align", "right");
- imgElement.src = LINK_ICON;
- hideInfobar(infobarElement, linkElement, infoElement, closeElement);
- infobarElement.onmouseover = () => setProperty(infobarElement, "opacity", 1);
- document.addEventListener("click", event => {
- if (event.button === 0) {
- let element = event.target;
- while (element && element != infobarElement) {
- element = element.parentElement;
- }
- if (element != infobarElement) {
- hideInfobar(infobarElement, linkElement, infoElement, closeElement);
- }
- }
- });
- setTimeout(() => {
- setProperty(infobarElement, "transition-property", "opacity");
- setProperty(infobarElement, "transition-duration", "250ms");
- });
- }
- }
- function displayInfobar(infobarElement, linkElement, infoElement, closeElement) {
- setProperty(infobarElement, "font-size", "13px");
- setProperty(infobarElement, "opacity", 1);
- setProperty(infobarElement, "width", "auto");
- setProperty(infobarElement, "background-color", "#f9f9f9");
- setProperty(infobarElement, "cursor", "auto");
- setProperty(infobarElement, "color", "#9aa0a6");
- setProperty(infobarElement, "padding-left", "8px");
- setProperty(infobarElement, "padding-right", "4px");
- setProperty(infobarElement, "padding-top", "2px");
- setProperty(infobarElement, "padding-bottom", "2px");
- setProperty(infobarElement, "-webkit-padding-start", "8px");
- setProperty(infobarElement, "-webkit-padding-end", "4px");
- setProperty(infobarElement, "border", "2px solid #555");
- setProperty(infobarElement, "-webkit-border-start", "2px solid #555");
- setProperty(infobarElement, "-webkit-border-before", "2px solid #555");
- setProperty(infobarElement, "-webkit-border-end", "2px solid #555");
- setProperty(infobarElement, "-webkit-border-after", "2px solid #555");
- setProperty(infobarElement, "background-image", "none");
- setProperty(infobarElement, "border-radius", "8px");
- setProperty(infoElement, "display", "inline-block");
- setProperty(linkElement, "display", "inline-block");
- setProperty(closeElement, "display", "inline-block");
- setProperty(infobarElement, "user-select", "initial");
- setProperty(infobarElement, "-moz-user-select", "initial");
- infobarElement.onclick = null;
- infobarElement.onmouseout = null;
- }
- function hideInfobar(infobarElement, linkElement, infoElement, closeElement) {
- setProperty(infobarElement, "user-select", "none");
- setProperty(infobarElement, "-moz-user-select", "none");
- setProperty(infobarElement, "opacity", .7);
- infobarElement.onmouseout = () => setProperty(infobarElement, "opacity", .7);
- setProperty(infobarElement, "width", "24px");
- setProperty(infobarElement, "background-color", "#737373");
- setProperty(infobarElement, "cursor", "pointer");
- setProperty(infobarElement, "color", "white");
- setProperty(infobarElement, "padding-left", 0);
- setProperty(infobarElement, "padding-right", 0);
- setProperty(infobarElement, "padding-top", 0);
- setProperty(infobarElement, "padding-bottom", 0);
- setProperty(infobarElement, "-webkit-padding-start", 0);
- setProperty(infobarElement, "-webkit-padding-end", 0);
- setProperty(infobarElement, "border", "2px solid #eee");
- setProperty(infobarElement, "-webkit-border-start", "2px solid #eee");
- setProperty(infobarElement, "-webkit-border-before", "2px solid #eee");
- setProperty(infobarElement, "-webkit-border-end", "2px solid #eee");
- setProperty(infobarElement, "-webkit-border-after", "2px solid #eee");
- setProperty(infobarElement, "background-image", "url(" + IMAGE_ICON + ")");
- setProperty(infobarElement, "background-size", "70% 70%");
- setProperty(infobarElement, "border-radius", "16px");
- setProperty(closeElement, "display", "none");
- setProperty(linkElement, "display", "none");
- setProperty(infoElement, "display", "none");
- infobarElement.onclick = event => {
- if (event.button === 0) {
- displayInfobar(infobarElement, linkElement, infoElement, closeElement);
- return false;
- }
- };
- }
- function createElement(tagName, parentElement) {
- const element = document.createElement(tagName);
- parentElement.appendChild(element);
- Array.from(getComputedStyle(element)).forEach(property => setProperty(element, property, "initial"));
- return element;
- }
- function setProperty(element, name, value) {
- element.style.setProperty(name, value, "important");
- }
- })();
|