chrome-browser-polyfill.js 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. /*
  2. * Copyright 2010-2019 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. (() => {
  24. const FEATURE_TESTS = {};
  25. if (!this.browser && this.chrome) {
  26. const nativeAPI = this.chrome;
  27. this.__defineGetter__("browser", () => ({
  28. browserAction: {
  29. onClicked: {
  30. addListener: listener => nativeAPI.browserAction.onClicked.addListener(listener)
  31. },
  32. setBadgeText: options => new Promise((resolve, reject) => {
  33. if (!FEATURE_TESTS["browserAction.setBadgeText"] || !FEATURE_TESTS["browserAction.setBadgeText"].callbackNotSupported) {
  34. try {
  35. nativeAPI.browserAction.setBadgeText(options, () => {
  36. if (nativeAPI.runtime.lastError) {
  37. reject(nativeAPI.runtime.lastError);
  38. } else {
  39. resolve();
  40. }
  41. });
  42. } catch (error) {
  43. FEATURE_TESTS["browserAction.setBadgeText"] = { callbackNotSupported: true };
  44. }
  45. }
  46. if (FEATURE_TESTS["browserAction.setBadgeText"] && FEATURE_TESTS["browserAction.setBadgeText"].callbackNotSupported) {
  47. nativeAPI.browserAction.setBadgeText(options);
  48. if (nativeAPI.runtime.lastError) {
  49. reject(nativeAPI.runtime.lastError);
  50. } else {
  51. resolve();
  52. }
  53. }
  54. }),
  55. setBadgeBackgroundColor: options => new Promise((resolve, reject) => {
  56. if (!FEATURE_TESTS["browserAction.setBadgeBackgroundColor"] || !FEATURE_TESTS["browserAction.setBadgeBackgroundColor"].callbackNotSupported) {
  57. try {
  58. nativeAPI.browserAction.setBadgeBackgroundColor(options, () => {
  59. if (nativeAPI.runtime.lastError) {
  60. reject(nativeAPI.runtime.lastError);
  61. } else {
  62. resolve();
  63. }
  64. });
  65. } catch (error) {
  66. FEATURE_TESTS["browserAction.setBadgeBackgroundColor"] = { callbackNotSupported: true };
  67. }
  68. }
  69. if (FEATURE_TESTS["browserAction.setBadgeBackgroundColor"] && FEATURE_TESTS["browserAction.setBadgeBackgroundColor"].callbackNotSupported) {
  70. nativeAPI.browserAction.setBadgeBackgroundColor(options);
  71. if (nativeAPI.runtime.lastError) {
  72. reject(nativeAPI.runtime.lastError);
  73. } else {
  74. resolve();
  75. }
  76. }
  77. }),
  78. setTitle: options => new Promise((resolve, reject) => {
  79. if (!FEATURE_TESTS["browserAction.setTitle"] || !FEATURE_TESTS["browserAction.setTitle"].callbackNotSupported) {
  80. try {
  81. nativeAPI.browserAction.setTitle(options, () => {
  82. if (nativeAPI.runtime.lastError) {
  83. reject(nativeAPI.runtime.lastError);
  84. } else {
  85. resolve();
  86. }
  87. });
  88. } catch (error) {
  89. FEATURE_TESTS["browserAction.setTitle"] = { callbackNotSupported: true };
  90. }
  91. }
  92. if (FEATURE_TESTS["browserAction.setTitle"] && FEATURE_TESTS["browserAction.setTitle"].callbackNotSupported) {
  93. nativeAPI.browserAction.setTitle(options);
  94. if (nativeAPI.runtime.lastError) {
  95. reject(nativeAPI.runtime.lastError);
  96. } else {
  97. resolve();
  98. }
  99. }
  100. }),
  101. setIcon: options => new Promise((resolve, reject) => {
  102. if (!FEATURE_TESTS["browserAction.setIcon"] || !FEATURE_TESTS["browserAction.setIcon"].callbackNotSupported) {
  103. try {
  104. nativeAPI.browserAction.setIcon(options, () => {
  105. if (nativeAPI.runtime.lastError) {
  106. reject(nativeAPI.runtime.lastError);
  107. } else {
  108. resolve();
  109. }
  110. });
  111. } catch (error) {
  112. FEATURE_TESTS["browserAction.setIcon"] = { callbackNotSupported: true };
  113. }
  114. }
  115. if (FEATURE_TESTS["browserAction.setIcon"] && FEATURE_TESTS["browserAction.setIcon"].callbackNotSupported) {
  116. nativeAPI.browserAction.setIcon(options);
  117. if (nativeAPI.runtime.lastError) {
  118. reject(nativeAPI.runtime.lastError);
  119. } else {
  120. resolve();
  121. }
  122. }
  123. })
  124. },
  125. downloads: {
  126. download: options => new Promise((resolve, reject) => {
  127. nativeAPI.downloads.download(options, downloadId => {
  128. if (nativeAPI.runtime.lastError) {
  129. reject(nativeAPI.runtime.lastError);
  130. } else {
  131. resolve(downloadId);
  132. }
  133. });
  134. }),
  135. onChanged: {
  136. addListener: listener => nativeAPI.downloads.onChanged.addListener(listener),
  137. removeListener: listener => nativeAPI.downloads.onChanged.removeListener(listener)
  138. }
  139. },
  140. i18n: {
  141. getMessage: (messageName, substitutions) => nativeAPI.i18n.getMessage(messageName, substitutions)
  142. },
  143. menus: {
  144. onClicked: {
  145. addListener: listener => nativeAPI.contextMenus.onClicked.addListener(listener)
  146. },
  147. create: options => nativeAPI.contextMenus.create(options),
  148. update: (menuItemId, options) => new Promise((resolve, reject) => {
  149. nativeAPI.contextMenus.update(menuItemId, options, () => {
  150. if (nativeAPI.runtime.lastError) {
  151. reject(nativeAPI.runtime.lastError);
  152. } else {
  153. resolve();
  154. }
  155. });
  156. }),
  157. removeAll: () => new Promise((resolve, reject) => {
  158. nativeAPI.contextMenus.removeAll(() => {
  159. if (nativeAPI.runtime.lastError) {
  160. reject(nativeAPI.runtime.lastError);
  161. } else {
  162. resolve();
  163. }
  164. });
  165. })
  166. },
  167. runtime: {
  168. onMessage: {
  169. addListener: listener => nativeAPI.runtime.onMessage.addListener((message, sender, sendResponse) => {
  170. const response = listener(message, sender);
  171. if (response && typeof response.then == "function") {
  172. response
  173. .then(response => {
  174. if (response !== undefined) {
  175. try {
  176. sendResponse(response);
  177. } catch (error) {
  178. /* ignored */
  179. }
  180. }
  181. });
  182. return true;
  183. }
  184. }),
  185. removeListener: listener => nativeAPI.runtime.onMessage.removeListener(listener)
  186. },
  187. onMessageExternal: {
  188. addListener: listener => nativeAPI.runtime.onMessageExternal.addListener((message, sender, sendResponse) => {
  189. const response = listener(message, sender);
  190. if (response && typeof response.then == "function") {
  191. response
  192. .then(response => {
  193. if (response !== undefined) {
  194. try {
  195. sendResponse(response);
  196. } catch (error) {
  197. /* ignored */
  198. }
  199. }
  200. });
  201. return true;
  202. }
  203. })
  204. },
  205. sendMessage: message => new Promise((resolve, reject) =>
  206. nativeAPI.runtime.sendMessage(message, response => {
  207. if (nativeAPI.runtime.lastError) {
  208. if (nativeAPI.runtime.lastError.message == "The message port closed before a response was received.") {
  209. resolve();
  210. } else {
  211. reject(nativeAPI.runtime.lastError);
  212. }
  213. } else {
  214. resolve(response);
  215. }
  216. })
  217. ),
  218. getURL: (path) => nativeAPI.runtime.getURL(path),
  219. get lastError() {
  220. return nativeAPI.runtime.lastError;
  221. }
  222. },
  223. storage: {
  224. local: {
  225. set: value => new Promise((resolve, reject) => {
  226. nativeAPI.storage.local.set(value, () => {
  227. if (nativeAPI.runtime.lastError) {
  228. reject(nativeAPI.runtime.lastError);
  229. } else {
  230. resolve();
  231. }
  232. });
  233. }),
  234. get: () => new Promise((resolve, reject) => {
  235. nativeAPI.storage.local.get(null, value => {
  236. if (nativeAPI.runtime.lastError) {
  237. reject(nativeAPI.runtime.lastError);
  238. } else {
  239. resolve(value);
  240. }
  241. });
  242. }),
  243. clear: () => new Promise((resolve, reject) => {
  244. nativeAPI.storage.local.clear(() => {
  245. if (nativeAPI.runtime.lastError) {
  246. reject(nativeAPI.runtime.lastError);
  247. } else {
  248. resolve();
  249. }
  250. });
  251. }),
  252. remove: keys => new Promise((resolve, reject) => {
  253. nativeAPI.storage.local.remove(keys, () => {
  254. if (nativeAPI.runtime.lastError) {
  255. reject(nativeAPI.runtime.lastError);
  256. } else {
  257. resolve();
  258. }
  259. });
  260. })
  261. }
  262. },
  263. tabs: {
  264. onCreated: {
  265. addListener: listener => nativeAPI.tabs.onCreated.addListener(listener)
  266. },
  267. onActivated: {
  268. addListener: listener => nativeAPI.tabs.onActivated.addListener(listener)
  269. },
  270. onUpdated: {
  271. addListener: listener => nativeAPI.tabs.onUpdated.addListener(listener)
  272. },
  273. onRemoved: {
  274. addListener: listener => nativeAPI.tabs.onRemoved.addListener(listener)
  275. },
  276. executeScript: (tabId, details) => new Promise((resolve, reject) => {
  277. nativeAPI.tabs.executeScript(tabId, details, () => {
  278. if (nativeAPI.runtime.lastError) {
  279. reject(nativeAPI.runtime.lastError);
  280. } else {
  281. resolve();
  282. }
  283. });
  284. }),
  285. sendMessage: (tabId, message, options = {}) => new Promise((resolve, reject) =>
  286. nativeAPI.tabs.sendMessage(tabId, message, options, response => {
  287. if (nativeAPI.runtime.lastError) {
  288. if (nativeAPI.runtime.lastError.message == "The message port closed before a response was received.") {
  289. resolve();
  290. } else {
  291. reject(nativeAPI.runtime.lastError);
  292. }
  293. } else {
  294. resolve(response);
  295. }
  296. })
  297. ),
  298. query: options => new Promise((resolve, reject) => {
  299. nativeAPI.tabs.query(options, tabs => {
  300. if (nativeAPI.runtime.lastError) {
  301. reject(nativeAPI.runtime.lastError);
  302. } else {
  303. resolve(tabs);
  304. }
  305. });
  306. }),
  307. get: options => new Promise((resolve, reject) => {
  308. nativeAPI.tabs.get(options, tab => {
  309. if (nativeAPI.runtime.lastError) {
  310. reject(nativeAPI.runtime.lastError);
  311. } else {
  312. resolve(tab);
  313. }
  314. });
  315. })
  316. }
  317. }));
  318. }
  319. })();