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