chrome-browser-polyfill.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550
  1. /*
  2. * Copyright 2010-2020 Gildas Lormeau
  3. * contact : gildas.lormeau <at> gmail.com
  4. *
  5. * This file is part of SingleFile.
  6. *
  7. * The code in this file is free software: you can redistribute it and/or
  8. * modify it under the terms of the GNU Affero General Public License
  9. * (GNU AGPL) as published by the Free Software Foundation, either version 3
  10. * of the License, or (at your option) any later version.
  11. *
  12. * The code in this file is distributed in the hope that it will be useful,
  13. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero
  15. * General Public License for more details.
  16. *
  17. * As additional permission under GNU AGPL version 3 section 7, you may
  18. * distribute UNMODIFIED VERSIONS OF THIS file without the copy of the GNU
  19. * AGPL normally required by section 4, provided you include this license
  20. * notice and a URL through which recipients can access the Corresponding
  21. * Source.
  22. */
  23. /* global globalThis, window */
  24. if (typeof globalThis == "undefined") {
  25. window.globalThis = window;
  26. }
  27. (() => {
  28. const FEATURE_TESTS = {};
  29. const NON_COMPLIANT_IMPLEMENTATION = globalThis.origin && globalThis.origin.startsWith("safari-web-extension://");
  30. if ((!globalThis.browser || NON_COMPLIANT_IMPLEMENTATION) && globalThis.chrome) {
  31. const nativeAPI = globalThis.chrome;
  32. globalThis.__defineGetter__("browser", () => ({
  33. browserAction: {
  34. onClicked: {
  35. addListener: listener => nativeAPI.browserAction.onClicked.addListener(listener)
  36. },
  37. setBadgeText: options => new Promise((resolve, reject) => {
  38. if (!FEATURE_TESTS["browserAction.setBadgeText"] || !FEATURE_TESTS["browserAction.setBadgeText"].callbackNotSupported) {
  39. try {
  40. nativeAPI.browserAction.setBadgeText(options, () => {
  41. if (nativeAPI.runtime.lastError) {
  42. reject(nativeAPI.runtime.lastError);
  43. } else {
  44. resolve();
  45. }
  46. });
  47. } catch (error) {
  48. FEATURE_TESTS["browserAction.setBadgeText"] = { callbackNotSupported: true };
  49. }
  50. }
  51. if (FEATURE_TESTS["browserAction.setBadgeText"] && FEATURE_TESTS["browserAction.setBadgeText"].callbackNotSupported) {
  52. nativeAPI.browserAction.setBadgeText(options);
  53. if (nativeAPI.runtime.lastError) {
  54. reject(nativeAPI.runtime.lastError);
  55. } else {
  56. resolve();
  57. }
  58. }
  59. }),
  60. setBadgeBackgroundColor: options => new Promise((resolve, reject) => {
  61. if (!FEATURE_TESTS["browserAction.setBadgeBackgroundColor"] || !FEATURE_TESTS["browserAction.setBadgeBackgroundColor"].callbackNotSupported) {
  62. try {
  63. nativeAPI.browserAction.setBadgeBackgroundColor(options, () => {
  64. if (nativeAPI.runtime.lastError) {
  65. reject(nativeAPI.runtime.lastError);
  66. } else {
  67. resolve();
  68. }
  69. });
  70. } catch (error) {
  71. FEATURE_TESTS["browserAction.setBadgeBackgroundColor"] = { callbackNotSupported: true };
  72. }
  73. }
  74. if (FEATURE_TESTS["browserAction.setBadgeBackgroundColor"] && FEATURE_TESTS["browserAction.setBadgeBackgroundColor"].callbackNotSupported) {
  75. nativeAPI.browserAction.setBadgeBackgroundColor(options);
  76. if (nativeAPI.runtime.lastError) {
  77. reject(nativeAPI.runtime.lastError);
  78. } else {
  79. resolve();
  80. }
  81. }
  82. }),
  83. setTitle: options => new Promise((resolve, reject) => {
  84. if (!FEATURE_TESTS["browserAction.setTitle"] || !FEATURE_TESTS["browserAction.setTitle"].callbackNotSupported) {
  85. try {
  86. nativeAPI.browserAction.setTitle(options, () => {
  87. if (nativeAPI.runtime.lastError) {
  88. reject(nativeAPI.runtime.lastError);
  89. } else {
  90. resolve();
  91. }
  92. });
  93. } catch (error) {
  94. FEATURE_TESTS["browserAction.setTitle"] = { callbackNotSupported: true };
  95. }
  96. }
  97. if (FEATURE_TESTS["browserAction.setTitle"] && FEATURE_TESTS["browserAction.setTitle"].callbackNotSupported) {
  98. nativeAPI.browserAction.setTitle(options);
  99. if (nativeAPI.runtime.lastError) {
  100. reject(nativeAPI.runtime.lastError);
  101. } else {
  102. resolve();
  103. }
  104. }
  105. }),
  106. setIcon: options => new Promise((resolve, reject) => {
  107. if (!FEATURE_TESTS["browserAction.setIcon"] || !FEATURE_TESTS["browserAction.setIcon"].callbackNotSupported) {
  108. try {
  109. nativeAPI.browserAction.setIcon(options, () => {
  110. if (nativeAPI.runtime.lastError) {
  111. reject(nativeAPI.runtime.lastError);
  112. } else {
  113. resolve();
  114. }
  115. });
  116. } catch (error) {
  117. FEATURE_TESTS["browserAction.setIcon"] = { callbackNotSupported: true };
  118. }
  119. }
  120. if (FEATURE_TESTS["browserAction.setIcon"] && FEATURE_TESTS["browserAction.setIcon"].callbackNotSupported) {
  121. nativeAPI.browserAction.setIcon(options);
  122. if (nativeAPI.runtime.lastError) {
  123. reject(nativeAPI.runtime.lastError);
  124. } else {
  125. resolve();
  126. }
  127. }
  128. })
  129. },
  130. bookmarks: {
  131. get: id => new Promise((resolve, reject) => {
  132. nativeAPI.bookmarks.get(id, result => {
  133. if (nativeAPI.runtime.lastError) {
  134. reject(nativeAPI.runtime.lastError);
  135. } else {
  136. resolve(result);
  137. }
  138. });
  139. }),
  140. onCreated: {
  141. addListener: listener => nativeAPI.bookmarks.onCreated.addListener(listener),
  142. removeListener: listener => nativeAPI.bookmarks.onCreated.removeListener(listener)
  143. },
  144. onChanged: {
  145. addListener: listener => nativeAPI.bookmarks.onChanged.addListener(listener),
  146. removeListener: listener => nativeAPI.bookmarks.onChanged.removeListener(listener)
  147. },
  148. onMoved: {
  149. addListener: listener => nativeAPI.bookmarks.onMoved.addListener(listener),
  150. removeListener: listener => nativeAPI.bookmarks.onMoved.removeListener(listener)
  151. },
  152. update: (id, changes) => new Promise((resolve, reject) => {
  153. nativeAPI.bookmarks.update(id, changes, node => {
  154. if (nativeAPI.runtime.lastError) {
  155. reject(nativeAPI.runtime.lastError);
  156. } else {
  157. resolve(node);
  158. }
  159. });
  160. })
  161. },
  162. commands: {
  163. onCommand: {
  164. addListener: listener => nativeAPI.commands.onCommand.addListener(listener)
  165. }
  166. },
  167. downloads: {
  168. download: options => new Promise((resolve, reject) => {
  169. nativeAPI.downloads.download(options, downloadId => {
  170. if (nativeAPI.runtime.lastError) {
  171. reject(nativeAPI.runtime.lastError);
  172. } else {
  173. resolve(downloadId);
  174. }
  175. });
  176. }),
  177. onChanged: {
  178. addListener: listener => nativeAPI.downloads.onChanged.addListener(listener),
  179. removeListener: listener => nativeAPI.downloads.onChanged.removeListener(listener)
  180. },
  181. search: query => new Promise((resolve, reject) => {
  182. nativeAPI.downloads.search(query, downloadItems => {
  183. if (nativeAPI.runtime.lastError) {
  184. reject(nativeAPI.runtime.lastError);
  185. } else {
  186. resolve(downloadItems);
  187. }
  188. });
  189. })
  190. },
  191. i18n: {
  192. getUILanguage: () => nativeAPI.i18n.getUILanguage(),
  193. getMessage: (messageName, substitutions) => nativeAPI.i18n.getMessage(messageName, substitutions)
  194. },
  195. identity: {
  196. getRedirectURL() {
  197. return nativeAPI.identity.getRedirectURL();
  198. },
  199. get getAuthToken() {
  200. return nativeAPI.identity && nativeAPI.identity.getAuthToken && (details => new Promise((resolve, reject) =>
  201. nativeAPI.identity.getAuthToken(details, token => {
  202. if (nativeAPI.runtime.lastError) {
  203. reject(nativeAPI.runtime.lastError);
  204. } else {
  205. resolve(token);
  206. }
  207. })
  208. ));
  209. },
  210. get launchWebAuthFlow() {
  211. return nativeAPI.identity && nativeAPI.identity.launchWebAuthFlow && (options => new Promise((resolve, reject) => {
  212. nativeAPI.identity.launchWebAuthFlow(options, responseUrl => {
  213. if (nativeAPI.runtime.lastError) {
  214. reject(nativeAPI.runtime.lastError);
  215. } else {
  216. resolve(responseUrl);
  217. }
  218. });
  219. }));
  220. },
  221. get removeCachedAuthToken() {
  222. return nativeAPI.identity && nativeAPI.identity.removeCachedAuthToken && (details => new Promise((resolve, reject) =>
  223. nativeAPI.identity.removeCachedAuthToken(details, () => {
  224. if (nativeAPI.runtime.lastError) {
  225. reject(nativeAPI.runtime.lastError);
  226. } else {
  227. resolve();
  228. }
  229. })
  230. ));
  231. }
  232. },
  233. menus: {
  234. onClicked: {
  235. addListener: listener => nativeAPI.contextMenus.onClicked.addListener(listener)
  236. },
  237. create: options => new Promise((resolve, reject) => {
  238. nativeAPI.contextMenus.create(options, () => {
  239. if (nativeAPI.runtime.lastError) {
  240. reject(nativeAPI.runtime.lastError);
  241. } else {
  242. resolve();
  243. }
  244. });
  245. }),
  246. update: (menuItemId, options) => new Promise((resolve, reject) => {
  247. nativeAPI.contextMenus.update(menuItemId, options, () => {
  248. if (nativeAPI.runtime.lastError) {
  249. reject(nativeAPI.runtime.lastError);
  250. } else {
  251. resolve();
  252. }
  253. });
  254. }),
  255. removeAll: () => new Promise((resolve, reject) => {
  256. nativeAPI.contextMenus.removeAll(() => {
  257. if (nativeAPI.runtime.lastError) {
  258. reject(nativeAPI.runtime.lastError);
  259. } else {
  260. resolve();
  261. }
  262. });
  263. })
  264. },
  265. permissions: {
  266. request: permissions => new Promise((resolve, reject) => {
  267. nativeAPI.permissions.request(permissions, result => {
  268. if (nativeAPI.runtime.lastError) {
  269. reject(nativeAPI.runtime.lastError);
  270. } else {
  271. resolve(result);
  272. }
  273. });
  274. }),
  275. remove: permissions => new Promise((resolve, reject) => {
  276. nativeAPI.permissions.remove(permissions, result => {
  277. if (nativeAPI.runtime.lastError) {
  278. reject(nativeAPI.runtime.lastError);
  279. } else {
  280. resolve(result);
  281. }
  282. });
  283. })
  284. },
  285. runtime: {
  286. sendNativeMessage: (application, message) => new Promise((resolve, reject) => {
  287. nativeAPI.runtime.sendNativeMessage(application, message, result => {
  288. if (nativeAPI.runtime.lastError) {
  289. reject(nativeAPI.runtime.lastError);
  290. } else {
  291. resolve(result);
  292. }
  293. });
  294. }),
  295. getManifest: () => nativeAPI.runtime.getManifest(),
  296. onMessage: {
  297. addListener: listener => nativeAPI.runtime.onMessage.addListener((message, sender, sendResponse) => {
  298. const response = listener(message, sender);
  299. if (response && typeof response.then == "function") {
  300. response
  301. .then(response => {
  302. if (response !== undefined) {
  303. try {
  304. sendResponse(response);
  305. } catch (error) {
  306. // ignored
  307. }
  308. }
  309. });
  310. return true;
  311. }
  312. }),
  313. removeListener: listener => nativeAPI.runtime.onMessage.removeListener(listener)
  314. },
  315. onMessageExternal: {
  316. addListener: listener => nativeAPI.runtime.onMessageExternal.addListener((message, sender, sendResponse) => {
  317. const response = listener(message, sender);
  318. if (response && typeof response.then == "function") {
  319. response
  320. .then(response => {
  321. if (response !== undefined) {
  322. try {
  323. sendResponse(response);
  324. } catch (error) {
  325. // ignored
  326. }
  327. }
  328. });
  329. return true;
  330. }
  331. })
  332. },
  333. onInstalled: {
  334. addListener: listener => nativeAPI.runtime.onInstalled.addListener(listener)
  335. },
  336. sendMessage: message => new Promise((resolve, reject) => {
  337. nativeAPI.runtime.sendMessage(message, response => {
  338. if (nativeAPI.runtime.lastError) {
  339. reject(nativeAPI.runtime.lastError);
  340. } else {
  341. resolve(response);
  342. }
  343. });
  344. if (nativeAPI.runtime.lastError) {
  345. reject(nativeAPI.runtime.lastError);
  346. }
  347. }),
  348. getURL: (path) => nativeAPI.runtime.getURL(path),
  349. get lastError() {
  350. return nativeAPI.runtime.lastError;
  351. }
  352. },
  353. storage: {
  354. local: {
  355. set: value => new Promise((resolve, reject) => {
  356. nativeAPI.storage.local.set(value, () => {
  357. if (nativeAPI.runtime.lastError) {
  358. reject(nativeAPI.runtime.lastError);
  359. } else {
  360. resolve();
  361. }
  362. });
  363. }),
  364. get: keys => new Promise((resolve, reject) => {
  365. nativeAPI.storage.local.get(keys, value => {
  366. if (nativeAPI.runtime.lastError) {
  367. reject(nativeAPI.runtime.lastError);
  368. } else {
  369. resolve(value);
  370. }
  371. });
  372. }),
  373. clear: () => new Promise((resolve, reject) => {
  374. nativeAPI.storage.local.clear(() => {
  375. if (nativeAPI.runtime.lastError) {
  376. reject(nativeAPI.runtime.lastError);
  377. } else {
  378. resolve();
  379. }
  380. });
  381. }),
  382. remove: keys => new Promise((resolve, reject) => {
  383. nativeAPI.storage.local.remove(keys, () => {
  384. if (nativeAPI.runtime.lastError) {
  385. reject(nativeAPI.runtime.lastError);
  386. } else {
  387. resolve();
  388. }
  389. });
  390. })
  391. },
  392. sync: {
  393. set: value => new Promise((resolve, reject) => {
  394. nativeAPI.storage.sync.set(value, () => {
  395. if (nativeAPI.runtime.lastError) {
  396. reject(nativeAPI.runtime.lastError);
  397. } else {
  398. resolve();
  399. }
  400. });
  401. }),
  402. get: keys => new Promise((resolve, reject) => {
  403. nativeAPI.storage.sync.get(keys, value => {
  404. if (nativeAPI.runtime.lastError) {
  405. reject(nativeAPI.runtime.lastError);
  406. } else {
  407. resolve(value);
  408. }
  409. });
  410. }),
  411. clear: () => new Promise((resolve, reject) => {
  412. nativeAPI.storage.sync.clear(() => {
  413. if (nativeAPI.runtime.lastError) {
  414. reject(nativeAPI.runtime.lastError);
  415. } else {
  416. resolve();
  417. }
  418. });
  419. }),
  420. remove: keys => new Promise((resolve, reject) => {
  421. nativeAPI.storage.sync.remove(keys, () => {
  422. if (nativeAPI.runtime.lastError) {
  423. reject(nativeAPI.runtime.lastError);
  424. } else {
  425. resolve();
  426. }
  427. });
  428. })
  429. }
  430. },
  431. tabs: {
  432. onCreated: {
  433. addListener: listener => nativeAPI.tabs.onCreated.addListener(listener)
  434. },
  435. onActivated: {
  436. addListener: listener => nativeAPI.tabs.onActivated.addListener(listener)
  437. },
  438. onUpdated: {
  439. addListener: listener => nativeAPI.tabs.onUpdated.addListener(listener),
  440. removeListener: listener => nativeAPI.tabs.onUpdated.removeListener(listener)
  441. },
  442. onRemoved: {
  443. addListener: listener => nativeAPI.tabs.onRemoved.addListener(listener),
  444. removeListener: listener => nativeAPI.tabs.onRemoved.removeListener(listener)
  445. },
  446. onReplaced: {
  447. addListener: listener => nativeAPI.tabs.onReplaced.addListener(listener),
  448. removeListener: listener => nativeAPI.tabs.onReplaced.removeListener(listener)
  449. },
  450. captureVisibleTab: (windowId, options) => new Promise((resolve, reject) => {
  451. nativeAPI.tabs.captureVisibleTab(windowId, options, dataUrl => {
  452. if (nativeAPI.runtime.lastError) {
  453. reject(nativeAPI.runtime.lastError);
  454. } else {
  455. resolve(dataUrl);
  456. }
  457. });
  458. }),
  459. executeScript: (tabId, details) => new Promise((resolve, reject) => {
  460. nativeAPI.tabs.executeScript(tabId, details, () => {
  461. if (nativeAPI.runtime.lastError) {
  462. reject(nativeAPI.runtime.lastError);
  463. } else {
  464. resolve();
  465. }
  466. });
  467. }),
  468. sendMessage: (tabId, message, options = {}) => new Promise((resolve, reject) => {
  469. nativeAPI.tabs.sendMessage(tabId, message, options, response => {
  470. if (nativeAPI.runtime.lastError) {
  471. reject(nativeAPI.runtime.lastError);
  472. } else {
  473. resolve(response);
  474. }
  475. });
  476. if (nativeAPI.runtime.lastError) {
  477. reject(nativeAPI.runtime.lastError);
  478. }
  479. }),
  480. query: options => new Promise((resolve, reject) => {
  481. nativeAPI.tabs.query(options, tabs => {
  482. if (nativeAPI.runtime.lastError) {
  483. reject(nativeAPI.runtime.lastError);
  484. } else {
  485. resolve(tabs);
  486. }
  487. });
  488. }),
  489. create: createProperties => new Promise((resolve, reject) => {
  490. nativeAPI.tabs.create(createProperties, tab => {
  491. if (nativeAPI.runtime.lastError) {
  492. reject(nativeAPI.runtime.lastError);
  493. } else {
  494. resolve(tab);
  495. }
  496. });
  497. }),
  498. get: options => new Promise((resolve, reject) => {
  499. nativeAPI.tabs.get(options, tab => {
  500. if (nativeAPI.runtime.lastError) {
  501. reject(nativeAPI.runtime.lastError);
  502. } else {
  503. resolve(tab);
  504. }
  505. });
  506. }),
  507. remove: tabId => new Promise((resolve, reject) => {
  508. nativeAPI.tabs.remove(tabId, () => {
  509. if (nativeAPI.runtime.lastError) {
  510. reject(nativeAPI.runtime.lastError);
  511. } else {
  512. resolve();
  513. }
  514. });
  515. }),
  516. update: (tabId, updateProperties) => new Promise((resolve, reject) => {
  517. nativeAPI.tabs.update(tabId, updateProperties, tab => {
  518. if (nativeAPI.runtime.lastError) {
  519. reject(nativeAPI.runtime.lastError);
  520. } else {
  521. resolve(tab);
  522. }
  523. });
  524. })
  525. },
  526. devtools: nativeAPI.devtools && {
  527. inspectedWindow: nativeAPI.devtools.inspectedWindow && {
  528. onResourceContentCommitted: nativeAPI.devtools.inspectedWindow.onResourceContentCommitted && {
  529. addListener: listener => nativeAPI.devtools.inspectedWindow.onResourceContentCommitted.addListener(listener)
  530. },
  531. get tabId() {
  532. return nativeAPI.devtools.inspectedWindow.tabId;
  533. }
  534. }
  535. },
  536. webRequest: {
  537. onBeforeSendHeaders: {
  538. addListener: (listener, filters, extraInfoSpec) => nativeAPI.webRequest.onBeforeSendHeaders.addListener(listener, filters, extraInfoSpec),
  539. removeListener: listener => nativeAPI.webRequest.onBeforeSendHeaders.removeListener(listener)
  540. }
  541. }
  542. }));
  543. }
  544. })();