| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428 | /* Copyright 2013 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. *//* globals chrome */import { DefaultExternalServices, PDFViewerApplication } from "./app.js";import { AppOptions } from "./app_options.js";import { BasePreferences } from "./preferences.js";import { DownloadManager } from "./download_manager.js";import { GenericL10n } from "./genericl10n.js";import { GenericScripting } from "./generic_scripting.js";if (typeof PDFJSDev === "undefined" || !PDFJSDev.test("CHROME")) {  throw new Error(    'Module "pdfjs-web/chromecom" shall not be used outside CHROME build.'  );}const ChromeCom = {  /**   * Creates an event that the extension is listening for and will   * asynchronously respond by calling the callback.   *   * @param {string} action - The action to trigger.   * @param {string} [data] - The data to send.   * @param {Function} [callback] - Response callback that will be called with   *   one data argument. When the request cannot be handled, the callback is   *   immediately invoked with no arguments.   */  request(action, data, callback) {    const message = {      action,      data,    };    if (!chrome.runtime) {      console.error("chrome.runtime is undefined.");      callback?.();    } else if (callback) {      chrome.runtime.sendMessage(message, callback);    } else {      chrome.runtime.sendMessage(message);    }  },  /**   * Resolves a PDF file path and attempts to detects length.   *   * @param {string} file - Absolute URL of PDF file.   * @param {OverlayManager} overlayManager - Manager for the viewer overlays.   * @param {Function} callback - A callback with resolved URL and file length.   */  resolvePDFFile(file, overlayManager, callback) {    // Expand drive:-URLs to filesystem URLs (Chrome OS)    file = file.replace(      /^drive:/i,      "filesystem:" + location.origin + "/external/"    );    if (/^https?:/.test(file)) {      // Assumption: The file being opened is the file that was requested.      // There is no UI to input a different URL, so this assumption will hold      // for now.      setReferer(file, function () {        callback(file);      });      return;    }    if (/^file?:/.test(file)) {      getEmbedderOrigin(function (origin) {        // If the origin cannot be determined, let Chrome decide whether to        // allow embedding files. Otherwise, only allow local files to be        // embedded from local files or Chrome extensions.        // Even without this check, the file load in frames is still blocked,        // but this may change in the future (https://crbug.com/550151).        if (origin && !/^file:|^chrome-extension:/.test(origin)) {          PDFViewerApplication._documentError(            "Blocked " +              origin +              " from loading " +              file +              ". Refused to load a local file in a non-local page " +              "for security reasons."          );          return;        }        isAllowedFileSchemeAccess(function (isAllowedAccess) {          if (isAllowedAccess) {            callback(file);          } else {            requestAccessToLocalFile(file, overlayManager, callback);          }        });      });      return;    }    callback(file);  },};function getEmbedderOrigin(callback) {  const origin = window === top ? location.origin : location.ancestorOrigins[0];  if (origin === "null") {    // file:-URLs, data-URLs, sandboxed frames, etc.    getParentOrigin(callback);  } else {    callback(origin);  }}function getParentOrigin(callback) {  ChromeCom.request("getParentOrigin", null, callback);}function isAllowedFileSchemeAccess(callback) {  ChromeCom.request("isAllowedFileSchemeAccess", null, callback);}function isRuntimeAvailable() {  try {    // When the extension is reloaded, the extension runtime is destroyed and    // the extension APIs become unavailable.    if (chrome.runtime?.getManifest()) {      return true;    }  } catch (e) {}  return false;}function reloadIfRuntimeIsUnavailable() {  if (!isRuntimeAvailable()) {    location.reload();  }}let chromeFileAccessOverlayPromise;function requestAccessToLocalFile(fileUrl, overlayManager, callback) {  const dialog = document.getElementById("chromeFileAccessDialog");  if (top !== window) {    // When the extension reloads after receiving new permissions, the pages    // have to be reloaded to restore the extension runtime. Auto-reload    // frames, because users should not have to reload the whole page just to    // update the viewer.    // Top-level frames are closed by Chrome upon reload, so there is no need    // for detecting unload of the top-level frame. Should this ever change    // (crbug.com/511670), then the user can just reload the tab.    window.addEventListener("focus", reloadIfRuntimeIsUnavailable);    dialog.addEventListener("close", function () {      window.removeEventListener("focus", reloadIfRuntimeIsUnavailable);      reloadIfRuntimeIsUnavailable();    });  }  chromeFileAccessOverlayPromise ||= overlayManager.register(    dialog,    /* canForceClose = */ true  );  chromeFileAccessOverlayPromise.then(function () {    const iconPath = chrome.runtime.getManifest().icons[48];    document.getElementById("chrome-pdfjs-logo-bg").style.backgroundImage =      "url(" + chrome.runtime.getURL(iconPath) + ")";    // Use Chrome's definition of UI language instead of PDF.js's #lang=...,    // because the shown string should match the UI at chrome://extensions.    // These strings are from chrome/app/resources/generated_resources_*.xtb.    const i18nFileAccessLabel = PDFJSDev.json(      "$ROOT/web/chrome-i18n-allow-access-to-file-urls.json"    )[chrome.i18n.getUILanguage?.()];    if (i18nFileAccessLabel) {      document.getElementById("chrome-file-access-label").textContent =        i18nFileAccessLabel;    }    const link = document.getElementById("chrome-link-to-extensions-page");    link.href = "chrome://extensions/?id=" + chrome.runtime.id;    link.onclick = function (e) {      // Direct navigation to chrome:// URLs is blocked by Chrome, so we      // have to ask the background page to open chrome://extensions/?id=...      e.preventDefault();      // Open in the current tab by default, because toggling the file access      // checkbox causes the extension to reload, and Chrome will close all      // tabs upon reload.      ChromeCom.request("openExtensionsPageForFileAccess", {        newTab: e.ctrlKey || e.metaKey || e.button === 1 || window !== top,      });    };    // Show which file is being opened to help the user with understanding    // why this permission request is shown.    document.getElementById("chrome-url-of-local-file").textContent = fileUrl;    document.getElementById("chrome-file-fallback").onchange = function () {      const file = this.files[0];      if (file) {        const originalFilename = decodeURIComponent(fileUrl.split("/").pop());        let originalUrl = fileUrl;        if (originalFilename !== file.name) {          const msg =            "The selected file does not match the original file." +            "\nOriginal: " +            originalFilename +            "\nSelected: " +            file.name +            "\nDo you want to open the selected file?";          // eslint-disable-next-line no-alert          if (!confirm(msg)) {            this.value = "";            return;          }          // There is no way to retrieve the original URL from the File object.          // So just generate a fake path.          originalUrl = "file:///fakepath/to/" + encodeURIComponent(file.name);        }        callback(URL.createObjectURL(file), file.size, originalUrl);        overlayManager.close(dialog);      }    };    overlayManager.open(dialog);  });}if (window === top) {  // Chrome closes all extension tabs (crbug.com/511670) when the extension  // reloads. To counter this, the tab URL and history state is saved to  // localStorage and restored by extension-router.js.  // Unfortunately, the window and tab index are not restored. And if it was  // the only tab in an incognito window, then the tab is not restored either.  addEventListener("unload", function () {    // If the runtime is still available, the unload is most likely a normal    // tab closure. Otherwise it is most likely an extension reload.    if (!isRuntimeAvailable()) {      localStorage.setItem(        "unload-" + Date.now() + "-" + document.hidden + "-" + location.href,        JSON.stringify(history.state)      );    }  });}// This port is used for several purposes:// 1. When disconnected, the background page knows that the frame has unload.// 2. When the referrer was saved in history.state.chromecomState, it is sent//    to the background page.// 3. When the background page knows the referrer of the page, the referrer is//    saved in history.state.chromecomState.let port;// Set the referer for the given URL.// 0. Background: If loaded via a http(s) URL: Save referer.// 1. Page -> background: send URL and referer from history.state// 2. Background: Bind referer to URL (via webRequest).// 3. Background -> page: Send latest referer and save to history.// 4. Page: Invoke callback.function setReferer(url, callback) {  if (!port) {    // The background page will accept the port, and keep adding the Referer    // request header to requests to |url| until the port is disconnected.    port = chrome.runtime.connect({ name: "chromecom-referrer" });  }  port.onDisconnect.addListener(onDisconnect);  port.onMessage.addListener(onMessage);  // Initiate the information exchange.  port.postMessage({    referer: window.history.state?.chromecomState,    requestUrl: url,  });  function onMessage(referer) {    if (referer) {      // The background extracts the Referer from the initial HTTP request for      // the PDF file. When the viewer is reloaded or when the user navigates      // back and forward, the background page will not observe a HTTP request      // with Referer. To make sure that the Referer is preserved, store it in      // history.state, which is preserved across reloads/navigations.      const state = window.history.state || {};      state.chromecomState = referer;      window.history.replaceState(state, "");    }    onCompleted();  }  function onDisconnect() {    // When the connection fails, ignore the error and call the callback.    port = null;    callback();  }  function onCompleted() {    port.onDisconnect.removeListener(onDisconnect);    port.onMessage.removeListener(onMessage);    callback();  }}// chrome.storage.sync is not supported in every Chromium-derivate.// Note: The background page takes care of migrating values from// chrome.storage.local to chrome.storage.sync when needed.const storageArea = chrome.storage.sync || chrome.storage.local;class ChromePreferences extends BasePreferences {  async _writeToStorage(prefObj) {    return new Promise(resolve => {      if (prefObj === this.defaults) {        const keysToRemove = Object.keys(this.defaults);        // If the storage is reset, remove the keys so that the values from        // managed storage are applied again.        storageArea.remove(keysToRemove, function () {          resolve();        });      } else {        storageArea.set(prefObj, function () {          resolve();        });      }    });  }  async _readFromStorage(prefObj) {    return new Promise(resolve => {      const getPreferences = defaultPrefs => {        if (chrome.runtime.lastError) {          // Managed storage not supported, e.g. in Opera.          defaultPrefs = this.defaults;        }        storageArea.get(defaultPrefs, function (readPrefs) {          resolve(readPrefs);        });      };      if (chrome.storage.managed) {        // Get preferences as set by the system administrator.        // See extensions/chromium/preferences_schema.json for more information.        // These preferences can be overridden by the user.        // Deprecated preferences are removed from web/default_preferences.json,        // but kept in extensions/chromium/preferences_schema.json for backwards        // compatibility with managed preferences.        const defaultManagedPrefs = Object.assign(          {            enableHandToolOnLoad: false,            disableTextLayer: false,            enhanceTextSelection: false,            showPreviousViewOnLoad: true,            disablePageMode: false,          },          this.defaults        );        chrome.storage.managed.get(defaultManagedPrefs, function (items) {          items = items || defaultManagedPrefs;          // Migration logic for deprecated preferences: If the new preference          // is not defined by an administrator (i.e. the value is the same as          // the default value), and a deprecated preference is set with a          // non-default value, migrate the deprecated preference value to the          // new preference value.          // Never remove this, because we have no means of modifying managed          // preferences.          // Migration code for https://github.com/mozilla/pdf.js/pull/7635.          if (items.enableHandToolOnLoad && !items.cursorToolOnLoad) {            items.cursorToolOnLoad = 1;          }          delete items.enableHandToolOnLoad;          // Migration code for https://github.com/mozilla/pdf.js/pull/9479.          if (items.textLayerMode !== 1 && items.disableTextLayer) {            items.textLayerMode = 0;          }          delete items.disableTextLayer;          delete items.enhanceTextSelection;          // Migration code for https://github.com/mozilla/pdf.js/pull/10502.          if (!items.showPreviousViewOnLoad && !items.viewOnLoad) {            items.viewOnLoad = 1;          }          delete items.showPreviousViewOnLoad;          delete items.disablePageMode;          getPreferences(items);        });      } else {        // Managed storage not supported, e.g. in old Chromium versions.        getPreferences(this.defaults);      }    });  }}class ChromeExternalServices extends DefaultExternalServices {  static initPassiveLoading(callbacks) {    // defaultUrl is set in viewer.js    ChromeCom.resolvePDFFile(      AppOptions.get("defaultUrl"),      PDFViewerApplication.overlayManager,      function (url, length, originalUrl) {        callbacks.onOpenWithURL(url, length, originalUrl);      }    );  }  static createDownloadManager() {    return new DownloadManager();  }  static createPreferences() {    return new ChromePreferences();  }  static createL10n(options) {    return new GenericL10n(navigator.language);  }  static createScripting({ sandboxBundleSrc }) {    return new GenericScripting(sandboxBundleSrc);  }}PDFViewerApplication.externalServices = ChromeExternalServices;export { ChromeCom };
 |