| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450 | /* 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 "../extensions/firefox/tools/l10n.js";import { DefaultExternalServices, PDFViewerApplication } from "./app.js";import { isPdfFile, PDFDataRangeTransport, shadow } from "pdfjs-lib";import { BasePreferences } from "./preferences.js";import { DEFAULT_SCALE_VALUE } from "./ui_utils.js";import { getL10nFallback } from "./l10n_utils.js";if (typeof PDFJSDev === "undefined" || !PDFJSDev.test("MOZCENTRAL")) {  throw new Error(    'Module "./firefoxcom.js" shall not be used outside MOZCENTRAL builds.'  );}class FirefoxCom {  /**   * Creates an event that the extension is listening for and will   * synchronously respond to.   * NOTE: It is recommended to use requestAsync() instead since one day we may   *       not be able to synchronously reply.   * @param {string} action - The action to trigger.   * @param {Object|string} [data] - The data to send.   * @returns {*} The response.   */  static requestSync(action, data) {    const request = document.createTextNode("");    document.documentElement.append(request);    const sender = document.createEvent("CustomEvent");    sender.initCustomEvent("pdf.js.message", true, false, {      action,      data,      sync: true,    });    request.dispatchEvent(sender);    const response = sender.detail.response;    request.remove();    return response;  }  /**   * 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<any>} 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 = document.createEvent("CustomEvent");    sender.initCustomEvent("pdf.js.message", true, false, {      action,      data,      sync: false,      responseExpected: !!callback,    });    request.dispatchEvent(sender);  }}class DownloadManager {  #openBlobUrls = new WeakMap();  downloadUrl(url, filename) {    FirefoxCom.request("download", {      originalUrl: url,      filename,    });  }  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(element, data, filename) {    const isPdfData = isPdfFile(filename);    const contentType = isPdfData ? "application/pdf" : "";    if (isPdfData) {      let blobUrl = this.#openBlobUrls.get(element);      if (!blobUrl) {        blobUrl = URL.createObjectURL(new Blob([data], { type: contentType }));        this.#openBlobUrls.set(element, blobUrl);      }      // Let Firefox's content handler catch the URL and display the PDF.      const viewerUrl = blobUrl + "#filename=" + encodeURIComponent(filename);      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(element);      }    }    this.downloadData(data, filename, contentType);    return false;  }  download(blob, url, filename) {    const blobUrl = URL.createObjectURL(blob);    FirefoxCom.request("download", {      blobUrl,      originalUrl: url,      filename,    });  }}class FirefoxPreferences extends BasePreferences {  async _writeToStorage(prefObj) {    return FirefoxCom.requestAsync("setPreferences", prefObj);  }  async _readFromStorage(prefObj) {    const prefStr = await FirefoxCom.requestAsync("getPreferences", prefObj);    return JSON.parse(prefStr);  }}class MozL10n {  constructor(mozL10n) {    this.mozL10n = mozL10n;  }  async getLanguage() {    return this.mozL10n.getLanguage();  }  async getDirection() {    return this.mozL10n.getDirection();  }  async get(key, args = null, fallback = getL10nFallback(key, args)) {    return this.mozL10n.get(key, args, fallback);  }  async translate(element) {    this.mozL10n.translate(element);  }}(function listenFindEvents() {  const events = [    "find",    "findagain",    "findhighlightallchange",    "findcasesensitivitychange",    "findentirewordchange",    "findbarclose",    "finddiacriticmatchingchange",  ];  const findLen = "find".length;  const handleEvent = function ({ type, detail }) {    if (!PDFViewerApplication.initialized) {      return;    }    if (type === "findbarclose") {      PDFViewerApplication.eventBus.dispatch(type, { source: window });      return;    }    PDFViewerApplication.eventBus.dispatch("find", {      source: window,      type: type.substring(findLen),      query: detail.query,      phraseSearch: true,      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 (!PDFViewerApplication.initialized) {      return;    }    // Avoid attempting to needlessly reset the zoom level *twice* in a row,    // when using the `Ctrl + 0` keyboard shortcut.    if (      type === "zoomreset" &&      PDFViewerApplication.pdfViewer.currentScaleValue === DEFAULT_SCALE_VALUE    ) {      return;    }    PDFViewerApplication.eventBus.dispatch(type, { source: window });  };  for (const event of events) {    window.addEventListener(event, handleEvent);  }})();(function listenSaveEvent() {  const handleEvent = function ({ type, detail }) {    if (!PDFViewerApplication.initialized) {      return;    }    PDFViewerApplication.eventBus.dispatch("download", { source: window });  };  window.addEventListener("save", handleEvent);})();(function listenEditingEvent() {  const handleEvent = function ({ detail }) {    if (!PDFViewerApplication.initialized) {      return;    }    PDFViewerApplication.eventBus.dispatch("editingaction", {      source: window,      name: detail.name,    });  };  window.addEventListener("editingaction", handleEvent);})();class FirefoxComDataRangeTransport extends PDFDataRangeTransport {  requestDataRange(begin, end) {    FirefoxCom.request("requestDataRange", { begin, end });  }  abort() {    // Sync call to ensure abort is really started.    FirefoxCom.requestSync("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 FirefoxExternalServices extends DefaultExternalServices {  static updateFindControlState(data) {    FirefoxCom.request("updateFindControlState", data);  }  static updateFindMatchesCount(data) {    FirefoxCom.request("updateFindMatchesCount", data);  }  static 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(            args.pdfUrl,            args.length,            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.requestSync("initPassiveLoading", null);  }  static reportTelemetry(data) {    FirefoxCom.request("reportTelemetry", JSON.stringify(data));  }  static createDownloadManager() {    return new DownloadManager();  }  static createPreferences() {    return new FirefoxPreferences();  }  static updateEditorStates(data) {    FirefoxCom.request("updateEditorStates", data);  }  static createL10n(options) {    const mozL10n = document.mozL10n;    // TODO refactor mozL10n.setExternalLocalizerServices    return new MozL10n(mozL10n);  }  static createScripting(options) {    return FirefoxScripting;  }  static get supportsIntegratedFind() {    const support = FirefoxCom.requestSync("supportsIntegratedFind");    return shadow(this, "supportsIntegratedFind", support);  }  static get supportsDocumentFonts() {    const support = FirefoxCom.requestSync("supportsDocumentFonts");    return shadow(this, "supportsDocumentFonts", support);  }  static get supportedMouseWheelZoomModifierKeys() {    const support = FirefoxCom.requestSync(      "supportedMouseWheelZoomModifierKeys"    );    return shadow(this, "supportedMouseWheelZoomModifierKeys", support);  }  static get isInAutomation() {    // Returns the value of `Cu.isInAutomation`, which is only `true` when e.g.    // various test-suites are running in mozilla-central.    const isInAutomation = FirefoxCom.requestSync("isInAutomation");    return shadow(this, "isInAutomation", isInAutomation);  }}PDFViewerApplication.externalServices = FirefoxExternalServices;// l10n.js for Firefox extension expects services to be set.document.mozL10n.setExternalLocalizerServices({  getLocale() {    return FirefoxCom.requestSync("getLocale", null);  },  getStrings(key) {    return FirefoxCom.requestSync("getStrings", null);  },});export { DownloadManager, FirefoxCom };
 |