chrome-browser-polyfill.js 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317
  1. /*
  2. * Copyright 2010-2019 Gildas Lormeau
  3. * contact : gildas.lormeau <at> gmail.com
  4. *
  5. * This file is part of SingleFile.
  6. *
  7. * SingleFile is free software: you can redistribute it and/or modify
  8. * it under the terms of the GNU Lesser General Public License as published by
  9. * the Free Software Foundation, either version 3 of the License, or
  10. * (at your option) any later version.
  11. *
  12. * SingleFile 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
  15. * GNU Lesser General Public License for more details.
  16. *
  17. * You should have received a copy of the GNU Lesser General Public License
  18. * along with SingleFile. If not, see <http://www.gnu.org/licenses/>.
  19. */
  20. /* global navigator */
  21. (() => {
  22. const isChrome = navigator.userAgent.includes("Chrome");
  23. const FEATURE_TESTS = {};
  24. if (isChrome && !this.browser && this.chrome) {
  25. const nativeAPI = this.chrome;
  26. this.__defineGetter__("browser", () => ({
  27. browserAction: {
  28. onClicked: {
  29. addListener: listener => nativeAPI.browserAction.onClicked.addListener(listener)
  30. },
  31. enable: tabId => nativeAPI.browserAction.enable(tabId),
  32. disable: tabId => nativeAPI.browserAction.disable(tabId),
  33. setBadgeText: options => new Promise((resolve, reject) => {
  34. if (nativeAPI.runtime.lastError) {
  35. reject(nativeAPI.runtime.lastError);
  36. } else {
  37. if (!FEATURE_TESTS["browserAction.setBadgeText"] || !FEATURE_TESTS["browserAction.setBadgeText"].callbackNotSupported) {
  38. try {
  39. nativeAPI.browserAction.setBadgeText(options, resolve);
  40. } catch (error) {
  41. FEATURE_TESTS["browserAction.setBadgeText"] = { callbackNotSupported: true };
  42. nativeAPI.browserAction.setBadgeText(options);
  43. resolve();
  44. }
  45. } else {
  46. nativeAPI.browserAction.setBadgeText(options);
  47. resolve();
  48. }
  49. }
  50. }),
  51. setBadgeBackgroundColor: options => new Promise((resolve, reject) => {
  52. if (nativeAPI.runtime.lastError) {
  53. reject(nativeAPI.runtime.lastError);
  54. } else {
  55. if (!FEATURE_TESTS["browserAction.setBadgeBackgroundColor"] || !FEATURE_TESTS["browserAction.setBadgeBackgroundColor"].callbackNotSupported) {
  56. try {
  57. nativeAPI.browserAction.setBadgeBackgroundColor(options, resolve);
  58. } catch (error) {
  59. FEATURE_TESTS["browserAction.setBadgeBackgroundColor"] = { callbackNotSupported: true };
  60. nativeAPI.browserAction.setBadgeBackgroundColor(options);
  61. resolve();
  62. }
  63. } else {
  64. nativeAPI.browserAction.setBadgeBackgroundColor(options);
  65. resolve();
  66. }
  67. }
  68. }),
  69. setTitle: options => new Promise((resolve, reject) => {
  70. if (nativeAPI.runtime.lastError) {
  71. reject(nativeAPI.runtime.lastError);
  72. } else {
  73. if (!FEATURE_TESTS["browserAction.setTitle"] || !FEATURE_TESTS["browserAction.setTitle"].callbackNotSupported) {
  74. try {
  75. nativeAPI.browserAction.setTitle(options, resolve);
  76. } catch (error) {
  77. FEATURE_TESTS["browserAction.setTitle"] = { callbackNotSupported: true };
  78. nativeAPI.browserAction.setTitle(options);
  79. resolve();
  80. }
  81. } else {
  82. nativeAPI.browserAction.setTitle(options);
  83. resolve();
  84. }
  85. }
  86. }),
  87. setIcon: options => new Promise((resolve, reject) => {
  88. if (nativeAPI.runtime.lastError) {
  89. reject(nativeAPI.runtime.lastError);
  90. } else {
  91. if (!FEATURE_TESTS["browserAction.setIcon"] || !FEATURE_TESTS["browserAction.setIcon"].callbackNotSupported) {
  92. try {
  93. nativeAPI.browserAction.setIcon(options, resolve);
  94. } catch (error) {
  95. FEATURE_TESTS["browserAction.setIcon"] = { callbackNotSupported: true };
  96. nativeAPI.browserAction.setIcon(options);
  97. resolve();
  98. }
  99. } else {
  100. nativeAPI.browserAction.setIcon(options);
  101. resolve();
  102. }
  103. }
  104. })
  105. },
  106. downloads: {
  107. download: options => new Promise((resolve, reject) => {
  108. nativeAPI.downloads.download(options, downloadId => {
  109. if (nativeAPI.runtime.lastError) {
  110. reject(nativeAPI.runtime.lastError);
  111. } else {
  112. resolve(downloadId);
  113. }
  114. });
  115. }),
  116. onChanged: {
  117. addListener: listener => nativeAPI.downloads.onChanged.addListener(listener),
  118. removeListener: listener => nativeAPI.downloads.onChanged.removeListener(listener)
  119. }
  120. },
  121. i18n: {
  122. getMessage: (messageName, substitutions) => nativeAPI.i18n.getMessage(messageName, substitutions)
  123. },
  124. menus: {
  125. onClicked: {
  126. addListener: listener => nativeAPI.contextMenus.onClicked.addListener(listener)
  127. },
  128. create: options => nativeAPI.contextMenus.create(options),
  129. update: (menuItemId, options) => new Promise((resolve, reject) => {
  130. nativeAPI.contextMenus.update(menuItemId, options, () => {
  131. if (nativeAPI.runtime.lastError) {
  132. reject(nativeAPI.runtime.lastError);
  133. } else {
  134. resolve();
  135. }
  136. });
  137. }),
  138. removeAll: () => new Promise((resolve, reject) => {
  139. nativeAPI.contextMenus.removeAll(() => {
  140. if (nativeAPI.runtime.lastError) {
  141. reject(nativeAPI.runtime.lastError);
  142. } else {
  143. resolve();
  144. }
  145. });
  146. })
  147. },
  148. runtime: {
  149. getBrowserInfo: () => Promise.resolve({ name: isChrome ? "Chrome" : "Unknown" }),
  150. onMessage: {
  151. addListener: listener => nativeAPI.runtime.onMessage.addListener((message, sender, sendResponse) => {
  152. const response = listener(message, sender);
  153. if (response && typeof response.then == "function") {
  154. response
  155. .then(response => {
  156. if (response !== undefined) {
  157. try {
  158. sendResponse(response);
  159. } catch (error) {
  160. /* ignored */
  161. }
  162. }
  163. });
  164. return true;
  165. }
  166. }),
  167. removeListener: listener => nativeAPI.runtime.onMessage.removeListener(listener)
  168. },
  169. onMessageExternal: {
  170. addListener: listener => nativeAPI.runtime.onMessageExternal.addListener((message, sender, sendResponse) => {
  171. const response = listener(message, sender);
  172. if (response && typeof response.then == "function") {
  173. response
  174. .then(response => {
  175. if (response !== undefined) {
  176. try {
  177. sendResponse(response);
  178. } catch (error) {
  179. /* ignored */
  180. }
  181. }
  182. });
  183. return true;
  184. }
  185. })
  186. },
  187. sendMessage: message => new Promise((resolve, reject) =>
  188. nativeAPI.runtime.sendMessage(message, response => {
  189. if (nativeAPI.runtime.lastError) {
  190. if (nativeAPI.runtime.lastError.message == "The message port closed before a response was received.") {
  191. resolve();
  192. } else {
  193. reject(nativeAPI.runtime.lastError);
  194. }
  195. } else {
  196. resolve(response);
  197. }
  198. })
  199. ),
  200. getBackgroundPage: () => new Promise((resolve, reject) =>
  201. nativeAPI.runtime.getBackgroundPage(bgPage => {
  202. if (nativeAPI.runtime.lastError) {
  203. reject(nativeAPI.runtime.lastError);
  204. } else {
  205. resolve(bgPage);
  206. }
  207. })
  208. ),
  209. getURL: (path) => nativeAPI.runtime.getURL(path),
  210. get lastError() {
  211. return nativeAPI.runtime.lastError;
  212. }
  213. },
  214. storage: {
  215. local: {
  216. set: value => new Promise((resolve, reject) => {
  217. nativeAPI.storage.local.set(value, () => {
  218. if (nativeAPI.runtime.lastError) {
  219. reject(nativeAPI.runtime.lastError);
  220. } else {
  221. resolve();
  222. }
  223. });
  224. }),
  225. get: () => new Promise((resolve, reject) => {
  226. nativeAPI.storage.local.get(null, value => {
  227. if (nativeAPI.runtime.lastError) {
  228. reject(nativeAPI.runtime.lastError);
  229. } else {
  230. resolve(value);
  231. }
  232. });
  233. }),
  234. clear: () => new Promise((resolve, reject) => {
  235. nativeAPI.storage.local.clear(() => {
  236. if (nativeAPI.runtime.lastError) {
  237. reject(nativeAPI.runtime.lastError);
  238. } else {
  239. resolve();
  240. }
  241. });
  242. }),
  243. remove: keys => new Promise((resolve, reject) => {
  244. nativeAPI.storage.local.remove(keys, () => {
  245. if (nativeAPI.runtime.lastError) {
  246. reject(nativeAPI.runtime.lastError);
  247. } else {
  248. resolve();
  249. }
  250. });
  251. })
  252. }
  253. },
  254. tabs: {
  255. onCreated: {
  256. addListener: listener => nativeAPI.tabs.onCreated.addListener(listener)
  257. },
  258. onActivated: {
  259. addListener: listener => nativeAPI.tabs.onActivated.addListener(listener)
  260. },
  261. onUpdated: {
  262. addListener: listener => nativeAPI.tabs.onUpdated.addListener(listener)
  263. },
  264. onRemoved: {
  265. addListener: listener => nativeAPI.tabs.onRemoved.addListener(listener)
  266. },
  267. executeScript: (tabId, details) => new Promise((resolve, reject) => {
  268. nativeAPI.tabs.executeScript(tabId, details, () => {
  269. if (nativeAPI.runtime.lastError) {
  270. reject(nativeAPI.runtime.lastError);
  271. } else {
  272. resolve();
  273. }
  274. });
  275. }),
  276. sendMessage: (tabId, message, options = {}) => new Promise((resolve, reject) =>
  277. nativeAPI.tabs.sendMessage(tabId, message, options, response => {
  278. if (nativeAPI.runtime.lastError) {
  279. if (nativeAPI.runtime.lastError.message == "The message port closed before a response was received.") {
  280. resolve();
  281. } else {
  282. reject(nativeAPI.runtime.lastError);
  283. }
  284. } else {
  285. resolve(response);
  286. }
  287. })
  288. ),
  289. query: options => new Promise((resolve, reject) => {
  290. nativeAPI.tabs.query(options, tabs => {
  291. if (nativeAPI.runtime.lastError) {
  292. reject(nativeAPI.runtime.lastError);
  293. } else {
  294. resolve(tabs);
  295. }
  296. });
  297. }),
  298. get: options => new Promise((resolve, reject) => {
  299. nativeAPI.tabs.get(options, tab => {
  300. if (nativeAPI.runtime.lastError) {
  301. reject(nativeAPI.runtime.lastError);
  302. } else {
  303. resolve(tab);
  304. }
  305. });
  306. })
  307. }
  308. }));
  309. }
  310. })();