(function () { 'use strict'; /* * Copyright 2010-2020 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 document, getComputedStyle, FileReader, Image, OffscreenCanvas, createImageBitmap */ const singlefile$1 = globalThis.singlefile; const CLOSE_ICON = ""; const SINGLE_FILE_UI_ELEMENT_CLASS$1 = singlefile$1.helper.SINGLE_FILE_UI_ELEMENT_CLASS; const SHARE_PAGE_BAR_TAGNAME = "singlefile-share-page-bar"; let EMBEDDED_IMAGE_BUTTON_MESSAGE$1, SHARE_PAGE_BUTTON_MESSAGE$1, SHARE_SELECTION_BUTTON_MESSAGE$1, ERROR_TITLE_MESSAGE$1; const CSS_PROPERTIES$1 = new Set(Array.from(getComputedStyle(document.documentElement))); function setLabels(labels) { ({ EMBEDDED_IMAGE_BUTTON_MESSAGE: EMBEDDED_IMAGE_BUTTON_MESSAGE$1, SHARE_PAGE_BUTTON_MESSAGE: SHARE_PAGE_BUTTON_MESSAGE$1, SHARE_SELECTION_BUTTON_MESSAGE: SHARE_SELECTION_BUTTON_MESSAGE$1, ERROR_TITLE_MESSAGE: ERROR_TITLE_MESSAGE$1 } = labels); } function getSharePageBar() { let resolvePromise; return { display: async function (selectedContent) { return new Promise(resolve => { resolvePromise = resolve; displayBar(SHARE_PAGE_BAR_TAGNAME, "", { buttonLabel: selectedContent ? SHARE_SELECTION_BUTTON_MESSAGE$1 : SHARE_PAGE_BUTTON_MESSAGE$1, buttonOnclick: resolve }); }); }, hide: function () { const barElement = document.querySelector(SHARE_PAGE_BAR_TAGNAME); if (barElement) { barElement.remove(); } }, cancel: function () { this.hide(); if (resolvePromise) { resolvePromise(true); } } }; } function displayBar(tagName, message, { link, buttonLabel, buttonOnclick } = {}) { try { const barElement = document.querySelector(tagName); if (!barElement) { const barElement = createElement$1(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, .singlefile-share-page-bar.container { background-color: gainsboro; border-block-end: gray 1px solid; } .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 button, .singlefile-share-page-bar button{ background-color: dimgrey; } .singlefile-open-file-bar .close-button, .singlefile-share-page-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.documentElement.appendChild(barElement); } // eslint-disable-next-line no-unused-vars } catch (error) { // iignored } } function createElement$1(tagName, parentElement) { const element = document.createElement(tagName); element.className = SINGLE_FILE_UI_ELEMENT_CLASS$1; CSS_PROPERTIES$1.forEach(property => element.style.setProperty(property, "initial", "important")); return element; } /* global TextEncoder, TextDecoder */ const TYPE_REFERENCE = 0; const SPECIAL_TYPES = [TYPE_REFERENCE]; const EMPTY_SLOT_VALUE = Symbol(); const textEncoder = new TextEncoder(); const textDecoder = new TextDecoder(); const types = new Array(256); let typeIndex = 0; registerType(serializeCircularReference, parseCircularReference, testCircularReference, TYPE_REFERENCE); registerType(null, parseObject, testObject); registerType(serializeArray, parseArray, testArray); registerType(serializeString, parseString, testString); registerType(serializeTypedArray, parseFloat64Array, testFloat64Array); registerType(serializeTypedArray, parseFloat32Array, testFloat32Array); registerType(serializeTypedArray, parseUint32Array, testUint32Array); registerType(serializeTypedArray, parseInt32Array, testInt32Array); registerType(serializeTypedArray, parseUint16Array, testUint16Array); registerType(serializeTypedArray, parseInt16Array, testInt16Array); registerType(serializeTypedArray, parseUint8ClampedArray, testUint8ClampedArray); registerType(serializeTypedArray, parseUint8Array, testUint8Array); registerType(serializeTypedArray, parseInt8Array, testInt8Array); registerType(serializeArrayBuffer, parseArrayBuffer, testArrayBuffer); registerType(serializeNumber, parseNumber, testNumber); registerType(serializeUint32, parseUint32, testUint32); registerType(serializeInt32, parseInt32, testInt32); registerType(serializeUint16, parseUint16, testUint16); registerType(serializeInt16, parseInt16, testInt16); registerType(serializeUint8, parseUint8, testUint8); registerType(serializeInt8, parseInt8, testInt8); registerType(null, parseUndefined, testUndefined); registerType(null, parseNull, testNull); registerType(null, parseNaN, testNaN); registerType(serializeBoolean, parseBoolean, testBoolean); registerType(serializeSymbol, parseSymbol, testSymbol); registerType(null, parseEmptySlot, testEmptySlot); registerType(serializeMap, parseMap, testMap); registerType(serializeSet, parseSet, testSet); registerType(serializeDate, parseDate, testDate); registerType(serializeError, parseError, testError); registerType(serializeRegExp, parseRegExp, testRegExp); registerType(serializeStringObject, parseStringObject, testStringObject); registerType(serializeNumberObject, parseNumberObject, testNumberObject); registerType(serializeBooleanObject, parseBooleanObject, testBooleanObject); function registerType(serialize, parse, test, type) { if (type === undefined) { typeIndex++; if (types.length - typeIndex >= SPECIAL_TYPES.length) { types[types.length - typeIndex] = { serialize, parse, test }; } else { throw new Error("Reached maximum number of custom types"); } } else { types[type] = { serialize, parse, test }; } } async function serializeValue(data, value) { const type = types.findIndex(({ test } = {}) => test && test(value, data)); data.addObject(value); await data.append(new Uint8Array([type])); const serialize = types[type].serialize; if (serialize) { await serialize(data, value); } if (type != TYPE_REFERENCE && testObject(value)) { await serializeSymbols(data, value); await serializeOwnProperties(data, value); } } async function serializeSymbols(data, value) { const ownPropertySymbols = Object.getOwnPropertySymbols(value); const symbols = ownPropertySymbols.map(propertySymbol => [propertySymbol, value[propertySymbol]]); await serializeArray(data, symbols); } async function serializeOwnProperties(data, value) { if (!ArrayBuffer.isView(value)) { let entries = Object.entries(value); if (testArray(value)) { entries = entries.filter(([key]) => !testInteger(Number(key))); } await serializeValue(data, entries.length); for (const [key, value] of entries) { await serializeString(data, key); await serializeValue(data, value); } } else { await serializeValue(data, 0); } } async function serializeCircularReference(data, value) { const index = data.objects.indexOf(value); await serializeValue(data, index); } async function serializeArray(data, array) { await serializeValue(data, array.length); const notEmptyIndexes = Object.keys(array).filter(key => testInteger(Number(key))).map(key => Number(key)); let indexNotEmptyIndexes = 0, currentNotEmptyIndex = notEmptyIndexes[indexNotEmptyIndexes]; for (const [indexArray, value] of array.entries()) { if (currentNotEmptyIndex == indexArray) { currentNotEmptyIndex = notEmptyIndexes[++indexNotEmptyIndexes]; await serializeValue(data, value); } else { await serializeValue(data, EMPTY_SLOT_VALUE); } } } async function serializeString(data, string) { const encodedString = textEncoder.encode(string); await serializeValue(data, encodedString.length); await data.append(encodedString); } async function serializeTypedArray(data, array) { await serializeValue(data, array.length); await data.append(array.constructor.name == "Uint8Array" ? array : new Uint8Array(array.buffer)); } async function serializeArrayBuffer(data, arrayBuffer) { await serializeValue(data, arrayBuffer.byteLength); await data.append(new Uint8Array(arrayBuffer)); } async function serializeNumber(data, number) { const serializedNumber = new Uint8Array(new Float64Array([number]).buffer); await data.append(serializedNumber); } async function serializeUint32(data, number) { const serializedNumber = new Uint8Array(new Uint32Array([number]).buffer); await data.append(serializedNumber); } async function serializeInt32(data, number) { const serializedNumber = new Uint8Array(new Int32Array([number]).buffer); await data.append(serializedNumber); } async function serializeUint16(data, number) { const serializedNumber = new Uint8Array(new Uint16Array([number]).buffer); await data.append(serializedNumber); } async function serializeInt16(data, number) { const serializedNumber = new Uint8Array(new Int16Array([number]).buffer); await data.append(serializedNumber); } async function serializeUint8(data, number) { const serializedNumber = new Uint8Array([number]); await data.append(serializedNumber); } async function serializeInt8(data, number) { const serializedNumber = new Uint8Array(new Int8Array([number]).buffer); await data.append(serializedNumber); } async function serializeBoolean(data, boolean) { const serializedBoolean = new Uint8Array([Number(boolean)]); await data.append(serializedBoolean); } async function serializeMap(data, map) { const entries = map.entries(); await serializeValue(data, map.size); for (const [key, value] of entries) { await serializeValue(data, key); await serializeValue(data, value); } } async function serializeSet(data, set) { await serializeValue(data, set.size); for (const value of set) { await serializeValue(data, value); } } async function serializeDate(data, date) { await serializeNumber(data, date.getTime()); } async function serializeError(data, error) { await serializeString(data, error.message); await serializeString(data, error.stack); } async function serializeRegExp(data, regExp) { await serializeString(data, regExp.source); await serializeString(data, regExp.flags); } async function serializeStringObject(data, string) { await serializeString(data, string.valueOf()); } async function serializeNumberObject(data, number) { await serializeNumber(data, number.valueOf()); } async function serializeBooleanObject(data, boolean) { await serializeBoolean(data, boolean.valueOf()); } async function serializeSymbol(data, symbol) { await serializeString(data, symbol.description); } class Reference { constructor(index, data) { this.index = index; this.data = data; } getObject() { return this.data.objects[this.index]; } } async function parseValue(data) { const array = await data.consume(1); const parserType = array[0]; const parse = types[parserType].parse; const valueId = data.getObjectId(); const result = await parse(data); if (parserType != TYPE_REFERENCE && testObject(result)) { await parseSymbols(data, result); await parseOwnProperties(data, result); } data.resolveObject(valueId, result); return result; } async function parseSymbols(data, value) { const symbols = await parseArray(data); data.setObject([symbols], symbols => symbols.forEach(([symbol, propertyValue]) => value[symbol] = propertyValue)); } async function parseOwnProperties(data, object) { const size = await parseValue(data); if (size) { await parseNextProperty(); } async function parseNextProperty(indexKey = 0) { const key = await parseString(data); const value = await parseValue(data); data.setObject([value], value => object[key] = value); if (indexKey < size - 1) { await parseNextProperty(indexKey + 1); } } } async function parseCircularReference(data) { const index = await parseValue(data); const result = new Reference(index, data); return result; } function parseObject() { return {}; } async function parseArray(data) { const length = await parseValue(data); const array = new Array(length); if (length) { await parseNextSlot(); } return array; async function parseNextSlot(indexArray = 0) { const value = await parseValue(data); if (!testEmptySlot(value)) { data.setObject([value], value => array[indexArray] = value); } if (indexArray < length - 1) { await parseNextSlot(indexArray + 1); } } } function parseEmptySlot() { return EMPTY_SLOT_VALUE; } async function parseString(data) { const size = await parseValue(data); const array = await data.consume(size); return textDecoder.decode(array); } async function parseFloat64Array(data) { const length = await parseValue(data); const array = await data.consume(length * 8); return new Float64Array(array.buffer); } async function parseFloat32Array(data) { const length = await parseValue(data); const array = await data.consume(length * 4); return new Float32Array(array.buffer); } async function parseUint32Array(data) { const length = await parseValue(data); const array = await data.consume(length * 4); return new Uint32Array(array.buffer); } async function parseInt32Array(data) { const length = await parseValue(data); const array = await data.consume(length * 4); return new Int32Array(array.buffer); } async function parseUint16Array(data) { const length = await parseValue(data); const array = await data.consume(length * 2); return new Uint16Array(array.buffer); } async function parseInt16Array(data) { const length = await parseValue(data); const array = await data.consume(length * 2); return new Int16Array(array.buffer); } async function parseUint8ClampedArray(data) { const length = await parseValue(data); const array = await data.consume(length); return new Uint8ClampedArray(array.buffer); } async function parseUint8Array(data) { const length = await parseValue(data); const array = await data.consume(length); return array; } async function parseInt8Array(data) { const length = await parseValue(data); const array = await data.consume(length); return new Int8Array(array.buffer); } async function parseArrayBuffer(data) { const length = await parseValue(data); const array = await data.consume(length); return array.buffer; } async function parseNumber(data) { const array = await data.consume(8); return new Float64Array(array.buffer)[0]; } async function parseUint32(data) { const array = await data.consume(4); return new Uint32Array(array.buffer)[0]; } async function parseInt32(data) { const array = await data.consume(4); return new Int32Array(array.buffer)[0]; } async function parseUint16(data) { const array = await data.consume(2); return new Uint16Array(array.buffer)[0]; } async function parseInt16(data) { const array = await data.consume(2); return new Int16Array(array.buffer)[0]; } async function parseUint8(data) { const array = await data.consume(1); return new Uint8Array(array.buffer)[0]; } async function parseInt8(data) { const array = await data.consume(1); return new Int8Array(array.buffer)[0]; } function parseUndefined() { return undefined; } function parseNull() { return null; } function parseNaN() { return NaN; } async function parseBoolean(data) { const array = await data.consume(1); return Boolean(array[0]); } async function parseMap(data) { const size = await parseValue(data); const map = new Map(); if (size) { await parseNextEntry(); } return map; async function parseNextEntry(indexKey = 0) { const key = await parseValue(data); const value = await parseValue(data); data.setObject([key, value], (key, value) => map.set(key, value)); if (indexKey < size - 1) { await parseNextEntry(indexKey + 1); } } } async function parseSet(data) { const size = await parseValue(data); const set = new Set(); if (size) { await parseNextEntry(); } return set; async function parseNextEntry(indexKey = 0) { const value = await parseValue(data); data.setObject([value], value => set.add(value)); if (indexKey < size - 1) { await parseNextEntry(indexKey + 1); } } } async function parseDate(data) { const milliseconds = await parseNumber(data); return new Date(milliseconds); } async function parseError(data) { const message = await parseString(data); const stack = await parseString(data); const error = new Error(message); error.stack = stack; return error; } async function parseRegExp(data) { const source = await parseString(data); const flags = await parseString(data); return new RegExp(source, flags); } async function parseStringObject(data) { return new String(await parseString(data)); } async function parseNumberObject(data) { return new Number(await parseNumber(data)); } async function parseBooleanObject(data) { return new Boolean(await parseBoolean(data)); } async function parseSymbol(data) { const description = await parseString(data); return Symbol(description); } function testCircularReference(value, data) { return testObject(value) && data.objects.includes(value); } function testObject(value) { return value === Object(value); } function testArray(value) { return typeof value.length == "number"; } function testEmptySlot(value) { return value === EMPTY_SLOT_VALUE; } function testString(value) { return typeof value == "string"; } function testFloat64Array(value) { return value.constructor.name == "Float64Array"; } function testUint32Array(value) { return value.constructor.name == "Uint32Array"; } function testInt32Array(value) { return value.constructor.name == "Int32Array"; } function testUint16Array(value) { return value.constructor.name == "Uint16Array"; } function testFloat32Array(value) { return value.constructor.name == "Float32Array"; } function testInt16Array(value) { return value.constructor.name == "Int16Array"; } function testUint8ClampedArray(value) { return value.constructor.name == "Uint8ClampedArray"; } function testUint8Array(value) { return value.constructor.name == "Uint8Array"; } function testInt8Array(value) { return value.constructor.name == "Int8Array"; } function testArrayBuffer(value) { return value.constructor.name == "ArrayBuffer"; } function testNumber(value) { return typeof value == "number"; } function testUint32(value) { return testInteger(value) && value >= 0 && value <= 4294967295; } function testInt32(value) { return testInteger(value) && value >= -2147483648 && value <= 2147483647; } function testUint16(value) { return testInteger(value) && value >= 0 && value <= 65535; } function testInt16(value) { return testInteger(value) && value >= -32768 && value <= 32767; } function testUint8(value) { return testInteger(value) && value >= 0 && value <= 255; } function testInt8(value) { return testInteger(value) && value >= -128 && value <= 127; } function testInteger(value) { return testNumber(value) && Number.isInteger(value); } function testUndefined(value) { return value === undefined; } function testNull(value) { return value === null; } function testNaN(value) { return Number.isNaN(value); } function testBoolean(value) { return typeof value == "boolean"; } function testMap(value) { return value instanceof Map; } function testSet(value) { return value instanceof Set; } function testDate(value) { return value instanceof Date; } function testError(value) { return value instanceof Error; } function testRegExp(value) { return value instanceof RegExp; } function testStringObject(value) { return value instanceof String; } function testNumberObject(value) { return value instanceof Number; } function testBooleanObject(value) { return value instanceof Boolean; } function testSymbol(value) { return typeof value == "symbol"; } /* * Copyright 2010-2020 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, prompt, getComputedStyle, addEventListener, removeEventListener, requestAnimationFrame, setTimeout, getSelection, Node, URL */ const singlefile = globalThis.singlefile; singlefile.helper.SELECTED_CONTENT_ATTRIBUTE_NAME; const LOGS_WINDOW_TAGNAME = "singlefile-logs-window"; const LOGS_CLASSNAME = "singlefile-logs"; const LOGS_LINE_CLASSNAME = "singlefile-logs-line"; const LOGS_LINE_TEXT_ELEMENT_CLASSNAME = "singlefile-logs-line-text"; const LOGS_LINE_STATUS_ELEMENT_CLASSNAME = "singlefile-logs-line-icon"; const SINGLE_FILE_UI_ELEMENT_CLASS = singlefile.helper.SINGLE_FILE_UI_ELEMENT_CLASS; const CSS_PROPERTIES = new Set(Array.from(getComputedStyle(document.documentElement))); let LOG_PANEL_WIDTH, LOG_PANEL_DEFERRED_IMAGES_MESSAGE, LOG_PANEL_FRAME_CONTENTS_MESSAGE, LOG_PANEL_EMBEDDED_IMAGE_MESSAGE, LOG_PANEL_STEP_MESSAGE; try { LOG_PANEL_WIDTH = browser.i18n.getMessage("logPanelWidth"); LOG_PANEL_DEFERRED_IMAGES_MESSAGE = browser.i18n.getMessage("logPanelDeferredImages"); LOG_PANEL_FRAME_CONTENTS_MESSAGE = browser.i18n.getMessage("logPanelFrameContents"); LOG_PANEL_EMBEDDED_IMAGE_MESSAGE = browser.i18n.getMessage("logPanelEmbeddedImage"); LOG_PANEL_STEP_MESSAGE = browser.i18n.getMessage("logPanelStep"); // eslint-disable-next-line no-unused-vars } catch (error) { // ignored } let logsWindowElement; createLogsWindowElement(); function promptMessage(message, defaultValue) { return prompt(message, defaultValue); } function createLogsWindowElement() { try { logsWindowElement = document.querySelector(LOGS_WINDOW_TAGNAME); if (!logsWindowElement) { logsWindowElement = createElement(LOGS_WINDOW_TAGNAME); const shadowRoot = logsWindowElement.attachShadow({ mode: "open" }); const styleElement = document.createElement("style"); styleElement.textContent = ` @keyframes single-file-pulse { 0% { opacity: .25; } 100% { opacity: 1; } } .${LOGS_CLASSNAME} { position: fixed; bottom: 24px; left: 8px; z-index: 2147483647; opacity: 0.9; padding: 4px; background-color: white; min-width: ${LOG_PANEL_WIDTH}px; min-height: 16px; transition: height 100ms; } .${LOGS_LINE_CLASSNAME} { display: flex; justify-content: space-between; padding: 2px; font-family: arial, sans-serif; color: black; background-color: white; } .${LOGS_LINE_TEXT_ELEMENT_CLASSNAME} { font-size: 13px; opacity: 1; transition: opacity 200ms; } .${LOGS_LINE_STATUS_ELEMENT_CLASSNAME} { font-size: 11px; min-width: 15px; text-align: center; position: relative; top: 1px; } `; shadowRoot.appendChild(styleElement); const logsContentElement = document.createElement("div"); logsContentElement.classList.add(LOGS_CLASSNAME); shadowRoot.appendChild(logsContentElement); } // eslint-disable-next-line no-unused-vars } catch (error) { // ignored } } function createElement(tagName, parentElement) { const element = document.createElement(tagName); element.className = SINGLE_FILE_UI_ELEMENT_CLASS; CSS_PROPERTIES.forEach(property => element.style.setProperty(property, "initial", "important")); return element; } /* * Copyright 2010-2020 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. */ let EMBEDDED_IMAGE_BUTTON_MESSAGE, SHARE_PAGE_BUTTON_MESSAGE, SHARE_SELECTION_BUTTON_MESSAGE, ERROR_TITLE_MESSAGE; try { EMBEDDED_IMAGE_BUTTON_MESSAGE = browser.i18n.getMessage("topPanelEmbeddedImageButton"); SHARE_PAGE_BUTTON_MESSAGE = browser.i18n.getMessage("topPanelSharePageButton"); SHARE_SELECTION_BUTTON_MESSAGE = browser.i18n.getMessage("topPanelShareSelectionButton"); ERROR_TITLE_MESSAGE = browser.i18n.getMessage("topPanelError"); // eslint-disable-next-line no-unused-vars } catch (error) { // ignored } let sharePageBar; setLabels({ EMBEDDED_IMAGE_BUTTON_MESSAGE, SHARE_PAGE_BUTTON_MESSAGE, SHARE_SELECTION_BUTTON_MESSAGE, ERROR_TITLE_MESSAGE }); async function downloadPageForeground(pageData, options) { if (Array.isArray(pageData.content)) { pageData.content = new Uint8Array(pageData.content); } if (options.sharePage && navigator.share) { await sharePage(pageData, options); } else { let filename = pageData.filename; if (options.confirmFilename) { filename = promptMessage("Save as", pageData.filename); if (filename) { pageData.filename = filename; } else { browser.runtime.sendMessage({ method: "downloads.cancel" }); browser.runtime.sendMessage({ method: "ui.processCancelled" }); } } if (filename) { const link = document.createElement("a"); link.download = pageData.filename; link.href = URL.createObjectURL(new Blob([pageData.content], { type: pageData.mimeType })); link.dispatchEvent(new MouseEvent("click")); return new Promise(resolve => setTimeout(() => { URL.revokeObjectURL(link.href); resolve(); }, 1000)); } } } async function sharePage(pageData, options) { sharePageBar = getSharePageBar(); const cancelled = await sharePageBar.display(options.selected); if (!cancelled) { const data = { files: [new File([pageData.content], pageData.filename, { type: pageData.mimeType })] }; try { await navigator.share(data); sharePageBar.hide(); } catch (error) { sharePageBar.hide(); if (error.name === "AbortError") { await sharePage(pageData, options); } else { throw error; } } } } /* * Copyright 2010-2020 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. */ (globalThis => { const IS_NOT_SAFARI = !/Safari/.test(navigator.userAgent) || /Chrome/.test(navigator.userAgent) || /Vivaldi/.test(navigator.userAgent) || /OPR/.test(navigator.userAgent); const singlefile = globalThis.singlefile; const FORBIDDEN_TAG_NAMES = ["a", "area", "audio", "base", "br", "col", "command", "embed", "hr", "img", "iframe", "input", "keygen", "link", "meta", "param", "source", "track", "video", "wbr"]; const BUTTON_ANCHOR_URL = ""; const BUTTON_CLOSE_URL = ""; const SHADOWROOT_ATTRIBUTE_NAME = "shadowrootmode"; const SCRIPT_TEMPLATE_SHADOW_ROOT = "data-template-shadow-root"; const SCRIPT_OPTIONS = "data-single-file-options"; const NOTE_TAGNAME = "single-file-note"; const NOTE_CLASS = "note"; const NOTE_MASK_CLASS = "note-mask"; const NOTE_HIDDEN_CLASS = "note-hidden"; const NOTE_ANCHORED_CLASS = "note-anchored"; const NOTE_SELECTED_CLASS = "note-selected"; const NOTE_MOVING_CLASS = "note-moving"; const NOTE_MASK_MOVING_CLASS = "note-mask-moving"; const PAGE_MASK_CLASS = "page-mask"; const MASK_CLASS = "single-file-mask"; const PAGE_MASK_CONTAINER_CLASS = "single-file-page-mask"; const HIGHLIGHT_CLASS = "single-file-highlight"; const HIGHLIGHTS_STYLESHEET_CLASS = "single-file-highlights-stylesheet"; const REMOVED_CONTENT_CLASS = "single-file-removed"; const HIGHLIGHT_HIDDEN_CLASS = "single-file-highlight-hidden"; const PAGE_MASK_ACTIVE_CLASS = "page-mask-active"; const CUT_HOVER_CLASS = "single-file-cut-hover"; const CUT_OUTER_HOVER_CLASS = "single-file-cut-outer-hover"; const CUT_SELECTED_CLASS = "single-file-cut-selected"; const CUT_OUTER_SELECTED_CLASS = "single-file-cut-outer-selected"; const CUT_MODE_CLASS = "single-file-cut-mode"; const NOTE_INITIAL_POSITION_X = 20; const NOTE_INITIAL_POSITION_Y = 20; const NOTE_INITIAL_WIDTH = 150; const NOTE_INITIAL_HEIGHT = 150; const NOTE_HEADER_HEIGHT = 25; const DISABLED_NOSCRIPT_ATTRIBUTE_NAME = "data-single-file-disabled-noscript"; const COMMENT_HEADER = "Page saved with SingleFile"; const COMMENT_HEADER_LEGACY = "Archive processed by SingleFile"; let NOTES_WEB_STYLESHEET, MASK_WEB_STYLESHEET, HIGHLIGHTS_WEB_STYLESHEET; let selectedNote, anchorElement, maskNoteElement, maskPageElement, highlightSelectionMode, removeHighlightMode, resizingNoteMode, movingNoteMode, highlightColor, collapseNoteTimeout, cuttingOuterMode, cuttingMode, cuttingTouchTarget, cuttingPath, cuttingPathIndex, previousContent; let removedElements = [], removedElementIndex = 0, initScriptContent, pageResources, pageUrl, pageCompressContent, includeInfobar, openInfobar, infobarPositionAbsolute, infobarPositionTop, infobarPositionBottom, infobarPositionLeft, infobarPositionRight; globalThis.zip = singlefile.helper.zip; initEventListeners(); new MutationObserver(initEventListeners).observe(document, { childList: true }); function initEventListeners() { window.onmessage = async event => { const message = JSON.parse(event.data); if (message.method == "init") { await init(message); } if (message.method == "addNote") { addNote(message); } if (message.method == "displayNotes") { document.querySelectorAll(NOTE_TAGNAME).forEach(noteElement => noteElement.shadowRoot.querySelector("." + NOTE_CLASS).classList.remove(NOTE_HIDDEN_CLASS)); } if (message.method == "hideNotes") { document.querySelectorAll(NOTE_TAGNAME).forEach(noteElement => noteElement.shadowRoot.querySelector("." + NOTE_CLASS).classList.add(NOTE_HIDDEN_CLASS)); } if (message.method == "enableHighlight") { if (highlightColor) { document.documentElement.classList.remove(highlightColor + "-mode"); } highlightColor = message.color; highlightSelectionMode = true; document.documentElement.classList.add(message.color + "-mode"); } if (message.method == "disableHighlight") { disableHighlight(); highlightSelectionMode = false; } if (message.method == "displayHighlights") { document.querySelectorAll("." + HIGHLIGHT_CLASS).forEach(noteElement => noteElement.classList.remove(HIGHLIGHT_HIDDEN_CLASS)); } if (message.method == "hideHighlights") { document.querySelectorAll("." + HIGHLIGHT_CLASS).forEach(noteElement => noteElement.classList.add(HIGHLIGHT_HIDDEN_CLASS)); } if (message.method == "enableRemoveHighlights") { removeHighlightMode = true; document.documentElement.classList.add("single-file-remove-highlights-mode"); } if (message.method == "disableRemoveHighlights") { removeHighlightMode = false; document.documentElement.classList.remove("single-file-remove-highlights-mode"); } if (message.method == "enableEditPage") { document.body.contentEditable = true; onUpdate(false); } if (message.method == "formatPage") { formatPage(!message.applySystemTheme, message.contentWidth); } if (message.method == "cancelFormatPage") { cancelFormatPage(); } if (message.method == "disableEditPage") { document.body.contentEditable = false; } if (message.method == "enableCutInnerPage") { cuttingMode = true; document.documentElement.classList.add(CUT_MODE_CLASS); } if (message.method == "enableCutOuterPage") { cuttingOuterMode = true; document.documentElement.classList.add(CUT_MODE_CLASS); } if (message.method == "disableCutInnerPage" || message.method == "disableCutOuterPage") { if (message.method == "disableCutInnerPage") { cuttingMode = false; } else { cuttingOuterMode = false; } document.documentElement.classList.remove(CUT_MODE_CLASS); resetSelectedElements(); if (cuttingPath) { unhighlightCutElement(); cuttingPath = null; } } if (message.method == "undoCutPage") { undoCutPage(); } if (message.method == "undoAllCutPage") { while (removedElementIndex) { removedElements[removedElementIndex - 1].forEach(element => element.classList.remove(REMOVED_CONTENT_CLASS)); removedElementIndex--; } } if (message.method == "redoCutPage") { redoCutPage(); } if (message.method == "getContent") { onUpdate(true); includeInfobar = message.includeInfobar; openInfobar = message.openInfobar; infobarPositionAbsolute = message.infobarPositionAbsolute; infobarPositionTop = message.infobarPositionTop; infobarPositionBottom = message.infobarPositionBottom; infobarPositionLeft = message.infobarPositionLeft; infobarPositionRight = message.infobarPositionRight; let content = getContent(message.compressHTML); if (initScriptContent) { content = content.replace(/"); const contentDocument = (new DOMParser()).parseFromString(content, "text/html"); if (detectSavedPage(contentDocument)) { if (contentDocument.doctype) { if (document.doctype) { document.replaceChild(contentDocument.doctype, document.doctype); } else { document.insertBefore(contentDocument.doctype, document.documentElement); } } else if (document.doctype) { document.doctype.remove(); } const infobarElement = contentDocument.querySelector(singlefile.helper.INFOBAR_TAGNAME); if (infobarElement) { infobarElement.remove(); } contentDocument.querySelectorAll("noscript").forEach(element => { element.setAttribute(DISABLED_NOSCRIPT_ATTRIBUTE_NAME, element.innerHTML); element.textContent = ""; }); contentDocument.querySelectorAll("iframe").forEach(element => { const pointerEvents = "pointer-events"; element.style.setProperty("-sf-" + pointerEvents, element.style.getPropertyValue(pointerEvents), element.style.getPropertyPriority(pointerEvents)); element.style.setProperty(pointerEvents, "none", "important"); }); document.replaceChild(contentDocument.documentElement, document.documentElement); singlefile.helper.fixInvalidNesting(document); document.querySelectorAll("[data-single-file-note-refs]").forEach(noteRefElement => noteRefElement.dataset.singleFileNoteRefs = noteRefElement.dataset.singleFileNoteRefs.replace(/,/g, " ")); deserializeShadowRoots(document); document.querySelectorAll(NOTE_TAGNAME).forEach(containerElement => attachNoteListeners(containerElement, true)); insertHighlightStylesheet(document); maskPageElement = getMaskElement(PAGE_MASK_CLASS, PAGE_MASK_CONTAINER_CLASS); maskNoteElement = getMaskElement(NOTE_MASK_CLASS); document.documentElement.onmousedown = onMouseDown; document.documentElement.onmouseup = document.documentElement.ontouchend = onMouseUp; document.documentElement.onmouseover = onMouseOver; document.documentElement.onmouseout = onMouseOut; document.documentElement.onkeydown = onKeyDown; document.documentElement.ontouchstart = document.documentElement.ontouchmove = onTouchMove; window.onclick = event => event.preventDefault(); const iconElement = document.querySelector("link[rel*=icon]"); window.parent.postMessage(JSON.stringify({ method: "onInit", title: document.title, icon: iconElement && iconElement.href, filename, reset, formatPageEnabled: isProbablyReaderable(document) }), "*"); } } } function loadOptionsFromPage(doc) { const optionsElement = doc.body.querySelector("script[type=\"application/json\"][" + SCRIPT_OPTIONS + "]"); if (optionsElement) { return JSON.parse(optionsElement.textContent); } } async function initPage() { document.querySelectorAll("iframe").forEach(element => { const pointerEvents = "pointer-events"; element.style.setProperty("-sf-" + pointerEvents, element.style.getPropertyValue(pointerEvents), element.style.getPropertyPriority(pointerEvents)); element.style.setProperty(pointerEvents, "none", "important"); }); document.querySelectorAll("[data-single-file-note-refs]").forEach(noteRefElement => noteRefElement.dataset.singleFileNoteRefs = noteRefElement.dataset.singleFileNoteRefs.replace(/,/g, " ")); deserializeShadowRoots(document); reflowNotes(); await waitResourcesLoad(); reflowNotes(); document.querySelectorAll(NOTE_TAGNAME).forEach(containerElement => attachNoteListeners(containerElement, true)); insertHighlightStylesheet(document); maskPageElement = getMaskElement(PAGE_MASK_CLASS, PAGE_MASK_CONTAINER_CLASS); maskNoteElement = getMaskElement(NOTE_MASK_CLASS); document.documentElement.onmousedown = onMouseDown; document.documentElement.onmouseup = document.documentElement.ontouchend = onMouseUp; document.documentElement.onmouseover = onMouseOver; document.documentElement.onmouseout = onMouseOut; document.documentElement.onkeydown = onKeyDown; document.documentElement.ontouchstart = document.documentElement.ontouchmove = onTouchMove; window.onclick = event => event.preventDefault(); } async function initConstants() { [NOTES_WEB_STYLESHEET, MASK_WEB_STYLESHEET, HIGHLIGHTS_WEB_STYLESHEET] = await Promise.all([ minifyText(await ((await fetch("../pages/editor-note-web.css")).text())), minifyText(await ((await fetch("../pages/editor-mask-web.css")).text())), minifyText(await ((await fetch("../pages/editor-frame-web.css")).text())) ]); } function addNote({ color }) { const containerElement = document.createElement(NOTE_TAGNAME); const noteElement = document.createElement("div"); const headerElement = document.createElement("header"); const blockquoteElement = document.createElement("blockquote"); const mainElement = document.createElement("textarea"); const resizeElement = document.createElement("div"); const removeNoteElement = document.createElement("img"); const anchorIconElement = document.createElement("img"); const noteShadow = containerElement.attachShadow({ mode: "open" }); headerElement.appendChild(anchorIconElement); headerElement.appendChild(removeNoteElement); blockquoteElement.appendChild(mainElement); noteElement.appendChild(headerElement); noteElement.appendChild(blockquoteElement); noteElement.appendChild(resizeElement); noteShadow.appendChild(getStyleElement(NOTES_WEB_STYLESHEET)); noteShadow.appendChild(noteElement); const notesElements = Array.from(document.querySelectorAll(NOTE_TAGNAME)); const noteId = Math.max.call(Math, 0, ...notesElements.map(noteElement => Number(noteElement.dataset.noteId))) + 1; blockquoteElement.cite = "https://www.w3.org/TR/annotation-model/#selector(type=CssSelector,value=[data-single-file-note-refs~=\"" + noteId + "\"])"; noteElement.classList.add(NOTE_CLASS); noteElement.classList.add(NOTE_ANCHORED_CLASS); noteElement.classList.add(color); noteElement.dataset.color = color; mainElement.dir = "auto"; const boundingRectDocument = document.documentElement.getBoundingClientRect(); let positionX = NOTE_INITIAL_WIDTH + NOTE_INITIAL_POSITION_X - 1 - boundingRectDocument.x; let positionY = NOTE_INITIAL_HEIGHT + NOTE_INITIAL_POSITION_Y - 1 - boundingRectDocument.y; while (Array.from(document.elementsFromPoint(positionX - window.scrollX, positionY - window.scrollY)).find(element => element.tagName.toLowerCase() == NOTE_TAGNAME)) { positionX += NOTE_INITIAL_POSITION_X; positionY += NOTE_INITIAL_POSITION_Y; } noteElement.style.setProperty("left", (positionX - NOTE_INITIAL_WIDTH - 1) + "px"); noteElement.style.setProperty("top", (positionY - NOTE_INITIAL_HEIGHT - 1) + "px"); resizeElement.className = "note-resize"; resizeElement.ondragstart = event => event.preventDefault(); removeNoteElement.className = "note-remove"; removeNoteElement.src = BUTTON_CLOSE_URL; removeNoteElement.ondragstart = event => event.preventDefault(); anchorIconElement.className = "note-anchor"; anchorIconElement.src = BUTTON_ANCHOR_URL; anchorIconElement.ondragstart = event => event.preventDefault(); containerElement.dataset.noteId = noteId; addNoteRef(document.documentElement, noteId); attachNoteListeners(containerElement, true); document.documentElement.insertBefore(containerElement, maskPageElement.getRootNode().host); noteElement.classList.add(NOTE_SELECTED_CLASS); selectedNote = noteElement; onUpdate(false); } function attachNoteListeners(containerElement, editable = false) { const SELECT_PX_THRESHOLD = 4; const COLLAPSING_NOTE_DELAY = 750; const noteShadow = containerElement.shadowRoot; const noteElement = noteShadow.childNodes[1]; const headerElement = noteShadow.querySelector("header"); const mainElement = noteShadow.querySelector("textarea"); const noteId = containerElement.dataset.noteId; const resizeElement = noteShadow.querySelector(".note-resize"); const anchorIconElement = noteShadow.querySelector(".note-anchor"); const removeNoteElement = noteShadow.querySelector(".note-remove"); mainElement.readOnly = !editable; if (!editable) { anchorIconElement.style.setProperty("display", "none", "important"); } else { anchorIconElement.style.removeProperty("display"); } headerElement.ontouchstart = headerElement.onmousedown = event => { if (event.target == headerElement) { collapseNoteTimeout = setTimeout(() => { noteElement.classList.toggle("note-collapsed"); hideMaskNote(); }, COLLAPSING_NOTE_DELAY); event.preventDefault(); const position = getPosition(event); const clientX = position.clientX; const clientY = position.clientY; const boundingRect = noteElement.getBoundingClientRect(); const deltaX = clientX - boundingRect.left; const deltaY = clientY - boundingRect.top; maskPageElement.classList.add(PAGE_MASK_ACTIVE_CLASS); document.documentElement.style.setProperty("user-select", "none", "important"); anchorElement = getAnchorElement(containerElement); displayMaskNote(); selectNote(noteElement); moveNote(event, deltaX, deltaY); movingNoteMode = { event, deltaX, deltaY }; document.documentElement.ontouchmove = document.documentElement.onmousemove = event => { clearTimeout(collapseNoteTimeout); if (!movingNoteMode) { movingNoteMode = { deltaX, deltaY }; } movingNoteMode.event = event; moveNote(event, deltaX, deltaY); }; } }; resizeElement.ontouchstart = resizeElement.onmousedown = event => { event.preventDefault(); resizingNoteMode = true; selectNote(noteElement); maskPageElement.classList.add(PAGE_MASK_ACTIVE_CLASS); document.documentElement.style.setProperty("user-select", "none", "important"); document.documentElement.ontouchmove = document.documentElement.onmousemove = event => { event.preventDefault(); const { clientX, clientY } = getPosition(event); const boundingRectNote = noteElement.getBoundingClientRect(); noteElement.style.width = clientX - boundingRectNote.left + "px"; noteElement.style.height = clientY - boundingRectNote.top + "px"; }; }; anchorIconElement.ontouchend = anchorIconElement.onclick = event => { event.preventDefault(); noteElement.classList.toggle(NOTE_ANCHORED_CLASS); if (!noteElement.classList.contains(NOTE_ANCHORED_CLASS)) { deleteNoteRef(containerElement, noteId); addNoteRef(document.documentElement, noteId); } onUpdate(false); }; removeNoteElement.ontouchend = removeNoteElement.onclick = event => { event.preventDefault(); deleteNoteRef(containerElement, noteId); containerElement.remove(); }; noteElement.onmousedown = () => { selectNote(noteElement); }; function moveNote(event, deltaX, deltaY) { event.preventDefault(); const { clientX, clientY } = getPosition(event); noteElement.classList.add(NOTE_MOVING_CLASS); if (editable) { if (noteElement.classList.contains(NOTE_ANCHORED_CLASS)) { deleteNoteRef(containerElement, noteId); anchorElement = getTarget(clientX, clientY) || document.documentElement; addNoteRef(anchorElement, noteId); } else { anchorElement = document.documentElement; } } document.documentElement.insertBefore(containerElement, maskPageElement.getRootNode().host); noteElement.style.setProperty("left", (clientX - deltaX) + "px"); noteElement.style.setProperty("top", (clientY - deltaY) + "px"); noteElement.style.setProperty("position", "fixed"); displayMaskNote(); } function displayMaskNote() { if (anchorElement == document.documentElement || anchorElement == document.documentElement) { hideMaskNote(); } else { const boundingRectAnchor = anchorElement.getBoundingClientRect(); maskNoteElement.classList.add(NOTE_MASK_MOVING_CLASS); if (selectedNote) { maskNoteElement.classList.add(selectedNote.dataset.color); } maskNoteElement.style.setProperty("top", (boundingRectAnchor.y - 3) + "px"); maskNoteElement.style.setProperty("left", (boundingRectAnchor.x - 3) + "px"); maskNoteElement.style.setProperty("width", (boundingRectAnchor.width + 3) + "px"); maskNoteElement.style.setProperty("height", (boundingRectAnchor.height + 3) + "px"); } } function hideMaskNote() { maskNoteElement.classList.remove(NOTE_MASK_MOVING_CLASS); if (selectedNote) { maskNoteElement.classList.remove(selectedNote.dataset.color); } } function selectNote(noteElement) { if (selectedNote) { selectedNote.classList.remove(NOTE_SELECTED_CLASS); maskNoteElement.classList.remove(selectedNote.dataset.color); } noteElement.classList.add(NOTE_SELECTED_CLASS); noteElement.classList.add(noteElement.dataset.color); selectedNote = noteElement; } function getTarget(clientX, clientY) { const targets = Array.from(document.elementsFromPoint(clientX, clientY)).filter(element => element.matches("html *:not(" + NOTE_TAGNAME + "):not(." + MASK_CLASS + ")")); if (!targets.includes(document.documentElement)) { targets.push(document.documentElement); } let newTarget, target = targets[0], boundingRect = target.getBoundingClientRect(); newTarget = determineTargetElement("floor", target, clientX - boundingRect.left, getMatchedParents(target, "left")); if (newTarget == target) { newTarget = determineTargetElement("ceil", target, boundingRect.left + boundingRect.width - clientX, getMatchedParents(target, "right")); } if (newTarget == target) { newTarget = determineTargetElement("floor", target, clientY - boundingRect.top, getMatchedParents(target, "top")); } if (newTarget == target) { newTarget = determineTargetElement("ceil", target, boundingRect.top + boundingRect.height - clientY, getMatchedParents(target, "bottom")); } target = newTarget; while (boundingRect = target && target.getBoundingClientRect(), boundingRect && boundingRect.width <= SELECT_PX_THRESHOLD && boundingRect.height <= SELECT_PX_THRESHOLD) { target = target.parentElement; } return target; } function getMatchedParents(target, property) { let element = target, matchedParent, parents = []; do { const boundingRect = element.getBoundingClientRect(); if (element.parentElement && !element.parentElement.tagName.toLowerCase() != NOTE_TAGNAME && !element.classList.contains(MASK_CLASS)) { const parentBoundingRect = element.parentElement.getBoundingClientRect(); matchedParent = Math.abs(parentBoundingRect[property] - boundingRect[property]) <= SELECT_PX_THRESHOLD; if (matchedParent) { if (element.parentElement.clientWidth > SELECT_PX_THRESHOLD && element.parentElement.clientHeight > SELECT_PX_THRESHOLD && ((element.parentElement.clientWidth - element.clientWidth > SELECT_PX_THRESHOLD) || (element.parentElement.clientHeight - element.clientHeight > SELECT_PX_THRESHOLD))) { parents.push(element.parentElement); } element = element.parentElement; } } else { matchedParent = false; } } while (matchedParent && element); return parents; } function determineTargetElement(roundingMethod, target, widthDistance, parents) { if (Math[roundingMethod](widthDistance / SELECT_PX_THRESHOLD) <= parents.length) { target = parents[parents.length - Math[roundingMethod](widthDistance / SELECT_PX_THRESHOLD) - 1]; } return target; } } function onMouseDown(event) { if ((cuttingMode || cuttingOuterMode) && cuttingPath) { event.preventDefault(); } } function onMouseUp(event) { if (highlightSelectionMode) { event.preventDefault(); highlightSelection(); onUpdate(false); } if (removeHighlightMode) { event.preventDefault(); let element = event.target, done; while (element && !done) { if (element.classList.contains(HIGHLIGHT_CLASS)) { document.querySelectorAll("." + HIGHLIGHT_CLASS + "[data-singlefile-highlight-id=" + JSON.stringify(element.dataset.singlefileHighlightId) + "]").forEach(highlightedElement => { resetHighlightedElement(highlightedElement); onUpdate(false); }); done = true; } element = element.parentElement; } } if (resizingNoteMode) { event.preventDefault(); resizingNoteMode = false; document.documentElement.style.removeProperty("user-select"); maskPageElement.classList.remove(PAGE_MASK_ACTIVE_CLASS); document.documentElement.ontouchmove = onTouchMove; document.documentElement.onmousemove = null; onUpdate(false); } if (movingNoteMode) { event.preventDefault(); anchorNote(movingNoteMode.event || event, selectedNote, movingNoteMode.deltaX, movingNoteMode.deltaY); movingNoteMode = null; document.documentElement.ontouchmove = onTouchMove; document.documentElement.onmousemove = null; onUpdate(false); } if ((cuttingMode || cuttingOuterMode) && cuttingPath) { event.preventDefault(); if (event.ctrlKey) { const element = cuttingPath[cuttingPathIndex]; element.classList.toggle(cuttingMode ? CUT_SELECTED_CLASS : CUT_OUTER_SELECTED_CLASS); } else { validateCutElement(event.shiftKey); } } if (collapseNoteTimeout) { clearTimeout(collapseNoteTimeout); collapseNoteTimeout = null; } } function onMouseOver(event) { if (cuttingMode || cuttingOuterMode) { const target = event.target; if (target.classList) { let ancestorFound; document.querySelectorAll("." + (cuttingMode ? CUT_SELECTED_CLASS : CUT_OUTER_SELECTED_CLASS)).forEach(element => { if (element == target || isAncestor(element, target) || isAncestor(target, element)) { ancestorFound = element; } }); if (ancestorFound) { cuttingPath = [ancestorFound]; } else { cuttingPath = getParents(event.target); } cuttingPathIndex = 0; highlightCutElement(); } } } function onMouseOut() { if (cuttingMode || cuttingOuterMode) { if (cuttingPath) { unhighlightCutElement(); cuttingPath = null; } } } function onTouchMove(event) { if (cuttingMode || cuttingOuterMode) { event.preventDefault(); const { clientX, clientY } = getPosition(event); const target = document.elementFromPoint(clientX, clientY); if (cuttingTouchTarget != target) { onMouseOut(); if (target) { cuttingTouchTarget = target; onMouseOver({ target }); } } } } function onKeyDown(event) { if (cuttingMode || cuttingOuterMode) { if (event.code == "Tab") { if (cuttingPath) { const delta = event.shiftKey ? -1 : 1; let element = cuttingPath[cuttingPathIndex]; let nextElement = cuttingPath[cuttingPathIndex + delta]; if (nextElement) { let pathIndex = cuttingPathIndex + delta; while ( nextElement && ( (delta == 1 && element.getBoundingClientRect().width >= nextElement.getBoundingClientRect().width && element.getBoundingClientRect().height >= nextElement.getBoundingClientRect().height) || (delta == -1 && element.getBoundingClientRect().width <= nextElement.getBoundingClientRect().width && element.getBoundingClientRect().height <= nextElement.getBoundingClientRect().height))) { pathIndex += delta; nextElement = cuttingPath[pathIndex]; } if (nextElement && nextElement.classList && nextElement != document.body && nextElement != document.documentElement) { unhighlightCutElement(); cuttingPathIndex = pathIndex; highlightCutElement(); } } } event.preventDefault(); } if (event.code == "Space") { if (cuttingPath) { if (event.ctrlKey) { const element = cuttingPath[cuttingPathIndex]; element.classList.add(cuttingMode ? CUT_SELECTED_CLASS : CUT_OUTER_SELECTED_CLASS); } else { validateCutElement(event.shiftKey); } event.preventDefault(); } } if (event.code == "Escape") { resetSelectedElements(); event.preventDefault(); } if (event.key.toLowerCase() == "z" && event.ctrlKey) { if (event.shiftKey) { redoCutPage(); } else { undoCutPage(); } event.preventDefault(); } } if (event.key.toLowerCase() == "s" && event.ctrlKey) { window.parent.postMessage(JSON.stringify({ "method": "savePage" }), "*"); event.preventDefault(); } if (event.key.toLowerCase() == "p" && event.ctrlKey) { printPage(); event.preventDefault(); } } function printPage() { unhighlightCutElement(); resetSelectedElements(); window.print(); } function highlightCutElement() { const element = cuttingPath[cuttingPathIndex]; element.classList.add(cuttingMode ? CUT_HOVER_CLASS : CUT_OUTER_HOVER_CLASS); } function unhighlightCutElement() { if (cuttingPath) { const element = cuttingPath[cuttingPathIndex]; element.classList.remove(CUT_HOVER_CLASS); element.classList.remove(CUT_OUTER_HOVER_CLASS); } } function disableHighlight(doc = document) { if (highlightColor) { doc.documentElement.classList.remove(highlightColor + "-mode"); } } function undoCutPage() { if (removedElementIndex) { removedElements[removedElementIndex - 1].forEach(element => element.classList.remove(REMOVED_CONTENT_CLASS)); removedElementIndex--; } } function redoCutPage() { if (removedElementIndex < removedElements.length) { removedElements[removedElementIndex].forEach(element => element.classList.add(REMOVED_CONTENT_CLASS)); removedElementIndex++; } } function validateCutElement(invert) { const selectedElement = cuttingPath[cuttingPathIndex]; if ((cuttingMode && !invert) || (cuttingOuterMode && invert)) { if (document.documentElement != selectedElement && selectedElement.tagName.toLowerCase() != NOTE_TAGNAME) { const elementsRemoved = [selectedElement].concat(...document.querySelectorAll("." + CUT_SELECTED_CLASS + ",." + CUT_SELECTED_CLASS + " *,." + CUT_HOVER_CLASS + " *")); resetSelectedElements(); if (elementsRemoved.length) { elementsRemoved.forEach(element => { unhighlightCutElement(); if (element.tagName.toLowerCase() == NOTE_TAGNAME) { resetAnchorNote(element); } else { element.classList.add(REMOVED_CONTENT_CLASS); } }); removedElements[removedElementIndex] = elementsRemoved; removedElementIndex++; removedElements.length = removedElementIndex; onUpdate(false); } } } else { if (document.documentElement != selectedElement && selectedElement.tagName.toLowerCase() != NOTE_TAGNAME) { const elements = []; const searchSelector = "*:not(style):not(meta):not(." + REMOVED_CONTENT_CLASS + ")"; const elementsKept = [selectedElement].concat(...document.querySelectorAll("." + CUT_OUTER_SELECTED_CLASS)); document.body.querySelectorAll(searchSelector).forEach(element => { let removed = true; elementsKept.forEach(elementKept => removed = removed && (elementKept != element && !isAncestor(elementKept, element) && !isAncestor(element, elementKept))); if (removed) { if (element.tagName.toLowerCase() == NOTE_TAGNAME) { resetAnchorNote(element); } else { elements.push(element); } } }); elementsKept.forEach(elementKept => { unhighlightCutElement(); const elementKeptRect = elementKept.getBoundingClientRect(); elementKept.querySelectorAll(searchSelector).forEach(descendant => { const descendantRect = descendant.getBoundingClientRect(); if (descendantRect.width && descendantRect.height && ( descendantRect.left + descendantRect.width < elementKeptRect.left || descendantRect.right > elementKeptRect.right + elementKeptRect.width || descendantRect.top + descendantRect.height < elementKeptRect.top || descendantRect.bottom > elementKeptRect.bottom + elementKeptRect.height )) { elements.push(descendant); } }); }); resetSelectedElements(); if (elements.length) { elements.forEach(element => element.classList.add(REMOVED_CONTENT_CLASS)); removedElements[removedElementIndex] = elements; removedElementIndex++; removedElements.length = removedElementIndex; onUpdate(false); } } } } function resetSelectedElements(doc = document) { doc.querySelectorAll("." + CUT_OUTER_SELECTED_CLASS + ",." + CUT_SELECTED_CLASS).forEach(element => { element.classList.remove(CUT_OUTER_SELECTED_CLASS); element.classList.remove(CUT_SELECTED_CLASS); }); } function anchorNote(event, noteElement, deltaX, deltaY) { event.preventDefault(); const { clientX, clientY } = getPosition(event); document.documentElement.style.removeProperty("user-select"); noteElement.classList.remove(NOTE_MOVING_CLASS); maskNoteElement.classList.remove(NOTE_MASK_MOVING_CLASS); maskPageElement.classList.remove(PAGE_MASK_ACTIVE_CLASS); maskNoteElement.classList.remove(noteElement.dataset.color); const headerElement = noteElement.querySelector("header"); headerElement.ontouchmove = document.documentElement.onmousemove = null; let currentElement = anchorElement; let positionedElement; while (currentElement.parentElement && !positionedElement) { if (!FORBIDDEN_TAG_NAMES.includes(currentElement.tagName.toLowerCase())) { const currentElementStyle = getComputedStyle(currentElement); if (currentElementStyle.position != "static") { positionedElement = currentElement; } } currentElement = currentElement.parentElement; } if (!positionedElement) { positionedElement = document.documentElement; } const containerElement = noteElement.getRootNode().host; if (positionedElement == document.documentElement) { const firstMaskElement = document.querySelector("." + MASK_CLASS); firstMaskElement.parentElement.insertBefore(containerElement, firstMaskElement); } else { positionedElement.appendChild(containerElement); } const boundingRectPositionedElement = positionedElement.getBoundingClientRect(); const stylePositionedElement = window.getComputedStyle(positionedElement); const borderX = parseInt(stylePositionedElement.getPropertyValue("border-left-width")); const borderY = parseInt(stylePositionedElement.getPropertyValue("border-top-width")); noteElement.style.setProperty("position", "absolute"); noteElement.style.setProperty("left", (clientX - boundingRectPositionedElement.x - deltaX - borderX) + "px"); noteElement.style.setProperty("top", (clientY - boundingRectPositionedElement.y - deltaY - borderY) + "px"); } function resetAnchorNote(containerElement) { const noteId = containerElement.dataset.noteId; const noteElement = containerElement.shadowRoot.childNodes[1]; noteElement.classList.remove(NOTE_ANCHORED_CLASS); deleteNoteRef(containerElement, noteId); addNoteRef(document.documentElement, noteId); document.documentElement.insertBefore(containerElement, maskPageElement.getRootNode().host); } function getPosition(event) { if (event.touches && event.touches.length) { const touch = event.touches[0]; return touch; } else { return event; } } function highlightSelection() { let highlightId = 0; document.querySelectorAll("." + HIGHLIGHT_CLASS).forEach(highlightedElement => highlightId = Math.max(highlightId, highlightedElement.dataset.singlefileHighlightId)); highlightId++; const selection = window.getSelection(); const highlightedNodes = new Set(); for (let indexRange = 0; indexRange < selection.rangeCount; indexRange++) { const range = selection.getRangeAt(indexRange); if (!range.collapsed) { if (range.commonAncestorContainer.nodeType == range.commonAncestorContainer.TEXT_NODE) { let contentText = range.startContainer.splitText(range.startOffset); contentText = contentText.splitText(range.endOffset); addHighLightedNode(contentText.previousSibling); } else { const treeWalker = document.createTreeWalker(range.commonAncestorContainer, NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_TEXT); let highlightNodes; while (treeWalker.nextNode()) { if (highlightNodes && !treeWalker.currentNode.contains(range.endContainer)) { addHighLightedNode(treeWalker.currentNode); } if (treeWalker.currentNode == range.startContainer) { if (range.startContainer.nodeType == range.startContainer.TEXT_NODE) { const contentText = range.startContainer.splitText(range.startOffset); treeWalker.nextNode(); addHighLightedNode(contentText); } else { addHighLightedNode(range.startContainer.childNodes[range.startOffset]); } highlightNodes = true; } if (treeWalker.currentNode == range.endContainer) { if (range.endContainer.nodeType == range.endContainer.TEXT_NODE) { const contentText = range.endContainer.splitText(range.endOffset); treeWalker.nextNode(); addHighLightedNode(contentText.previousSibling); } else { addHighLightedNode(range.endContainer.childNodes[range.endOffset]); } highlightNodes = false; } } range.collapse(); } } } highlightedNodes.forEach(node => highlightNode(node)); function addHighLightedNode(node) { if (node && node.textContent.trim()) { if (node.nodeType == node.TEXT_NODE && node.parentElement.childNodes.length == 1 && node.parentElement.classList.contains(HIGHLIGHT_CLASS)) { highlightedNodes.add(node.parentElement); } else { highlightedNodes.add(node); } } } function highlightNode(node) { if (node.nodeType == node.ELEMENT_NODE) { resetHighlightedElement(node); node.classList.add(HIGHLIGHT_CLASS); node.classList.add(highlightColor); node.dataset.singlefileHighlightId = highlightId; } else if (node.parentElement) { highlightTextNode(node); } } function highlightTextNode(node) { const spanElement = document.createElement("span"); spanElement.classList.add(HIGHLIGHT_CLASS); spanElement.classList.add(highlightColor); spanElement.textContent = node.textContent; spanElement.dataset.singlefileHighlightId = highlightId; node.parentNode.replaceChild(spanElement, node); return spanElement; } } function getParents(element) { const path = []; while (element) { path.push(element); element = element.parentElement; } return path; } function formatPage(applySystemTheme, contentWidth) { if (pageCompressContent) { serializeShadowRoots(document); previousContent = document.documentElement.cloneNode(true); deserializeShadowRoots(document); } else { previousContent = getContent(false); } const shadowRoots = {}; const classesToPreserve = ["single-file-highlight", "single-file-highlight-yellow", "single-file-highlight-green", "single-file-highlight-pink", "single-file-highlight-blue"]; document.querySelectorAll(NOTE_TAGNAME).forEach(containerElement => { shadowRoots[containerElement.dataset.noteId] = containerElement.shadowRoot; const className = "singlefile-note-id-" + containerElement.dataset.noteId; containerElement.classList.add(className); classesToPreserve.push(className); }); const optionsElement = document.querySelector("script[type=\"application/json\"][" + SCRIPT_OPTIONS + "]"); const article = new Readability(document, { classesToPreserve }).parse(); const articleMetadata = Object.assign({}, article); delete articleMetadata.content; delete articleMetadata.textContent; for (const key in articleMetadata) { if (articleMetadata[key] == null) { delete articleMetadata[key]; } } removedElements = []; removedElementIndex = 0; document.body.innerHTML = ""; const domParser = new DOMParser(); const doc = domParser.parseFromString(article.content, "text/html"); const contentEditable = document.body.contentEditable; document.documentElement.replaceChild(doc.body, document.body); if (optionsElement) { document.body.appendChild(optionsElement); } document.querySelectorAll(NOTE_TAGNAME).forEach(containerElement => { const noteId = (Array.from(containerElement.classList).find(className => /singlefile-note-id-\d+/.test(className))).split("singlefile-note-id-")[1]; containerElement.classList.remove("singlefile-note-id-" + noteId); containerElement.dataset.noteId = noteId; if (!containerElement.shadowRoot) { containerElement.attachShadow({ mode: "open" }); containerElement.shadowRoot.appendChild(shadowRoots[noteId]); } }); document.querySelectorAll(NOTE_TAGNAME).forEach(containerElement => shadowRoots[containerElement.dataset.noteId].childNodes.forEach(node => containerElement.shadowRoot.appendChild(node))); document.body.contentEditable = contentEditable; document.head.querySelectorAll("style").forEach(styleElement => styleElement.remove()); const styleElement = document.createElement("style"); styleElement.textContent = getStyleFormattedPage(contentWidth); document.head.appendChild(styleElement); document.body.classList.add("moz-reader-content"); document.body.classList.add("content-width6"); document.body.classList.add("reader-show-element"); document.body.classList.add("sans-serif"); document.body.classList.add("container"); document.body.classList.add("line-height4"); const prefersColorSchemeDark = matchMedia("(prefers-color-scheme: dark)"); if (applySystemTheme && prefersColorSchemeDark && prefersColorSchemeDark.matches) { document.body.classList.add("dark"); } document.body.style.setProperty("display", "block"); document.body.style.setProperty("padding", "24px"); const titleElement = document.createElement("h1"); titleElement.classList.add("reader-title"); titleElement.textContent = article.title; document.body.insertBefore(titleElement, document.body.firstChild); const existingMetaDataElement = document.querySelector("script[id=singlefile-readability-metadata]"); if (existingMetaDataElement) { existingMetaDataElement.remove(); } const metaDataElement = document.createElement("script"); metaDataElement.type = "application/json"; metaDataElement.id = "singlefile-readability-metadata"; metaDataElement.textContent = JSON.stringify(articleMetadata, null, 2); document.head.appendChild(metaDataElement); document.querySelectorAll("a[href]").forEach(element => { const href = element.getAttribute("href").trim(); if (href.startsWith(document.baseURI + "#")) { element.setAttribute("href", href.substring(document.baseURI.length)); } }); insertHighlightStylesheet(document); maskPageElement = getMaskElement(PAGE_MASK_CLASS, PAGE_MASK_CONTAINER_CLASS); maskNoteElement = getMaskElement(NOTE_MASK_CLASS); reflowNotes(); onUpdate(false); } async function cancelFormatPage() { if (previousContent) { const contentEditable = document.body.contentEditable; if (pageCompressContent) { document.replaceChild(previousContent, document.documentElement); deserializeShadowRoots(document); await initPage(); } else { await init({ content: previousContent }, { reset: true }); } document.body.contentEditable = contentEditable; onUpdate(false); previousContent = null; } } function insertHighlightStylesheet(doc) { if (!doc.querySelector("." + HIGHLIGHTS_STYLESHEET_CLASS)) { const styleheetHighlights = getStyleElement(HIGHLIGHTS_WEB_STYLESHEET); styleheetHighlights.classList.add(HIGHLIGHTS_STYLESHEET_CLASS); doc.documentElement.appendChild(styleheetHighlights); } } function getContent(compressHTML) { unhighlightCutElement(); serializeShadowRoots(document); singlefile.helper.markInvalidNesting(document); const doc = document.cloneNode(true); disableHighlight(doc); resetSelectedElements(doc); deserializeShadowRoots(doc); deserializeShadowRoots(document); doc.documentElement.classList.remove(CUT_MODE_CLASS); doc.querySelectorAll("[" + DISABLED_NOSCRIPT_ATTRIBUTE_NAME + "]").forEach(element => { element.textContent = element.getAttribute(DISABLED_NOSCRIPT_ATTRIBUTE_NAME); element.removeAttribute(DISABLED_NOSCRIPT_ATTRIBUTE_NAME); }); doc.querySelectorAll("." + MASK_CLASS + ", " + singlefile.helper.INFOBAR_TAGNAME + ", ." + REMOVED_CONTENT_CLASS).forEach(element => element.remove()); if (includeInfobar) { const options = singlefile.helper.extractInfobarData(doc); options.openInfobar = openInfobar; options.infobarPositionAbsolute = infobarPositionAbsolute; options.infobarPositionTop = infobarPositionTop; options.infobarPositionRight = infobarPositionRight; options.infobarPositionBottom = infobarPositionBottom; options.infobarPositionLeft = infobarPositionLeft; singlefile.helper.appendInfobar(doc, options); } doc.querySelectorAll("." + HIGHLIGHT_CLASS).forEach(noteElement => noteElement.classList.remove(HIGHLIGHT_HIDDEN_CLASS)); doc.querySelectorAll(`template[${SHADOWROOT_ATTRIBUTE_NAME}]`).forEach(templateElement => { const noteElement = templateElement.querySelector("." + NOTE_CLASS); if (noteElement) { noteElement.classList.remove(NOTE_HIDDEN_CLASS); } const mainElement = templateElement.querySelector("textarea"); if (mainElement) { mainElement.textContent = mainElement.value; } }); doc.querySelectorAll("iframe").forEach(element => { const pointerEvents = "pointer-events"; element.style.setProperty(pointerEvents, element.style.getPropertyValue("-sf-" + pointerEvents), element.style.getPropertyPriority("-sf-" + pointerEvents)); element.style.removeProperty("-sf-" + pointerEvents); }); doc.body.removeAttribute("contentEditable"); if (pageCompressContent) { const pageFilename = pageResources .filter(resource => resource.filename.endsWith("index.html")) .sort((resourceLeft, resourceRight) => resourceLeft.filename.length - resourceRight.filename.length)[0].filename; const resources = pageResources.filter(resource => resource.parentResources.includes(pageFilename)); doc.querySelectorAll("[src]").forEach(element => resources.forEach(resource => { if (element.src == resource.content) { element.src = resource.name; } })); let content = singlefile.helper.serialize(doc, compressHTML); resources.sort((resourceLeft, resourceRight) => resourceRight.content.length - resourceLeft.content.length); resources.forEach(resource => content = content.replaceAll(resource.content, resource.name)); return content + ""; } else { return singlefile.helper.serialize(doc, compressHTML) + ""; } } function onUpdate(saved) { window.parent.postMessage(JSON.stringify({ "method": "onUpdate", saved }), "*"); } function waitResourcesLoad() { return new Promise(resolve => { let counterMutations = 0; const done = () => { observer.disconnect(); resolve(); }; let timeoutInit = setTimeout(done, 100); const observer = new MutationObserver(() => { if (counterMutations < 20) { counterMutations++; clearTimeout(timeoutInit); timeoutInit = setTimeout(done, 100); } else { done(); } }); observer.observe(document, { subtree: true, childList: true, attributes: true }); }); } function reflowNotes() { document.querySelectorAll(NOTE_TAGNAME).forEach(containerElement => { const noteElement = containerElement.shadowRoot.querySelector("." + NOTE_CLASS); const noteBoundingRect = noteElement.getBoundingClientRect(); const anchorElement = getAnchorElement(containerElement); const anchorBoundingRect = anchorElement.getBoundingClientRect(); const maxX = anchorBoundingRect.x + Math.max(0, anchorBoundingRect.width - noteBoundingRect.width); const minX = anchorBoundingRect.x; const maxY = anchorBoundingRect.y + Math.max(0, anchorBoundingRect.height - NOTE_HEADER_HEIGHT); const minY = anchorBoundingRect.y; let left = parseInt(noteElement.style.getPropertyValue("left")); let top = parseInt(noteElement.style.getPropertyValue("top")); if (noteBoundingRect.x > maxX) { left -= noteBoundingRect.x - maxX; } if (noteBoundingRect.x < minX) { left += minX - noteBoundingRect.x; } if (noteBoundingRect.y > maxY) { top -= noteBoundingRect.y - maxY; } if (noteBoundingRect.y < minY) { top += minY - noteBoundingRect.y; } noteElement.style.setProperty("position", "absolute"); noteElement.style.setProperty("left", left + "px"); noteElement.style.setProperty("top", top + "px"); }); } function resetHighlightedElement(element) { element.classList.remove(HIGHLIGHT_CLASS); element.classList.remove("single-file-highlight-yellow"); element.classList.remove("single-file-highlight-pink"); element.classList.remove("single-file-highlight-blue"); element.classList.remove("single-file-highlight-green"); delete element.dataset.singlefileHighlightId; } function serializeShadowRoots(node) { node.querySelectorAll("*").forEach(element => { const shadowRoot = getShadowRoot(element); if (shadowRoot) { serializeShadowRoots(shadowRoot); const templateElement = document.createElement("template"); templateElement.setAttribute(SHADOWROOT_ATTRIBUTE_NAME, "open"); Array.from(shadowRoot.childNodes).forEach(childNode => templateElement.appendChild(childNode)); element.appendChild(templateElement); } }); } function deserializeShadowRoots(node) { node.querySelectorAll(`template[${SHADOWROOT_ATTRIBUTE_NAME}]`).forEach(element => { if (element.parentElement) { let shadowRoot = getShadowRoot(element.parentElement); if (shadowRoot) { Array.from(element.childNodes).forEach(node => shadowRoot.appendChild(node)); element.remove(); } else { try { shadowRoot = element.parentElement.attachShadow({ mode: "open" }); const contentDocument = (new DOMParser()).parseFromString(element.innerHTML, "text/html"); Array.from(contentDocument.head.childNodes).forEach(node => shadowRoot.appendChild(node)); Array.from(contentDocument.body.childNodes).forEach(node => shadowRoot.appendChild(node)); // eslint-disable-next-line no-unused-vars } catch (error) { // ignored } } if (shadowRoot) { deserializeShadowRoots(shadowRoot); } } }); } function getMaskElement(className, containerClassName) { let maskElement = document.documentElement.querySelector("." + className); if (!maskElement) { maskElement = document.createElement("div"); const maskContainerElement = document.createElement("div"); if (containerClassName) { maskContainerElement.classList.add(containerClassName); } maskContainerElement.classList.add(MASK_CLASS); const firstNote = document.querySelector(NOTE_TAGNAME); if (firstNote && firstNote.parentElement == document.documentElement) { document.documentElement.insertBefore(maskContainerElement, firstNote); } else { document.documentElement.appendChild(maskContainerElement); } maskElement.classList.add(className); const maskShadow = maskContainerElement.attachShadow({ mode: "open" }); maskShadow.appendChild(getStyleElement(MASK_WEB_STYLESHEET)); maskShadow.appendChild(maskElement); return maskElement; } } function getStyleFormattedPage(contentWidth) { return ` /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this file, * You can obtain one at http://mozilla.org/MPL/2.0/. */ /* Avoid adding ID selector rules in this style sheet, since they could * inadvertently match elements in the article content. */ :root { --grey-90-a10: rgba(12, 12, 13, 0.1); --grey-90-a20: rgba(12, 12, 13, 0.2); --grey-90-a30: rgba(12, 12, 13, 0.3); --grey-90-a80: rgba(12, 12, 13, 0.8); --grey-30: #d7d7db; --blue-40: #45a1ff; --blue-40-a30: rgba(69, 161, 255, 0.3); --blue-60: #0060df; --body-padding: 64px; --font-size: 12; --content-width: ${contentWidth}em; --line-height: 1.6em; } body { --main-background: #fff; --main-foreground: #333; --font-color: #000000; --primary-color: #0B83FF; --toolbar-border: var(--grey-90-a20); --toolbar-transparent-border: transparent; --toolbar-box-shadow: var(--grey-90-a10); --toolbar-button-background: transparent; --toolbar-button-background-hover: var(--grey-90-a10); --toolbar-button-foreground-hover: var(--font-color); --toolbar-button-background-active: var(--grey-90-a20); --toolbar-button-foreground-active: var(--primary-color); --toolbar-button-border: transparent; --toolbar-button-border-hover: transparent; --toolbar-button-border-active: transparent; --tooltip-background: var(--grey-90-a80); --tooltip-foreground: white; --tooltip-border: transparent; --popup-background: white; --popup-border: rgba(0, 0, 0, 0.12); --opaque-popup-border: #e0e0e0; --popup-line: var(--grey-30); --popup-shadow: rgba(49, 49, 49, 0.3); --popup-button-background: #edecf0; --popup-button-background-hover: hsla(0,0%,70%,.4); --popup-button-foreground-hover: var(--font-color); --popup-button-background-active: hsla(240,5%,5%,.15); --selected-background: var(--blue-40-a30); --selected-border: var(--blue-40); --font-value-border: var(--grey-30); --icon-fill: #3b3b3c; --icon-disabled-fill: #8080807F; --link-foreground: var(--blue-60); --link-selected-foreground: #333; --visited-link-foreground: #b5007f; /* light colours */ } body.sepia { --main-background: #f4ecd8; --main-foreground: #5b4636; --toolbar-border: #5b4636; } body.dark { --main-background: rgb(28, 27, 34); --main-foreground: #eee; --font-color: #fff; --toolbar-border: #4a4a4b; --toolbar-box-shadow: black; --toolbar-button-background-hover: var(--grey-90-a30); --toolbar-button-background-active: var(--grey-90-a80); --tooltip-background: black; --tooltip-foreground: white; --popup-background: rgb(66,65,77); --opaque-popup-border: #434146; --popup-line: rgb(82, 82, 94); --popup-button-background: #5c5c61; --popup-button-background-active: hsla(0,0%,70%,.6); --selected-background: #3E6D9A; --font-value-border: #656468; --icon-fill: #fff; --icon-disabled-fill: #ffffff66; --link-foreground: #45a1ff; --link-selected-foreground: #fff; --visited-link-foreground: #e675fd; /* dark colours */ } body.hcm { --main-background: Canvas; --main-foreground: CanvasText; --font-color: CanvasText; --primary-color: SelectedItem; --toolbar-border: CanvasText; /* We need a true transparent but in HCM this would compute to an actual color, so select the page's background color instead: */ --toolbar-transparent-border: Canvas; --toolbar-box-shadow: Canvas; --toolbar-button-background: ButtonFace; --toolbar-button-background-hover: ButtonText; --toolbar-button-foreground-hover: ButtonFace; --toolbar-button-background-active: SelectedItem; --toolbar-button-foreground-active: SelectedItemText; --toolbar-button-border: ButtonText; --toolbar-button-border-hover: ButtonText; --toolbar-button-border-active: ButtonText; --tooltip-background: Canvas; --tooltip-foreground: CanvasText; --tooltip-border: CanvasText; --popup-background: Canvas; --popup-border: CanvasText; --opaque-popup-border: CanvasText; --popup-line: CanvasText; --popup-button-background: ButtonFace; --popup-button-background-hover: ButtonText; --popup-button-foreground-hover: ButtonFace; --popup-button-background-active: ButtonText; --selected-background: Canvas; --selected-border: SelectedItem; --font-value-border: CanvasText; --icon-fill: ButtonText; --icon-disabled-fill: GrayText; --link-foreground: LinkText; --link-selected-foreground: ActiveText; --visited-link-foreground: VisitedText; } body { margin: 0; padding: var(--body-padding); background-color: var(--main-background); color: var(--main-foreground); } body.loaded { transition: color 0.4s, background-color 0.4s; } body.dark *::-moz-selection { background-color: var(--selected-background); } a::-moz-selection { color: var(--link-selected-foreground); } body.sans-serif, body.sans-serif .remove-button { font-family: Helvetica, Arial, sans-serif; } body.serif, body.serif .remove-button { font-family: Georgia, "Times New Roman", serif; } /* Override some controls and content styles based on color scheme */ body.light > .container > .header > .domain { border-bottom-color: #333333 !important; } body.sepia > .container > .header > .domain { border-bottom-color: #5b4636 !important; } body.dark > .container > .header > .domain { border-bottom-color: #eeeeee !important; } body.light blockquote { border-inline-start: 2px solid #333333 !important; } body.sepia blockquote { border-inline-start: 2px solid #5b4636 !important; } body.dark blockquote { border-inline-start: 2px solid #eeeeee !important; } .light-button { color: #333333; background-color: #ffffff; } .dark-button { color: #eeeeee; background-color: #1c1b22; } .sepia-button { color: #5b4636; background-color: #f4ecd8; } .auto-button { text-align: center; } @media (prefers-color-scheme: dark) { .auto-button { background-color: #1c1b22; color: #eeeeee; } } @media not (prefers-color-scheme: dark) { .auto-button { background-color: #ffffff; color: #333333; } } /* Loading/error message */ .reader-message { margin-top: 40px; display: none; text-align: center; width: 100%; font-size: 0.9em; } /* Detector element to see if we're at the top of the doc or not. */ .top-anchor { position: absolute; top: 0; width: 0; height: 5px; pointer-events: none; } /* Header */ .header { text-align: start; display: none; } .domain { font-size: 0.9em; line-height: 1.48em; padding-bottom: 4px; font-family: Helvetica, Arial, sans-serif; text-decoration: none; border-bottom: 1px solid; color: var(--link-foreground); } .header > h1 { font-size: 1.6em; line-height: 1.25em; width: 100%; margin: 30px 0; padding: 0; } .header > .credits { font-size: 0.9em; line-height: 1.48em; margin: 0 0 10px; padding: 0; font-style: italic; } .header > .meta-data { font-size: 0.65em; margin: 0 0 15px; } .reader-estimated-time { text-align: match-parent; } /* Controls toolbar */ .toolbar-container { position: sticky; z-index: 2; top: 32px; height: 0; /* take up no space, so body is at the top. */ /* As a stick container, we're positioned relative to the body. Move us to * the edge of the viewport using margins, and take the width of * the body padding into account for calculating our width. */ margin-inline-start: calc(-1 * var(--body-padding)); width: max(var(--body-padding), calc((100% - var(--content-width)) / 2 + var(--body-padding))); font-size: var(--font-size); /* Needed to ensure 'em' units match, is reset for .reader-controls */ } .toolbar { padding-block: 16px; border: 1px solid var(--toolbar-border); border-radius: 6px; box-shadow: 0 2px 8px var(--toolbar-box-shadow); width: 32px; /* basic width, without padding/border */ /* padding should be 16px, except if there's not enough space for that, in * which case use half the available space for padding (=25% on each side). * The 34px here is the width + borders. We use a variable because we need * to know this size for the margin calculation. */ --inline-padding: min(16px, calc(25% - 0.25 * 34px)); padding-inline: var(--inline-padding); /* Keep a maximum of 96px distance to the body, but center once the margin * gets too small. We need to set the start margin, however... * To center, we'd want 50% of the container, but we'd subtract half our * own width (16px) and half the border (1px) and the inline padding. * When the other margin would be 96px, we want 100% - 96px - the complete * width of the actual toolbar (34px + 2 * padding) */ margin-inline-start: max(calc(50% - 17px - var(--inline-padding)), calc(100% - 96px - 34px - 2 * var(--inline-padding))); font-family: Helvetica, Arial, sans-serif; list-style: none; user-select: none; } @media (prefers-reduced-motion: no-preference) { .toolbar { transition-property: border-color, box-shadow; transition-duration: 250ms; } .toolbar .toolbar-button { transition-property: opacity; transition-duration: 250ms; } .toolbar-container.scrolled .toolbar:not(:hover, :focus-within) { border-color: var(--toolbar-transparent-border); box-shadow: 0 2px 8px transparent; } .toolbar-container.scrolled .toolbar:not(:hover, :focus-within) .toolbar-button { opacity: 0.6; } .toolbar-container.transition-location { transition-duration: 250ms; transition-property: width; } } .toolbar-container.overlaps .toolbar-button { opacity: 0.1; } .dropdown-open .toolbar { border-color: var(--toolbar-transparent-border); box-shadow: 0 2px 8px transparent; } .reader-controls { /* We use 'em's above this node to get it to the right size. However, * the UI inside the toolbar should use a fixed, smaller size. */ font-size: 11px; } button { -moz-context-properties: fill; color: var(--font-color); fill: var(--icon-fill); } button:disabled { fill: var(--icon-disabled-fill); } .toolbar button::-moz-focus-inner { border: 0; } .toolbar-button { position: relative; width: 32px; height: 32px; padding: 0; border: 1px solid var(--toolbar-button-border); border-radius: 4px; margin: 4px 0; background-color: var(--toolbar-button-background); background-size: 16px 16px; background-position: center; background-repeat: no-repeat; } .toolbar-button:hover, .toolbar-button:focus-visible { background-color: var(--toolbar-button-background-hover); border-color: var(--toolbar-button-border-hover); fill: var(--toolbar-button-foreground-hover); } .open .toolbar-button, .toolbar-button:hover:active { background-color: var(--toolbar-button-background-active); border-color: var(--toolbar-button-border-active); color: var(--toolbar-button-foreground-active); fill: var(--toolbar-button-foreground-active); } .hover-label { position: absolute; top: 4px; inset-inline-start: 36px; line-height: 16px; white-space: pre; /* make sure we don't wrap */ background-color: var(--tooltip-background); color: var(--tooltip-foreground); padding: 4px 8px; border: 1px solid var(--tooltip-border); border-radius: 2px; visibility: hidden; pointer-events: none; /* Put above .dropdown .dropdown-popup, which has z-index: 1000. */ z-index: 1001; } /* Show the hover tooltip on non-dropdown buttons. */ .toolbar-button:not(.dropdown-toggle):hover > .hover-label, .toolbar-button:not(.dropdown-toggle):focus-visible > .hover-label, /* Show the hover tooltip for dropdown buttons unless its dropdown is open. */ :not(.open) > li > .dropdown-toggle:hover > .hover-label, :not(.open) > li > .dropdown-toggle:focus-visible > .hover-label { visibility: visible; } .dropdown { text-align: center; list-style: none; margin: 0; padding: 0; position: relative; } .dropdown li { margin: 0; padding: 0; } /* Popup */ .dropdown .dropdown-popup { text-align: start; position: absolute; inset-inline-start: 40px; z-index: 1000; background-color: var(--popup-background); visibility: hidden; border-radius: 4px; border: 1px solid var(--popup-border); box-shadow: 0 0 10px 0 var(--popup-shadow); top: 0; } .open > .dropdown-popup { visibility: visible; } .dropdown-arrow { position: absolute; height: 24px; width: 16px; inset-inline-start: -16px; background-image: url("chrome://global/skin/reader/RM-Type-Controls-Arrow.svg"); display: block; -moz-context-properties: fill, stroke; fill: var(--popup-background); stroke: var(--opaque-popup-border); pointer-events: none; } .dropdown-arrow:dir(rtl) { transform: scaleX(-1); } /* Align the style dropdown arrow (narrate does its own) */ .style-dropdown .dropdown-arrow { top: 7px; } /* Font style popup */ .radio-button { /* We visually hide these, but we keep them around so they can be focused * and changed by interacting with them via the label, or the keyboard, or * assistive technology. */ opacity: 0; pointer-events: none; position: absolute; } .radiorow, .buttonrow { display: flex; align-content: center; justify-content: center; } .content-width-value, .font-size-value, .line-height-value { box-sizing: border-box; width: 36px; height: 20px; line-height: 20px; display: flex; justify-content: center; align-content: center; margin: auto; border-radius: 10px; border: 1px solid var(--font-value-border); background-color: var(--popup-button-background); } .buttonrow > button { border: 0; height: 60px; width: 90px; background-color: transparent; background-repeat: no-repeat; background-position: center; } .buttonrow > button:enabled:hover, .buttonrow > button:enabled:focus-visible { background-color: var(--popup-button-background-hover); color: var(--popup-button-foreground-hover); fill: var(--popup-button-foreground-hover); } .buttonrow > button:enabled:hover:active { background-color: var(--popup-button-background-active); } .radiorow:not(:last-child), .buttonrow:not(:last-child) { border-bottom: 1px solid var(--popup-line); } body.hcm .buttonrow.line-height-buttons { /* On HCM the .color-scheme-buttons row is hidden, so remove the border from the row above it */ border-bottom: none; } .radiorow > label { position: relative; box-sizing: border-box; border-radius: 2px; border: 2px solid var(--popup-border); } .radiorow > label[checked] { border-color: var(--selected-border); } /* For the hover style, we draw a line under the item by means of a * pseudo-element. Because these items are variable height, and * because their contents are variable height, position it absolutely, * and give it a width of 100% (the content width) + 4px for the 2 * 2px * border width. */ .radiorow > input[type=radio]:focus-visible + label::after, .radiorow > label:hover::after { content: ""; display: block; border-bottom: 2px solid var(--selected-border); width: calc(100% + 4px); position: absolute; /* to skip the 2 * 2px border + 2px spacing. */ bottom: -6px; /* Match the start of the 2px border of the element: */ inset-inline-start: -2px; } .font-type-buttons > label { height: 64px; width: 105px; /* Slightly more space between these items. */ margin: 10px; /* Center the Sans-serif / Serif labels */ text-align: center; background-size: 63px 39px; background-repeat: no-repeat; background-position: center 18px; background-color: var(--popup-button-background); fill: currentColor; -moz-context-properties: fill; /* This mostly matches baselines, but because of differences in font * baselines between serif and sans-serif, this isn't always enough. */ line-height: 1; padding-top: 2px; } .font-type-buttons > label[checked] { background-color: var(--selected-background); } .sans-serif-button { font-family: Helvetica, Arial, sans-serif; background-image: url("chrome://global/skin/reader/RM-Sans-Serif.svg"); } /* Tweak padding to match the baseline on mac */ :root[platform=macosx] .sans-serif-button { padding-top: 3px; } .serif-button { font-family: Georgia, "Times New Roman", serif; background-image: url("chrome://global/skin/reader/RM-Serif.svg"); } body.hcm .color-scheme-buttons { /* Disallow selecting themes when HCM is on. */ display: none; } .color-scheme-buttons > label { padding: 12px; height: 34px; font-size: 12px; /* Center the labels horizontally as well as vertically */ display: inline-flex; align-items: center; justify-content: center; /* We want 10px between items, but there's no margin collapsing in flexbox. */ margin: 10px 5px; } .color-scheme-buttons > input:first-child + label { margin-inline-start: 10px; } .color-scheme-buttons > label:last-child { margin-inline-end: 10px; } /* Toolbar icons */ .close-button { background-image: url("chrome://global/skin/icons/close.svg"); } .style-button { background-image: url("chrome://global/skin/reader/RM-Type-Controls-24x24.svg"); } .minus-button { background-size: 18px 18px; background-image: url("chrome://global/skin/reader/RM-Minus-24x24.svg"); } .plus-button { background-size: 18px 18px; background-image: url("chrome://global/skin/reader/RM-Plus-24x24.svg"); } .content-width-minus-button { background-size: 42px 16px; background-image: url("chrome://global/skin/reader/RM-Content-Width-Minus-42x16.svg"); } .content-width-plus-button { background-size: 44px 16px; background-image: url("chrome://global/skin/reader/RM-Content-Width-Plus-44x16.svg"); } .line-height-minus-button { background-size: 34px 14px; background-image: url("chrome://global/skin/reader/RM-Line-Height-Minus-38x14.svg"); } .line-height-plus-button { background-size: 34px 24px; background-image: url("chrome://global/skin/reader/RM-Line-Height-Plus-38x24.svg"); } /* Mirror the line height buttons if the article is RTL. */ .reader-controls[articledir="rtl"] .line-height-minus-button, .reader-controls[articledir="rtl"] .line-height-plus-button { transform: scaleX(-1); } @media print { .toolbar { display: none !important; } } /* Article content */ /* Note that any class names from the original article that we want to match on * must be added to CLASSES_TO_PRESERVE in ReaderMode.jsm, so that * Readability.js doesn't strip them out */ .container { margin: 0 auto; font-size: var(--font-size); max-width: var(--content-width); line-height: var(--line-height); } pre { font-family: inherit; } .moz-reader-content { display: none; font-size: 1em; } @media print { .moz-reader-content p, .moz-reader-content code, .moz-reader-content pre, .moz-reader-content blockquote, .moz-reader-content ul, .moz-reader-content ol, .moz-reader-content li, .moz-reader-content figure, .moz-reader-content .wp-caption { margin: 0 0 10px !important; padding: 0 !important; } } .moz-reader-content h1, .moz-reader-content h2, .moz-reader-content h3 { font-weight: bold; } .moz-reader-content h1 { font-size: 1.6em; line-height: 1.25em; } .moz-reader-content h2 { font-size: 1.2em; line-height: 1.51em; } .moz-reader-content h3 { font-size: 1em; line-height: 1.66em; } .moz-reader-content a:link { text-decoration: underline; font-weight: normal; } .moz-reader-content a:link, .moz-reader-content a:link:hover, .moz-reader-content a:link:active { color: var(--link-foreground); } .moz-reader-content a:visited { color: var(--visited-link-foreground); } .moz-reader-content * { max-width: 100%; height: auto; } .moz-reader-content p, .moz-reader-content p, .moz-reader-content code, .moz-reader-content pre, .moz-reader-content blockquote, .moz-reader-content ul, .moz-reader-content ol, .moz-reader-content li, .moz-reader-content figure, .moz-reader-content .wp-caption { margin: -10px -10px 20px; padding: 10px; border-radius: 5px; } .moz-reader-content li { margin-bottom: 0; } .moz-reader-content li > ul, .moz-reader-content li > ol { margin-bottom: -10px; } .moz-reader-content p > img:only-child, .moz-reader-content p > a:only-child > img:only-child, .moz-reader-content .wp-caption img, .moz-reader-content figure img { display: block; } .moz-reader-content img[moz-reader-center] { margin-inline: auto; } .moz-reader-content .caption, .moz-reader-content .wp-caption-text .moz-reader-content figcaption { font-size: 0.9em; line-height: 1.48em; font-style: italic; } .moz-reader-content pre { white-space: pre-wrap; } .moz-reader-content blockquote { padding: 0; padding-inline-start: 16px; } .moz-reader-content ul, .moz-reader-content ol { padding: 0; } .moz-reader-content ul { padding-inline-start: 30px; list-style: disc; } .moz-reader-content ol { padding-inline-start: 30px; } table, th, td { border: 1px solid currentColor; border-collapse: collapse; padding: 6px; vertical-align: top; } table { margin: 5px; } /* Visually hide (but don't display: none) screen reader elements */ .moz-reader-content .visually-hidden, .moz-reader-content .visuallyhidden, .moz-reader-content .sr-only { display: inline-block; width: 1px; height: 1px; margin: -1px; overflow: hidden; padding: 0; border-width: 0; } /* Hide elements with common "hidden" class names */ .moz-reader-content .hidden, .moz-reader-content .invisible { display: none; } /* Enforce wordpress and similar emoji/smileys aren't sized to be full-width, * see bug 1399616 for context. */ .moz-reader-content img.wp-smiley, .moz-reader-content img.emoji { display: inline-block; border-width: 0; /* height: auto is implied from '.moz-reader-content *' rule. */ width: 1em; margin: 0 .07em; padding: 0; } .reader-show-element { display: initial; } /* Provide extra spacing for images that may be aided with accompanying element such as
*/ .moz-reader-block-img:not(:last-child) { margin-block-end: 12px; } .moz-reader-wide-table { overflow-x: auto; display: block; } pre code { background-color: var(--main-background); border: 1px solid var(--font-color); display: block; overflow: auto; }`; } function getEmbedScript() { return minifyText(`(() => { document.currentScript.remove(); const processNode = node => { node.querySelectorAll("template[${SHADOWROOT_ATTRIBUTE_NAME}]").forEach(element=>{ let shadowRoot = getShadowRoot(element.parentElement); if (!shadowRoot) { try { shadowRoot = element.parentElement.attachShadow({mode:element.getAttribute("${SHADOWROOT_ATTRIBUTE_NAME}")}); shadowRoot.innerHTML = element.innerHTML; element.remove(); } catch (error) {} if (shadowRoot) { processNode(shadowRoot); } } }) }; const FORBIDDEN_TAG_NAMES = ${JSON.stringify(FORBIDDEN_TAG_NAMES)}; const NOTE_TAGNAME = ${JSON.stringify(NOTE_TAGNAME)}; const NOTE_CLASS = ${JSON.stringify(NOTE_CLASS)}; const NOTE_ANCHORED_CLASS = ${JSON.stringify(NOTE_ANCHORED_CLASS)}; const NOTE_SELECTED_CLASS = ${JSON.stringify(NOTE_SELECTED_CLASS)}; const NOTE_MOVING_CLASS = ${JSON.stringify(NOTE_MOVING_CLASS)}; const NOTE_MASK_MOVING_CLASS = ${JSON.stringify(NOTE_MASK_MOVING_CLASS)}; const MASK_CLASS = ${JSON.stringify(MASK_CLASS)}; const HIGHLIGHT_CLASS = ${JSON.stringify(HIGHLIGHT_CLASS)}; const NOTES_WEB_STYLESHEET = ${JSON.stringify(NOTES_WEB_STYLESHEET)}; const MASK_WEB_STYLESHEET = ${JSON.stringify(MASK_WEB_STYLESHEET)}; const NOTE_HEADER_HEIGHT = ${JSON.stringify(NOTE_HEADER_HEIGHT)}; const PAGE_MASK_ACTIVE_CLASS = ${JSON.stringify(PAGE_MASK_ACTIVE_CLASS)}; const REMOVED_CONTENT_CLASS = ${JSON.stringify(REMOVED_CONTENT_CLASS)}; const NESTING_TRACK_ID_ATTRIBUTE_NAME = ${JSON.stringify(singlefile.helper.NESTING_TRACK_ID_ATTRIBUTE_NAME)}; const reflowNotes = ${minifyText(reflowNotes.toString())}; const addNoteRef = ${minifyText(addNoteRef.toString())}; const deleteNoteRef = ${minifyText(deleteNoteRef.toString())}; const getNoteRefs = ${minifyText(getNoteRefs.toString())}; const setNoteRefs = ${minifyText(setNoteRefs.toString())}; const getAnchorElement = ${minifyText(getAnchorElement.toString())}; const getMaskElement = ${minifyText(getMaskElement.toString())}; const getStyleElement = ${minifyText(getStyleElement.toString())}; const attachNoteListeners = ${minifyText(attachNoteListeners.toString())}; const anchorNote = ${minifyText(anchorNote.toString())}; const getPosition = ${minifyText(getPosition.toString())}; const onMouseUp = ${minifyText(onMouseUp.toString())}; const getShadowRoot = ${minifyText(getShadowRoot.toString())}; const waitResourcesLoad = ${minifyText(waitResourcesLoad.toString())}; const maskNoteElement = getMaskElement(${JSON.stringify(NOTE_MASK_CLASS)}); const maskPageElement = getMaskElement(${JSON.stringify(PAGE_MASK_CLASS)}, ${JSON.stringify(PAGE_MASK_CONTAINER_CLASS)}); let selectedNote, highlightSelectionMode, removeHighlightMode, resizingNoteMode, movingNoteMode, collapseNoteTimeout, cuttingMode, cuttingOuterMode; window.onresize = reflowNotes; window.onUpdate = () => {}; document.documentElement.onmouseup = document.documentElement.ontouchend = onMouseUp; processNode(document); reflowNotes(); document.querySelectorAll(${JSON.stringify(NOTE_TAGNAME)}).forEach(noteElement => attachNoteListeners(noteElement)); if (document.documentElement.dataset && document.documentElement.dataset.sfz !== undefined) { waitResourcesLoad().then(reflowNotes); } const trackIds = {}; document.querySelectorAll("[" + NESTING_TRACK_ID_ATTRIBUTE_NAME + "]").forEach(element => trackIds[element.getAttribute(NESTING_TRACK_ID_ATTRIBUTE_NAME)] = element); Object.keys(trackIds).forEach(id => { const element = trackIds[id]; const idParts = id.split("."); if (idParts.length > 1) { const parentId = idParts.slice(0, -1).join("."); const expectedParent = trackIds[parentId]; if (expectedParent && element.parentElement !== expectedParent) { expectedParent.appendChild(element); } } }); document.querySelectorAll("[" + NESTING_TRACK_ID_ATTRIBUTE_NAME + "]").forEach(element => element.removeAttribute(NESTING_TRACK_ID_ATTRIBUTE_NAME)); })()`); } function getStyleElement(stylesheet) { const linkElement = document.createElement("style"); linkElement.textContent = stylesheet; return linkElement; } function getAnchorElement(containerElement) { return document.querySelector("[data-single-file-note-refs~=\"" + containerElement.dataset.noteId + "\"]") || document.documentElement; } function addNoteRef(anchorElement, noteId) { const noteRefs = getNoteRefs(anchorElement); noteRefs.push(noteId); setNoteRefs(anchorElement, noteRefs); } function deleteNoteRef(containerElement, noteId) { const anchorElement = getAnchorElement(containerElement); const noteRefs = getNoteRefs(anchorElement).filter(noteRefs => noteRefs != noteId); if (noteRefs.length) { setNoteRefs(anchorElement, noteRefs); } else { delete anchorElement.dataset.singleFileNoteRefs; } } function getNoteRefs(anchorElement) { return anchorElement.dataset.singleFileNoteRefs ? anchorElement.dataset.singleFileNoteRefs.split(" ") : []; } function setNoteRefs(anchorElement, noteRefs) { anchorElement.dataset.singleFileNoteRefs = noteRefs.join(" "); } function minifyText(text) { return text.replace(/[\n\t\s]+/g, " "); } function isAncestor(element, otherElement) { return otherElement.parentElement && (element == otherElement.parentElement || isAncestor(element, otherElement.parentElement)); } function getShadowRoot(element) { const chrome = window.chrome; if (element.openOrClosedShadowRoot) { return element.openOrClosedShadowRoot; } else if (chrome && chrome.dom && chrome.dom.openOrClosedShadowRoot) { try { return chrome.dom.openOrClosedShadowRoot(element); /* eslint-disable-next-line no-unused-vars */ } catch (error) { return element.shadowRoot; } } else { return element.shadowRoot; } } function detectSavedPage(document) { const firstDocumentChild = document.documentElement.firstChild; return firstDocumentChild.nodeType == Node.COMMENT_NODE && (firstDocumentChild.textContent.includes(COMMENT_HEADER) || firstDocumentChild.textContent.includes(COMMENT_HEADER_LEGACY)); } })(typeof globalThis == "object" ? globalThis : window); })();