| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505 | /* Copyright 2021 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. *//** @typedef {import("./event_utils").EventBus} EventBus */import { apiPageLayoutToViewerModes, RenderingStates } from "./ui_utils.js";import { createPromiseCapability, shadow } from "pdfjs-lib";/** * @typedef {Object} PDFScriptingManagerOptions * @property {EventBus} eventBus - The application event bus. * @property {string} sandboxBundleSrc - The path and filename of the scripting *   bundle. * @property {Object} [scriptingFactory] - The factory that is used when *   initializing scripting; must contain a `createScripting` method. *   PLEASE NOTE: Primarily intended for the default viewer use-case. * @property {function} [docPropertiesLookup] - The function that is used to *   lookup the necessary document properties. */class PDFScriptingManager {  /**   * @param {PDFScriptingManagerOptions} options   */  constructor({    eventBus,    sandboxBundleSrc = null,    scriptingFactory = null,    docPropertiesLookup = null,  }) {    this._pdfDocument = null;    this._pdfViewer = null;    this._closeCapability = null;    this._destroyCapability = null;    this._scripting = null;    this._ready = false;    this._eventBus = eventBus;    this._sandboxBundleSrc = sandboxBundleSrc;    this._scriptingFactory = scriptingFactory;    this._docPropertiesLookup = docPropertiesLookup;    // The default viewer already handles adding/removing of DOM events,    // hence limit this to only the viewer components.    if (      typeof PDFJSDev !== "undefined" &&      PDFJSDev.test("COMPONENTS") &&      !this._scriptingFactory    ) {      window.addEventListener("updatefromsandbox", event => {        this._eventBus.dispatch("updatefromsandbox", {          source: window,          detail: event.detail,        });      });    }  }  setViewer(pdfViewer) {    this._pdfViewer = pdfViewer;  }  async setDocument(pdfDocument) {    if (this._pdfDocument) {      await this._destroyScripting();    }    this._pdfDocument = pdfDocument;    if (!pdfDocument) {      return;    }    const [objects, calculationOrder, docActions] = await Promise.all([      pdfDocument.getFieldObjects(),      pdfDocument.getCalculationOrderIds(),      pdfDocument.getJSActions(),    ]);    if (!objects && !docActions) {      // No FieldObjects or JavaScript actions were found in the document.      await this._destroyScripting();      return;    }    if (pdfDocument !== this._pdfDocument) {      return; // The document was closed while the data resolved.    }    try {      this._scripting = this._createScripting();    } catch (error) {      console.error(`PDFScriptingManager.setDocument: "${error?.message}".`);      await this._destroyScripting();      return;    }    this._internalEvents.set("updatefromsandbox", event => {      if (event?.source !== window) {        return;      }      this._updateFromSandbox(event.detail);    });    this._internalEvents.set("dispatcheventinsandbox", event => {      this._scripting?.dispatchEventInSandbox(event.detail);    });    this._internalEvents.set("pagechanging", ({ pageNumber, previous }) => {      if (pageNumber === previous) {        return; // The current page didn't change.      }      this._dispatchPageClose(previous);      this._dispatchPageOpen(pageNumber);    });    this._internalEvents.set("pagerendered", ({ pageNumber }) => {      if (!this._pageOpenPending.has(pageNumber)) {        return; // No pending "PageOpen" event for the newly rendered page.      }      if (pageNumber !== this._pdfViewer.currentPageNumber) {        return; // The newly rendered page is no longer the current one.      }      this._dispatchPageOpen(pageNumber);    });    this._internalEvents.set("pagesdestroy", async event => {      await this._dispatchPageClose(this._pdfViewer.currentPageNumber);      await this._scripting?.dispatchEventInSandbox({        id: "doc",        name: "WillClose",      });      this._closeCapability?.resolve();    });    for (const [name, listener] of this._internalEvents) {      this._eventBus._on(name, listener);    }    try {      const docProperties = await this._getDocProperties();      if (pdfDocument !== this._pdfDocument) {        return; // The document was closed while the properties resolved.      }      await this._scripting.createSandbox({        objects,        calculationOrder,        appInfo: {          platform: navigator.platform,          language: navigator.language,        },        docInfo: {          ...docProperties,          actions: docActions,        },      });      this._eventBus.dispatch("sandboxcreated", { source: this });    } catch (error) {      console.error(`PDFScriptingManager.setDocument: "${error?.message}".`);      await this._destroyScripting();      return;    }    await this._scripting?.dispatchEventInSandbox({      id: "doc",      name: "Open",    });    await this._dispatchPageOpen(      this._pdfViewer.currentPageNumber,      /* initialize = */ true    );    // Defer this slightly, to ensure that scripting is *fully* initialized.    Promise.resolve().then(() => {      if (pdfDocument === this._pdfDocument) {        this._ready = true;      }    });  }  async dispatchWillSave(detail) {    return this._scripting?.dispatchEventInSandbox({      id: "doc",      name: "WillSave",    });  }  async dispatchDidSave(detail) {    return this._scripting?.dispatchEventInSandbox({      id: "doc",      name: "DidSave",    });  }  async dispatchWillPrint(detail) {    return this._scripting?.dispatchEventInSandbox({      id: "doc",      name: "WillPrint",    });  }  async dispatchDidPrint(detail) {    return this._scripting?.dispatchEventInSandbox({      id: "doc",      name: "DidPrint",    });  }  get destroyPromise() {    return this._destroyCapability?.promise || null;  }  get ready() {    return this._ready;  }  /**   * @private   */  get _internalEvents() {    return shadow(this, "_internalEvents", new Map());  }  /**   * @private   */  get _pageOpenPending() {    return shadow(this, "_pageOpenPending", new Set());  }  /**   * @private   */  get _visitedPages() {    return shadow(this, "_visitedPages", new Map());  }  /**   * @private   */  async _updateFromSandbox(detail) {    // Ignore some events, see below, that don't make sense in PresentationMode.    const isInPresentationMode =      this._pdfViewer.isInPresentationMode ||      this._pdfViewer.isChangingPresentationMode;    const { id, siblings, command, value } = detail;    if (!id) {      switch (command) {        case "clear":          console.clear();          break;        case "error":          console.error(value);          break;        case "layout": {          // NOTE: Always ignore the pageLayout in GeckoView since there's          // no UI available to change Scroll/Spread modes for the user.          if (            (typeof PDFJSDev !== "undefined" && PDFJSDev.test("GECKOVIEW")) ||            isInPresentationMode          ) {            return;          }          const modes = apiPageLayoutToViewerModes(value);          this._pdfViewer.spreadMode = modes.spreadMode;          break;        }        case "page-num":          this._pdfViewer.currentPageNumber = value + 1;          break;        case "print":          await this._pdfViewer.pagesPromise;          this._eventBus.dispatch("print", { source: this });          break;        case "println":          console.log(value);          break;        case "zoom":          if (isInPresentationMode) {            return;          }          this._pdfViewer.currentScaleValue = value;          break;        case "SaveAs":          this._eventBus.dispatch("download", { source: this });          break;        case "FirstPage":          this._pdfViewer.currentPageNumber = 1;          break;        case "LastPage":          this._pdfViewer.currentPageNumber = this._pdfViewer.pagesCount;          break;        case "NextPage":          this._pdfViewer.nextPage();          break;        case "PrevPage":          this._pdfViewer.previousPage();          break;        case "ZoomViewIn":          if (isInPresentationMode) {            return;          }          this._pdfViewer.increaseScale();          break;        case "ZoomViewOut":          if (isInPresentationMode) {            return;          }          this._pdfViewer.decreaseScale();          break;      }      return;    }    if (isInPresentationMode) {      if (detail.focus) {        return;      }    }    delete detail.id;    delete detail.siblings;    const ids = siblings ? [id, ...siblings] : [id];    for (const elementId of ids) {      const element = document.querySelector(        `[data-element-id="${elementId}"]`      );      if (element) {        element.dispatchEvent(new CustomEvent("updatefromsandbox", { detail }));      } else {        // The element hasn't been rendered yet, use the AnnotationStorage.        this._pdfDocument?.annotationStorage.setValue(elementId, detail);      }    }  }  /**   * @private   */  async _dispatchPageOpen(pageNumber, initialize = false) {    const pdfDocument = this._pdfDocument,      visitedPages = this._visitedPages;    if (initialize) {      this._closeCapability = createPromiseCapability();    }    if (!this._closeCapability) {      return; // Scripting isn't fully initialized yet.    }    const pageView = this._pdfViewer.getPageView(/* index = */ pageNumber - 1);    if (pageView?.renderingState !== RenderingStates.FINISHED) {      this._pageOpenPending.add(pageNumber);      return; // Wait for the page to finish rendering.    }    this._pageOpenPending.delete(pageNumber);    const actionsPromise = (async () => {      // Avoid sending, and thus serializing, the `actions` data more than once.      const actions = await (!visitedPages.has(pageNumber)        ? pageView.pdfPage?.getJSActions()        : null);      if (pdfDocument !== this._pdfDocument) {        return; // The document was closed while the actions resolved.      }      await this._scripting?.dispatchEventInSandbox({        id: "page",        name: "PageOpen",        pageNumber,        actions,      });    })();    visitedPages.set(pageNumber, actionsPromise);  }  /**   * @private   */  async _dispatchPageClose(pageNumber) {    const pdfDocument = this._pdfDocument,      visitedPages = this._visitedPages;    if (!this._closeCapability) {      return; // Scripting isn't fully initialized yet.    }    if (this._pageOpenPending.has(pageNumber)) {      return; // The page is still rendering; no "PageOpen" event dispatched.    }    const actionsPromise = visitedPages.get(pageNumber);    if (!actionsPromise) {      return; // The "PageClose" event must be preceded by a "PageOpen" event.    }    visitedPages.set(pageNumber, null);    // Ensure that the "PageOpen" event is dispatched first.    await actionsPromise;    if (pdfDocument !== this._pdfDocument) {      return; // The document was closed while the actions resolved.    }    await this._scripting?.dispatchEventInSandbox({      id: "page",      name: "PageClose",      pageNumber,    });  }  /**   * @returns {Promise<Object>} A promise that is resolved with an {Object}   *   containing the necessary document properties; please find the expected   *   format in `PDFViewerApplication._scriptingDocProperties`.   * @private   */  async _getDocProperties() {    if (this._docPropertiesLookup) {      return this._docPropertiesLookup(this._pdfDocument);    }    if (typeof PDFJSDev !== "undefined" && PDFJSDev.test("COMPONENTS")) {      const { docPropertiesLookup } = require("./generic_scripting.js");      return docPropertiesLookup(this._pdfDocument);    }    throw new Error("_getDocProperties: Unable to lookup properties.");  }  /**   * @private   */  _createScripting() {    this._destroyCapability = createPromiseCapability();    if (this._scripting) {      throw new Error("_createScripting: Scripting already exists.");    }    if (this._scriptingFactory) {      return this._scriptingFactory.createScripting({        sandboxBundleSrc: this._sandboxBundleSrc,      });    }    if (typeof PDFJSDev !== "undefined" && PDFJSDev.test("COMPONENTS")) {      const { GenericScripting } = require("./generic_scripting.js");      return new GenericScripting(this._sandboxBundleSrc);    }    throw new Error("_createScripting: Cannot create scripting.");  }  /**   * @private   */  async _destroyScripting() {    if (!this._scripting) {      this._pdfDocument = null;      this._destroyCapability?.resolve();      return;    }    if (this._closeCapability) {      await Promise.race([        this._closeCapability.promise,        new Promise(resolve => {          // Avoid the scripting/sandbox-destruction hanging indefinitely.          setTimeout(resolve, 1000);        }),      ]).catch(reason => {        // Ignore any errors, to ensure that the sandbox is always destroyed.      });      this._closeCapability = null;    }    this._pdfDocument = null;    try {      await this._scripting.destroySandbox();    } catch (ex) {}    for (const [name, listener] of this._internalEvents) {      this._eventBus._off(name, listener);    }    this._internalEvents.clear();    this._pageOpenPending.clear();    this._visitedPages.clear();    this._scripting = null;    this._ready = false;    this._destroyCapability?.resolve();  }}export { PDFScriptingManager };
 |