custom-browser-polyfill.js 9.7 KB

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