/* Copyright 2012 Mozilla Foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import { isPdfFile, PDFDataRangeTransport } from "pdfjs-lib"; import { BaseExternalServices } from "./external_services.js"; import { BasePreferences } from "./preferences.js"; import { DEFAULT_SCALE_VALUE } from "./ui_utils.js"; import { L10n } from "./l10n.js"; if (typeof PDFJSDev === "undefined" || !PDFJSDev.test("MOZCENTRAL")) { throw new Error( 'Module "./firefoxcom.js" shall not be used outside MOZCENTRAL builds.' ); } let viewerApp = { initialized: false }; function initCom(app) { viewerApp = app; } class FirefoxCom { /** * Creates an event that the extension is listening for and will * asynchronously respond to. * @param {string} action - The action to trigger. * @param {Object|string} [data] - The data to send. * @returns {Promise} A promise that is resolved with the response data. */ static requestAsync(action, data) { return new Promise(resolve => { this.request(action, data, resolve); }); } /** * Creates an event that the extension is listening for and will, optionally, * asynchronously respond to. * @param {string} action - The action to trigger. * @param {Object|string} [data] - The data to send. */ static request(action, data, callback = null) { const request = document.createTextNode(""); if (callback) { request.addEventListener( "pdf.js.response", event => { const response = event.detail.response; event.target.remove(); callback(response); }, { once: true } ); } document.documentElement.append(request); const sender = new CustomEvent("pdf.js.message", { bubbles: true, cancelable: false, detail: { action, data, responseExpected: !!callback, }, }); request.dispatchEvent(sender); } } class DownloadManager { #openBlobUrls = new WeakMap(); downloadUrl(url, filename, options = {}) { FirefoxCom.request("download", { originalUrl: url, filename, options, }); } downloadData(data, filename, contentType) { const blobUrl = URL.createObjectURL( new Blob([data], { type: contentType }) ); FirefoxCom.request("download", { blobUrl, originalUrl: blobUrl, filename, isAttachment: true, }); } /** * @returns {boolean} Indicating if the data was opened. */ openOrDownloadData(data, filename, dest = null) { const isPdfData = isPdfFile(filename); const contentType = isPdfData ? "application/pdf" : ""; if (isPdfData) { let blobUrl = this.#openBlobUrls.get(data); if (!blobUrl) { blobUrl = URL.createObjectURL(new Blob([data], { type: contentType })); this.#openBlobUrls.set(data, blobUrl); } // Let Firefox's content handler catch the URL and display the PDF. // NOTE: This cannot use a query string for the filename, see // https://bugzilla.mozilla.org/show_bug.cgi?id=1632644#c5 let viewerUrl = blobUrl + "#filename=" + encodeURIComponent(filename); if (dest) { viewerUrl += `&filedest=${escape(dest)}`; } try { window.open(viewerUrl); return true; } catch (ex) { console.error(`openOrDownloadData: ${ex}`); // Release the `blobUrl`, since opening it failed, and fallback to // downloading the PDF file. URL.revokeObjectURL(blobUrl); this.#openBlobUrls.delete(data); } } this.downloadData(data, filename, contentType); return false; } download(blob, url, filename, options = {}) { const blobUrl = URL.createObjectURL(blob); FirefoxCom.request("download", { blobUrl, originalUrl: url, filename, options, }); } } class Preferences extends BasePreferences { async _readFromStorage(prefObj) { return FirefoxCom.requestAsync("getPreferences", prefObj); } } (function listenFindEvents() { const events = [ "find", "findagain", "findhighlightallchange", "findcasesensitivitychange", "findentirewordchange", "findbarclose", "finddiacriticmatchingchange", ]; const findLen = "find".length; const handleEvent = function ({ type, detail }) { if (!viewerApp.initialized) { return; } if (type === "findbarclose") { viewerApp.eventBus.dispatch(type, { source: window }); return; } viewerApp.eventBus.dispatch("find", { source: window, type: type.substring(findLen), query: detail.query, caseSensitive: !!detail.caseSensitive, entireWord: !!detail.entireWord, highlightAll: !!detail.highlightAll, findPrevious: !!detail.findPrevious, matchDiacritics: !!detail.matchDiacritics, }); }; for (const event of events) { window.addEventListener(event, handleEvent); } })(); (function listenZoomEvents() { const events = ["zoomin", "zoomout", "zoomreset"]; const handleEvent = function ({ type, detail }) { if (!viewerApp.initialized) { return; } // Avoid attempting to needlessly reset the zoom level *twice* in a row, // when using the `Ctrl + 0` keyboard shortcut. if ( type === "zoomreset" && viewerApp.pdfViewer.currentScaleValue === DEFAULT_SCALE_VALUE ) { return; } viewerApp.eventBus.dispatch(type, { source: window }); }; for (const event of events) { window.addEventListener(event, handleEvent); } })(); (function listenSaveEvent() { const handleEvent = function ({ type, detail }) { if (!viewerApp.initialized) { return; } viewerApp.eventBus.dispatch("download", { source: window }); }; window.addEventListener("save", handleEvent); })(); (function listenEditingEvent() { const handleEvent = function ({ detail }) { if (!viewerApp.initialized) { return; } viewerApp.eventBus.dispatch("editingaction", { source: window, name: detail.name, }); }; window.addEventListener("editingaction", handleEvent); })(); if (PDFJSDev.test("GECKOVIEW")) { (function listenQueryEvents() { window.addEventListener("pdf.js.query", async ({ detail: { queryId } }) => { let result = null; if (viewerApp.initialized && queryId === "canDownloadInsteadOfPrint") { result = false; const { pdfDocument, pdfViewer } = viewerApp; if (pdfDocument) { try { const hasUnchangedAnnotations = pdfDocument.annotationStorage.size === 0; // WillPrint is called just before printing the document and could // lead to have modified annotations. const hasWillPrint = pdfViewer.enableScripting && !!(await pdfDocument.getJSActions())?.WillPrint; const hasUnchangedOptionalContent = ( await pdfViewer.optionalContentConfigPromise ).hasInitialVisibility; result = hasUnchangedAnnotations && !hasWillPrint && hasUnchangedOptionalContent; } catch { console.warn("Unable to check if the document can be downloaded."); } } } window.dispatchEvent( new CustomEvent("pdf.js.query.answer", { bubbles: true, cancelable: false, detail: { queryId, value: result, }, }) ); }); })(); } class FirefoxComDataRangeTransport extends PDFDataRangeTransport { requestDataRange(begin, end) { FirefoxCom.request("requestDataRange", { begin, end }); } // NOTE: This method is currently not invoked in the Firefox PDF Viewer. abort() { FirefoxCom.request("abortLoading", null); } } class FirefoxScripting { static async createSandbox(data) { const success = await FirefoxCom.requestAsync("createSandbox", data); if (!success) { throw new Error("Cannot create sandbox."); } } static async dispatchEventInSandbox(event) { FirefoxCom.request("dispatchEventInSandbox", event); } static async destroySandbox() { FirefoxCom.request("destroySandbox", null); } } class MLManager { guess(data) { return FirefoxCom.requestAsync("mlGuess", data); } } class ExternalServices extends BaseExternalServices { updateFindControlState(data) { FirefoxCom.request("updateFindControlState", data); } updateFindMatchesCount(data) { FirefoxCom.request("updateFindMatchesCount", data); } initPassiveLoading(callbacks) { let pdfDataRangeTransport; window.addEventListener("message", function windowMessage(e) { if (e.source !== null) { // The message MUST originate from Chrome code. console.warn("Rejected untrusted message from " + e.origin); return; } const args = e.data; if (typeof args !== "object" || !("pdfjsLoadAction" in args)) { return; } switch (args.pdfjsLoadAction) { case "supportsRangedLoading": if (args.done && !args.data) { callbacks.onError(); break; } pdfDataRangeTransport = new FirefoxComDataRangeTransport( args.length, args.data, args.done, args.filename ); callbacks.onOpenWithTransport(pdfDataRangeTransport); break; case "range": pdfDataRangeTransport.onDataRange(args.begin, args.chunk); break; case "rangeProgress": pdfDataRangeTransport.onDataProgress(args.loaded); break; case "progressiveRead": pdfDataRangeTransport.onDataProgressiveRead(args.chunk); // Don't forget to report loading progress as well, since otherwise // the loadingBar won't update when `disableRange=true` is set. pdfDataRangeTransport.onDataProgress(args.loaded, args.total); break; case "progressiveDone": pdfDataRangeTransport?.onDataProgressiveDone(); break; case "progress": callbacks.onProgress(args.loaded, args.total); break; case "complete": if (!args.data) { callbacks.onError(args.errorCode); break; } callbacks.onOpenWithData(args.data, args.filename); break; } }); FirefoxCom.request("initPassiveLoading", null); } reportTelemetry(data) { FirefoxCom.request("reportTelemetry", data); } updateEditorStates(data) { FirefoxCom.request("updateEditorStates", data); } async createL10n() { const [localeProperties] = await Promise.all([ FirefoxCom.requestAsync("getLocaleProperties", null), document.l10n.ready, ]); return new L10n(localeProperties, document.l10n); } createScripting() { return FirefoxScripting; } async getNimbusExperimentData() { if (!PDFJSDev.test("GECKOVIEW")) { return null; } const nimbusData = await FirefoxCom.requestAsync( "getNimbusExperimentData", null ); return nimbusData && JSON.parse(nimbusData); } } export { DownloadManager, ExternalServices, initCom, MLManager, Preferences };