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