chromecom.js 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324
  1. /**
  2. * @licstart The following is the entire license notice for the
  3. * JavaScript code in this page
  4. *
  5. * Copyright 2022 Mozilla Foundation
  6. *
  7. * Licensed under the Apache License, Version 2.0 (the "License");
  8. * you may not use this file except in compliance with the License.
  9. * You may obtain a copy of the License at
  10. *
  11. * http://www.apache.org/licenses/LICENSE-2.0
  12. *
  13. * Unless required by applicable law or agreed to in writing, software
  14. * distributed under the License is distributed on an "AS IS" BASIS,
  15. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  16. * See the License for the specific language governing permissions and
  17. * limitations under the License.
  18. *
  19. * @licend The above is the entire license notice for the
  20. * JavaScript code in this page
  21. */
  22. "use strict";
  23. Object.defineProperty(exports, "__esModule", {
  24. value: true
  25. });
  26. exports.ChromeCom = void 0;
  27. var _app = require("./app.js");
  28. var _app_options = require("./app_options.js");
  29. var _preferences = require("./preferences.js");
  30. var _download_manager = require("./download_manager.js");
  31. var _genericl10n = require("./genericl10n.js");
  32. var _generic_scripting = require("./generic_scripting.js");
  33. {
  34. throw new Error('Module "pdfjs-web/chromecom" shall not be used outside CHROME build.');
  35. }
  36. const ChromeCom = {
  37. request(action, data, callback) {
  38. const message = {
  39. action,
  40. data
  41. };
  42. if (!chrome.runtime) {
  43. console.error("chrome.runtime is undefined.");
  44. callback?.();
  45. } else if (callback) {
  46. chrome.runtime.sendMessage(message, callback);
  47. } else {
  48. chrome.runtime.sendMessage(message);
  49. }
  50. },
  51. resolvePDFFile(file, overlayManager, callback) {
  52. file = file.replace(/^drive:/i, "filesystem:" + location.origin + "/external/");
  53. if (/^https?:/.test(file)) {
  54. setReferer(file, function () {
  55. callback(file);
  56. });
  57. return;
  58. }
  59. if (/^file?:/.test(file)) {
  60. getEmbedderOrigin(function (origin) {
  61. if (origin && !/^file:|^chrome-extension:/.test(origin)) {
  62. _app.PDFViewerApplication._documentError("Blocked " + origin + " from loading " + file + ". Refused to load a local file in a non-local page " + "for security reasons.");
  63. return;
  64. }
  65. isAllowedFileSchemeAccess(function (isAllowedAccess) {
  66. if (isAllowedAccess) {
  67. callback(file);
  68. } else {
  69. requestAccessToLocalFile(file, overlayManager, callback);
  70. }
  71. });
  72. });
  73. return;
  74. }
  75. callback(file);
  76. }
  77. };
  78. exports.ChromeCom = ChromeCom;
  79. function getEmbedderOrigin(callback) {
  80. const origin = window === top ? location.origin : location.ancestorOrigins[0];
  81. if (origin === "null") {
  82. getParentOrigin(callback);
  83. } else {
  84. callback(origin);
  85. }
  86. }
  87. function getParentOrigin(callback) {
  88. ChromeCom.request("getParentOrigin", null, callback);
  89. }
  90. function isAllowedFileSchemeAccess(callback) {
  91. ChromeCom.request("isAllowedFileSchemeAccess", null, callback);
  92. }
  93. function isRuntimeAvailable() {
  94. try {
  95. if (chrome.runtime?.getManifest()) {
  96. return true;
  97. }
  98. } catch (e) {}
  99. return false;
  100. }
  101. function reloadIfRuntimeIsUnavailable() {
  102. if (!isRuntimeAvailable()) {
  103. location.reload();
  104. }
  105. }
  106. let chromeFileAccessOverlayPromise;
  107. function requestAccessToLocalFile(fileUrl, overlayManager, callback) {
  108. const dialog = document.getElementById("chromeFileAccessDialog");
  109. if (top !== window) {
  110. window.addEventListener("focus", reloadIfRuntimeIsUnavailable);
  111. dialog.addEventListener("close", function () {
  112. window.removeEventListener("focus", reloadIfRuntimeIsUnavailable);
  113. reloadIfRuntimeIsUnavailable();
  114. });
  115. }
  116. chromeFileAccessOverlayPromise ||= overlayManager.register(dialog, true);
  117. chromeFileAccessOverlayPromise.then(function () {
  118. const iconPath = chrome.runtime.getManifest().icons[48];
  119. document.getElementById("chrome-pdfjs-logo-bg").style.backgroundImage = "url(" + chrome.runtime.getURL(iconPath) + ")";
  120. const i18nFileAccessLabel = {
  121. "am": "\u1208\u134b\u12ed\u120d \u12e9\u12a0\u122d\u12a4\u120d\u12ce\u127d \u1218\u12f3\u1228\u123b \u134d\u1240\u12f5",
  122. "ar": "\u200f\u0627\u0644\u0633\u0645\u0627\u062d \u0628\u0627\u0644\u062f\u062e\u0648\u0644 \u0625\u0644\u0649 \u0639\u0646\u0627\u0648\u064a\u0646 URL \u0644\u0644\u0645\u0644\u0641\u0627\u062a",
  123. "bg": "\u0414\u0430 \u0441\u0435 \u0440\u0430\u0437\u0440\u0435\u0448\u0438 \u0434\u043e\u0441\u0442\u044a\u043f \u0434\u043e URL \u0430\u0434\u0440\u0435\u0441\u0438\u0442\u0435 \u043d\u0430 \u0444\u0430\u0439\u043b\u043e\u0432\u0435\u0442\u0435",
  124. "bn": "\u09ab\u09be\u0987\u09b2 URL\u0997\u09c1\u09b2\u09bf\u09a4\u09c7 \u0985\u09cd\u09af\u09be\u0995\u09cd\u09b8\u09c7\u09b8 \u09ae\u099e\u09cd\u099c\u09c1\u09b0 \u0995\u09b0\u09c1\u09a8",
  125. "ca": "Permet l'acc\u00e9s als URL de fitxer",
  126. "cs": "Umo\u017enit p\u0159\u00edstup k adres\u00e1m URL soubor\u016f",
  127. "da": "Tillad adgang til webadresser p\u00e5 filer",
  128. "de": "Zugriff auf Datei-URLs zulassen",
  129. "el": "\u039d\u03b1 \u03b5\u03c0\u03b9\u03c4\u03c1\u03ad\u03c0\u03b5\u03c4\u03b1\u03b9 \u03b7 \u03c0\u03c1\u03cc\u03c3\u03b2\u03b1\u03c3\u03b7 \u03c3\u03b5 \u03b4\u03b9\u03b5\u03c5\u03b8\u03cd\u03bd\u03c3\u03b5\u03b9\u03c2 URL \u03b1\u03c1\u03c7\u03b5\u03af\u03c9\u03bd",
  130. "en-GB": "Allow access to file URLs",
  131. "es": "Permitir acceso a URL de archivo",
  132. "es-419": "Permitir el acceso a las URL del archivo",
  133. "et": "Luba juurdep\u00e4\u00e4s failide URL-idele",
  134. "fa": "\u200f\u0627\u062c\u0627\u0632\u0647\u0654 \u062f\u0633\u062a\u0631\u0633\u06cc \u0628\u0647 URL \u0647\u0627\u06cc \u0641\u0627\u06cc\u0644",
  135. "fi": "Salli tiedostojen URL-osoitteiden k\u00e4ytt\u00f6",
  136. "fil": "Payagan ang access na mag-file ng mga URL",
  137. "fr": "Autoriser l'acc\u00e8s aux URL de fichier",
  138. "gu": "URL \u0aab\u0abe\u0a87\u0ab2 \u0a95\u0ab0\u0ab5\u0abe \u0a8d\u0a95\u0acd\u0ab8\u0ac7\u0ab8\u0aa8\u0ac0 \u0aae\u0a82\u0a9c\u0ac2\u0ab0\u0ac0 \u0a86\u0aaa\u0acb",
  139. "hi": "\u092b\u093c\u093e\u0907\u0932 URL \u0924\u0915 \u092a\u0939\u0941\u0902\u091a\u0928\u0947 \u0915\u0940 \u0905\u0928\u0941\u092e\u0924\u093f \u0926\u0947\u0902",
  140. "hr": "Dozvoli pristup URL-ovima datoteke",
  141. "hu": "F\u00e1jl URL-ekhez val\u00f3 hozz\u00e1f\u00e9r\u00e9s enged\u00e9lyez\u00e9se",
  142. "id": "Izinkan akses ke URL file",
  143. "it": "Consenti l'accesso agli URL dei file",
  144. "iw": "\u05d0\u05e4\u05e9\u05e8 \u05d2\u05d9\u05e9\u05d4 \u05dc\u05db\u05ea\u05d5\u05d1\u05d5\u05ea \u05d0\u05ea\u05e8\u05d9\u05dd \u05e9\u05dc \u05e7\u05d1\u05e6\u05d9\u05dd",
  145. "ja": "\u30d5\u30a1\u30a4\u30eb\u306e URL \u3078\u306e\u30a2\u30af\u30bb\u30b9\u3092\u8a31\u53ef\u3059\u308b",
  146. "kn": "URL \u0c97\u0cb3\u0ca8\u0ccd\u0ca8\u0cc1 \u0cab\u0cc8\u0cb2\u0ccd\u200c\u0c97\u0cb3\u0cbf\u0c97\u0cc6 \u0caa\u0ccd\u0cb0\u0cb5\u0cc7\u0cb6\u0cbf\u0cb8\u0cb2\u0cc1 \u0c85\u0ca8\u0cc1\u0cae\u0ca4\u0cbf\u0cb8\u0cbf",
  147. "ko": "\ud30c\uc77c URL\uc5d0 \ub300\ud55c \uc561\uc138\uc2a4 \ud5c8\uc6a9",
  148. "lt": "Leisti pasiekti failo URL",
  149. "lv": "At\u013caut piek\u013cuvi faila vietr\u0101\u017eiem URL",
  150. "ml": "URL \u0d15\u0d33\u0d4d\u200d\u200c \u0d2b\u0d2f\u0d32\u0d4d\u200d\u200c \u0d1a\u0d46\u0d2f\u0d4d\u0d2f\u0d41\u0d28\u0d4d\u0d28\u0d24\u0d3f\u0d28\u0d4d \u0d06\u0d15\u0d4d\u200d\u0d38\u0d38\u0d4d\u0d38\u0d4d \u0d05\u0d28\u0d41\u0d35\u0d26\u0d3f\u0d15\u0d4d\u0d15\u0d41\u0d15",
  151. "mr": "\u092b\u093e\u0907\u0932 URL \u092e\u0927\u094d\u092f\u0947 \u092a\u094d\u0930\u0935\u0947\u0936\u093e\u0938 \u0905\u0928\u0941\u092e\u0924\u0940 \u0926\u094d\u092f\u093e",
  152. "ms": "Membenarkan akses ke URL fail",
  153. "nl": "Toegang tot bestand-URL's toestaan",
  154. "no": "Tillat tilgang til filnettadresser",
  155. "pl": "Zezwalaj na dost\u0119p do adres\u00f3w URL plik\u00f3w",
  156. "pt-BR": "Permitir acesso aos URLs do arquivo",
  157. "pt-PT": "Permitir acesso a URLs de ficheiro",
  158. "ro": "Permite accesul la adresele URL de fi\u0219iere",
  159. "ru": "\u0420\u0430\u0437\u0440\u0435\u0448\u0438\u0442\u044c \u043e\u0442\u043a\u0440\u044b\u0432\u0430\u0442\u044c \u0444\u0430\u0439\u043b\u044b \u043f\u043e \u0441\u0441\u044b\u043b\u043a\u0430\u043c",
  160. "sk": "Povoli\u0165 pr\u00edstup k webov\u00fdm adres\u00e1m s\u00faboru",
  161. "sl": "Dovoli dostop do URL-jev datoteke",
  162. "sr": "\u0414\u043e\u0437\u0432\u043e\u043b\u0438 \u043f\u0440\u0438\u0441\u0442\u0443\u043f URL \u0430\u0434\u0440\u0435\u0441\u0430\u043c\u0430 \u0434\u0430\u0442\u043e\u0442\u0435\u043a\u0430",
  163. "sv": "Till\u00e5t \u00e5tkomst till webbadresser i filen",
  164. "sw": "Ruhusu kufikia URL za faili",
  165. "ta": "\u0b95\u0bcb\u0baa\u0bcd\u0baa\u0bc1 URL\u0b95\u0bb3\u0bc1\u0b95\u0bcd\u0b95\u0bc1 \u0b85\u0ba3\u0bc1\u0b95\u0bb2\u0bc8 \u0b85\u0ba9\u0bc1\u0bae\u0ba4\u0bbf",
  166. "te": "\u0c2b\u0c48\u0c32\u0c4d URL\u0c32\u0c15\u0c41 \u0c2a\u0c4d\u0c30\u0c3e\u0c2a\u0c4d\u0c24\u0c3f\u0c28\u0c3f \u0c05\u0c28\u0c41\u0c2e\u0c24\u0c3f\u0c02\u0c1a\u0c41",
  167. "th": "\u0e2d\u0e19\u0e38\u0e0d\u0e32\u0e15\u0e43\u0e2b\u0e49\u0e40\u0e02\u0e49\u0e32\u0e16\u0e36\u0e07\u0e44\u0e1f\u0e25\u0e4c URL",
  168. "tr": "Dosya URL'lerine eri\u015fime izin ver",
  169. "uk": "\u041d\u0430\u0434\u0430\u0432\u0430\u0442\u0438 \u0434\u043e\u0441\u0442\u0443\u043f \u0434\u043e URL-\u0430\u0434\u0440\u0435\u0441 \u0444\u0430\u0439\u043b\u0443",
  170. "vi": "Cho ph\u00e9p truy c\u1eadp v\u00e0o c\u00e1c URL c\u1ee7a t\u1ec7p",
  171. "zh-CN": "\u5141\u8bb8\u8bbf\u95ee\u6587\u4ef6\u7f51\u5740",
  172. "zh-TW": "\u5141\u8a31\u5b58\u53d6\u6a94\u6848\u7db2\u5740"
  173. }[chrome.i18n.getUILanguage?.()];
  174. if (i18nFileAccessLabel) {
  175. document.getElementById("chrome-file-access-label").textContent = i18nFileAccessLabel;
  176. }
  177. const link = document.getElementById("chrome-link-to-extensions-page");
  178. link.href = "chrome://extensions/?id=" + chrome.runtime.id;
  179. link.onclick = function (e) {
  180. e.preventDefault();
  181. ChromeCom.request("openExtensionsPageForFileAccess", {
  182. newTab: e.ctrlKey || e.metaKey || e.button === 1 || window !== top
  183. });
  184. };
  185. document.getElementById("chrome-url-of-local-file").textContent = fileUrl;
  186. document.getElementById("chrome-file-fallback").onchange = function () {
  187. const file = this.files[0];
  188. if (file) {
  189. const originalFilename = decodeURIComponent(fileUrl.split("/").pop());
  190. let originalUrl = fileUrl;
  191. if (originalFilename !== file.name) {
  192. const msg = "The selected file does not match the original file." + "\nOriginal: " + originalFilename + "\nSelected: " + file.name + "\nDo you want to open the selected file?";
  193. if (!confirm(msg)) {
  194. this.value = "";
  195. return;
  196. }
  197. originalUrl = "file:///fakepath/to/" + encodeURIComponent(file.name);
  198. }
  199. callback(URL.createObjectURL(file), file.size, originalUrl);
  200. overlayManager.close(dialog);
  201. }
  202. };
  203. overlayManager.open(dialog);
  204. });
  205. }
  206. if (window === top) {
  207. addEventListener("unload", function () {
  208. if (!isRuntimeAvailable()) {
  209. localStorage.setItem("unload-" + Date.now() + "-" + document.hidden + "-" + location.href, JSON.stringify(history.state));
  210. }
  211. });
  212. }
  213. let port;
  214. function setReferer(url, callback) {
  215. if (!port) {
  216. port = chrome.runtime.connect({
  217. name: "chromecom-referrer"
  218. });
  219. }
  220. port.onDisconnect.addListener(onDisconnect);
  221. port.onMessage.addListener(onMessage);
  222. port.postMessage({
  223. referer: window.history.state?.chromecomState,
  224. requestUrl: url
  225. });
  226. function onMessage(referer) {
  227. if (referer) {
  228. const state = window.history.state || {};
  229. state.chromecomState = referer;
  230. window.history.replaceState(state, "");
  231. }
  232. onCompleted();
  233. }
  234. function onDisconnect() {
  235. port = null;
  236. callback();
  237. }
  238. function onCompleted() {
  239. port.onDisconnect.removeListener(onDisconnect);
  240. port.onMessage.removeListener(onMessage);
  241. callback();
  242. }
  243. }
  244. const storageArea = chrome.storage.sync || chrome.storage.local;
  245. class ChromePreferences extends _preferences.BasePreferences {
  246. async _writeToStorage(prefObj) {
  247. return new Promise(resolve => {
  248. if (prefObj === this.defaults) {
  249. const keysToRemove = Object.keys(this.defaults);
  250. storageArea.remove(keysToRemove, function () {
  251. resolve();
  252. });
  253. } else {
  254. storageArea.set(prefObj, function () {
  255. resolve();
  256. });
  257. }
  258. });
  259. }
  260. async _readFromStorage(prefObj) {
  261. return new Promise(resolve => {
  262. const getPreferences = defaultPrefs => {
  263. if (chrome.runtime.lastError) {
  264. defaultPrefs = this.defaults;
  265. }
  266. storageArea.get(defaultPrefs, function (readPrefs) {
  267. resolve(readPrefs);
  268. });
  269. };
  270. if (chrome.storage.managed) {
  271. const defaultManagedPrefs = Object.assign({
  272. enableHandToolOnLoad: false,
  273. disableTextLayer: false,
  274. enhanceTextSelection: false,
  275. showPreviousViewOnLoad: true,
  276. disablePageMode: false
  277. }, this.defaults);
  278. chrome.storage.managed.get(defaultManagedPrefs, function (items) {
  279. items = items || defaultManagedPrefs;
  280. if (items.enableHandToolOnLoad && !items.cursorToolOnLoad) {
  281. items.cursorToolOnLoad = 1;
  282. }
  283. delete items.enableHandToolOnLoad;
  284. if (items.textLayerMode !== 1 && items.disableTextLayer) {
  285. items.textLayerMode = 0;
  286. }
  287. delete items.disableTextLayer;
  288. delete items.enhanceTextSelection;
  289. if (!items.showPreviousViewOnLoad && !items.viewOnLoad) {
  290. items.viewOnLoad = 1;
  291. }
  292. delete items.showPreviousViewOnLoad;
  293. delete items.disablePageMode;
  294. getPreferences(items);
  295. });
  296. } else {
  297. getPreferences(this.defaults);
  298. }
  299. });
  300. }
  301. }
  302. class ChromeExternalServices extends _app.DefaultExternalServices {
  303. static initPassiveLoading(callbacks) {
  304. ChromeCom.resolvePDFFile(_app_options.AppOptions.get("defaultUrl"), _app.PDFViewerApplication.overlayManager, function (url, length, originalUrl) {
  305. callbacks.onOpenWithURL(url, length, originalUrl);
  306. });
  307. }
  308. static createDownloadManager() {
  309. return new _download_manager.DownloadManager();
  310. }
  311. static createPreferences() {
  312. return new ChromePreferences();
  313. }
  314. static createL10n(options) {
  315. return new _genericl10n.GenericL10n(navigator.language);
  316. }
  317. static createScripting({
  318. sandboxBundleSrc
  319. }) {
  320. return new _generic_scripting.GenericScripting(sandboxBundleSrc);
  321. }
  322. }
  323. _app.PDFViewerApplication.externalServices = ChromeExternalServices;