chromecom.js 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392
  1. /**
  2. * @licstart The following is the entire license notice for the
  3. * Javascript code in this page
  4. *
  5. * Copyright 2020 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. {
  33. throw new Error('Module "pdfjs-web/chromecom" shall not be used outside CHROME build.');
  34. }
  35. const ChromeCom = {
  36. request(action, data, callback) {
  37. const message = {
  38. action,
  39. data
  40. };
  41. if (!chrome.runtime) {
  42. console.error("chrome.runtime is undefined.");
  43. if (callback) {
  44. callback();
  45. }
  46. } else if (callback) {
  47. chrome.runtime.sendMessage(message, callback);
  48. } else {
  49. chrome.runtime.sendMessage(message);
  50. }
  51. },
  52. resolvePDFFile(file, overlayManager, callback) {
  53. file = file.replace(/^drive:/i, "filesystem:" + location.origin + "/external/");
  54. if (/^https?:/.test(file)) {
  55. setReferer(file, function () {
  56. callback(file);
  57. });
  58. return;
  59. }
  60. if (/^file?:/.test(file)) {
  61. getEmbedderOrigin(function (origin) {
  62. if (origin && !/^file:|^chrome-extension:/.test(origin)) {
  63. _app.PDFViewerApplication.error("Blocked " + origin + " from loading " + file + ". Refused to load a local file in a non-local page " + "for security reasons.");
  64. return;
  65. }
  66. isAllowedFileSchemeAccess(function (isAllowedAccess) {
  67. if (isAllowedAccess) {
  68. callback(file);
  69. } else {
  70. requestAccessToLocalFile(file, overlayManager, callback);
  71. }
  72. });
  73. });
  74. return;
  75. }
  76. callback(file);
  77. }
  78. };
  79. exports.ChromeCom = ChromeCom;
  80. function getEmbedderOrigin(callback) {
  81. const origin = window === top ? location.origin : location.ancestorOrigins[0];
  82. if (origin === "null") {
  83. getParentOrigin(callback);
  84. } else {
  85. callback(origin);
  86. }
  87. }
  88. function getParentOrigin(callback) {
  89. ChromeCom.request("getParentOrigin", null, callback);
  90. }
  91. function isAllowedFileSchemeAccess(callback) {
  92. ChromeCom.request("isAllowedFileSchemeAccess", null, callback);
  93. }
  94. function isRuntimeAvailable() {
  95. try {
  96. if (chrome.runtime && chrome.runtime.getManifest()) {
  97. return true;
  98. }
  99. } catch (e) {}
  100. return false;
  101. }
  102. function reloadIfRuntimeIsUnavailable() {
  103. if (!isRuntimeAvailable()) {
  104. location.reload();
  105. }
  106. }
  107. let chromeFileAccessOverlayPromise;
  108. function requestAccessToLocalFile(fileUrl, overlayManager, callback) {
  109. let onCloseOverlay = null;
  110. if (top !== window) {
  111. window.addEventListener("focus", reloadIfRuntimeIsUnavailable);
  112. onCloseOverlay = function () {
  113. window.removeEventListener("focus", reloadIfRuntimeIsUnavailable);
  114. reloadIfRuntimeIsUnavailable();
  115. overlayManager.close("chromeFileAccessOverlay");
  116. };
  117. }
  118. if (!chromeFileAccessOverlayPromise) {
  119. chromeFileAccessOverlayPromise = overlayManager.register("chromeFileAccessOverlay", document.getElementById("chromeFileAccessOverlay"), onCloseOverlay, true);
  120. }
  121. chromeFileAccessOverlayPromise.then(function () {
  122. const iconPath = chrome.runtime.getManifest().icons[48];
  123. document.getElementById("chrome-pdfjs-logo-bg").style.backgroundImage = "url(" + chrome.runtime.getURL(iconPath) + ")";
  124. const i18nFileAccessLabel = {
  125. "am": "\u1208\u134b\u12ed\u120d \u12e9\u12a0\u122d\u12a4\u120d\u12ce\u127d \u1218\u12f3\u1228\u123b \u134d\u1240\u12f5",
  126. "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",
  127. "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",
  128. "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",
  129. "ca": "Permet l'acc\u00e9s als URL de fitxer",
  130. "cs": "Umo\u017enit p\u0159\u00edstup k adres\u00e1m URL soubor\u016f",
  131. "da": "Tillad adgang til webadresser p\u00e5 filer",
  132. "de": "Zugriff auf Datei-URLs zulassen",
  133. "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",
  134. "en-GB": "Allow access to file URLs",
  135. "es": "Permitir acceso a URL de archivo",
  136. "es-419": "Permitir el acceso a las URL del archivo",
  137. "et": "Luba juurdep\u00e4\u00e4s failide URL-idele",
  138. "fa": "\u200f\u0627\u062c\u0627\u0632\u0647\u0654 \u062f\u0633\u062a\u0631\u0633\u06cc \u0628\u0647 URL \u0647\u0627\u06cc \u0641\u0627\u06cc\u0644",
  139. "fi": "Salli tiedostojen URL-osoitteiden k\u00e4ytt\u00f6",
  140. "fil": "Payagan ang access na mag-file ng mga URL",
  141. "fr": "Autoriser l'acc\u00e8s aux URL de fichier",
  142. "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",
  143. "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",
  144. "hr": "Dozvoli pristup URL-ovima datoteke",
  145. "hu": "F\u00e1jl URL-ekhez val\u00f3 hozz\u00e1f\u00e9r\u00e9s enged\u00e9lyez\u00e9se",
  146. "id": "Izinkan akses ke URL file",
  147. "it": "Consenti l'accesso agli URL dei file",
  148. "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",
  149. "ja": "\u30d5\u30a1\u30a4\u30eb\u306e URL \u3078\u306e\u30a2\u30af\u30bb\u30b9\u3092\u8a31\u53ef\u3059\u308b",
  150. "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",
  151. "ko": "\ud30c\uc77c URL\uc5d0 \ub300\ud55c \uc561\uc138\uc2a4 \ud5c8\uc6a9",
  152. "lt": "Leisti pasiekti failo URL",
  153. "lv": "At\u013caut piek\u013cuvi faila vietr\u0101\u017eiem URL",
  154. "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",
  155. "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",
  156. "ms": "Membenarkan akses ke URL fail",
  157. "nl": "Toegang tot bestand-URL's toestaan",
  158. "no": "Tillat tilgang til filnettadresser",
  159. "pl": "Zezwalaj na dost\u0119p do adres\u00f3w URL plik\u00f3w",
  160. "pt-BR": "Permitir acesso aos URLs do arquivo",
  161. "pt-PT": "Permitir acesso a URLs de ficheiro",
  162. "ro": "Permite accesul la adresele URL de fi\u0219iere",
  163. "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",
  164. "sk": "Povoli\u0165 pr\u00edstup k webov\u00fdm adres\u00e1m s\u00faboru",
  165. "sl": "Dovoli dostop do URL-jev datoteke",
  166. "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",
  167. "sv": "Till\u00e5t \u00e5tkomst till webbadresser i filen",
  168. "sw": "Ruhusu kufikia URL za faili",
  169. "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",
  170. "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",
  171. "th": "\u0e2d\u0e19\u0e38\u0e0d\u0e32\u0e15\u0e43\u0e2b\u0e49\u0e40\u0e02\u0e49\u0e32\u0e16\u0e36\u0e07\u0e44\u0e1f\u0e25\u0e4c URL",
  172. "tr": "Dosya URL'lerine eri\u015fime izin ver",
  173. "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",
  174. "vi": "Cho ph\u00e9p truy c\u1eadp v\u00e0o c\u00e1c URL c\u1ee7a t\u1ec7p",
  175. "zh-CN": "\u5141\u8bb8\u8bbf\u95ee\u6587\u4ef6\u7f51\u5740",
  176. "zh-TW": "\u5141\u8a31\u5b58\u53d6\u6a94\u6848\u7db2\u5740"
  177. }[chrome.i18n.getUILanguage && chrome.i18n.getUILanguage()];
  178. if (i18nFileAccessLabel) {
  179. document.getElementById("chrome-file-access-label").textContent = i18nFileAccessLabel;
  180. }
  181. const link = document.getElementById("chrome-link-to-extensions-page");
  182. link.href = "chrome://extensions/?id=" + chrome.runtime.id;
  183. link.onclick = function (e) {
  184. e.preventDefault();
  185. ChromeCom.request("openExtensionsPageForFileAccess", {
  186. newTab: e.ctrlKey || e.metaKey || e.button === 1 || window !== top
  187. });
  188. };
  189. document.getElementById("chrome-url-of-local-file").textContent = fileUrl;
  190. document.getElementById("chrome-file-fallback").onchange = function () {
  191. const file = this.files[0];
  192. if (file) {
  193. const originalFilename = decodeURIComponent(fileUrl.split("/").pop());
  194. let originalUrl = fileUrl;
  195. if (originalFilename !== file.name) {
  196. const msg = "The selected file does not match the original file." + "\nOriginal: " + originalFilename + "\nSelected: " + file.name + "\nDo you want to open the selected file?";
  197. if (!confirm(msg)) {
  198. this.value = "";
  199. return;
  200. }
  201. originalUrl = "file:///fakepath/to/" + encodeURIComponent(file.name);
  202. }
  203. callback(URL.createObjectURL(file), file.size, originalUrl);
  204. overlayManager.close("chromeFileAccessOverlay");
  205. }
  206. };
  207. overlayManager.open("chromeFileAccessOverlay");
  208. });
  209. }
  210. if (window === top) {
  211. addEventListener("unload", function () {
  212. if (!isRuntimeAvailable()) {
  213. localStorage.setItem("unload-" + Date.now() + "-" + document.hidden + "-" + location.href, JSON.stringify(history.state));
  214. }
  215. });
  216. }
  217. let port;
  218. function setReferer(url, callback) {
  219. if (!port) {
  220. port = chrome.runtime.connect({
  221. name: "chromecom-referrer"
  222. });
  223. }
  224. port.onDisconnect.addListener(onDisconnect);
  225. port.onMessage.addListener(onMessage);
  226. port.postMessage({
  227. referer: window.history.state && window.history.state.chromecomState,
  228. requestUrl: url
  229. });
  230. function onMessage(referer) {
  231. if (referer) {
  232. const state = window.history.state || {};
  233. state.chromecomState = referer;
  234. window.history.replaceState(state, "");
  235. }
  236. onCompleted();
  237. }
  238. function onDisconnect() {
  239. port = null;
  240. callback();
  241. }
  242. function onCompleted() {
  243. port.onDisconnect.removeListener(onDisconnect);
  244. port.onMessage.removeListener(onMessage);
  245. callback();
  246. }
  247. }
  248. const storageArea = chrome.storage.sync || chrome.storage.local;
  249. class ChromePreferences extends _preferences.BasePreferences {
  250. async _writeToStorage(prefObj) {
  251. return new Promise(resolve => {
  252. if (prefObj === this.defaults) {
  253. const keysToRemove = Object.keys(this.defaults);
  254. storageArea.remove(keysToRemove, function () {
  255. resolve();
  256. });
  257. } else {
  258. storageArea.set(prefObj, function () {
  259. resolve();
  260. });
  261. }
  262. });
  263. }
  264. async _readFromStorage(prefObj) {
  265. return new Promise(resolve => {
  266. const getPreferences = defaultPrefs => {
  267. if (chrome.runtime.lastError) {
  268. defaultPrefs = this.defaults;
  269. }
  270. storageArea.get(defaultPrefs, function (readPrefs) {
  271. resolve(readPrefs);
  272. });
  273. };
  274. if (chrome.storage.managed) {
  275. const defaultManagedPrefs = Object.assign({
  276. enableHandToolOnLoad: false,
  277. disableTextLayer: false,
  278. enhanceTextSelection: false,
  279. showPreviousViewOnLoad: true,
  280. disablePageMode: false
  281. }, this.defaults);
  282. chrome.storage.managed.get(defaultManagedPrefs, function (items) {
  283. items = items || defaultManagedPrefs;
  284. if (items.enableHandToolOnLoad && !items.cursorToolOnLoad) {
  285. items.cursorToolOnLoad = 1;
  286. }
  287. delete items.enableHandToolOnLoad;
  288. if (items.textLayerMode !== 1) {
  289. if (items.disableTextLayer) {
  290. items.textLayerMode = 0;
  291. } else if (items.enhanceTextSelection) {
  292. items.textLayerMode = 2;
  293. }
  294. }
  295. delete items.disableTextLayer;
  296. delete items.enhanceTextSelection;
  297. if (!items.showPreviousViewOnLoad && !items.viewOnLoad) {
  298. items.viewOnLoad = 1;
  299. }
  300. delete items.showPreviousViewOnLoad;
  301. delete items.disablePageMode;
  302. getPreferences(items);
  303. });
  304. } else {
  305. getPreferences(this.defaults);
  306. }
  307. });
  308. }
  309. }
  310. class ChromeExternalServices extends _app.DefaultExternalServices {
  311. static initPassiveLoading(callbacks) {
  312. ChromeCom.resolvePDFFile(_app_options.AppOptions.get("defaultUrl"), _app.PDFViewerApplication.overlayManager, function (url, length, originalUrl) {
  313. callbacks.onOpenWithURL(url, length, originalUrl);
  314. });
  315. }
  316. static createDownloadManager(options) {
  317. return new _download_manager.DownloadManager(options);
  318. }
  319. static createPreferences() {
  320. return new ChromePreferences();
  321. }
  322. static createL10n(options) {
  323. return new _genericl10n.GenericL10n(navigator.language);
  324. }
  325. }
  326. _app.PDFViewerApplication.externalServices = ChromeExternalServices;