chrome-api.js 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. const {ipcRenderer} = require('electron')
  2. const Event = require('./extensions/event')
  3. const url = require('url')
  4. let nextId = 0
  5. class Tab {
  6. constructor (tabId) {
  7. this.id = tabId
  8. }
  9. }
  10. class MessageSender {
  11. constructor (tabId, extensionId) {
  12. this.tab = tabId ? new Tab(tabId) : null
  13. this.id = extensionId
  14. this.url = `chrome-extension://${extensionId}`
  15. }
  16. }
  17. class Port {
  18. constructor (tabId, portId, extensionId, name) {
  19. this.tabId = tabId
  20. this.portId = portId
  21. this.disconnected = false
  22. this.name = name
  23. this.onDisconnect = new Event()
  24. this.onMessage = new Event()
  25. this.sender = new MessageSender(tabId, extensionId)
  26. ipcRenderer.once(`CHROME_PORT_DISCONNECT_${portId}`, () => {
  27. this._onDisconnect()
  28. })
  29. ipcRenderer.on(`CHROME_PORT_POSTMESSAGE_${portId}`, (event, message) => {
  30. const sendResponse = function () { console.error('sendResponse is not implemented') }
  31. this.onMessage.emit(message, this.sender, sendResponse)
  32. })
  33. }
  34. disconnect () {
  35. if (this.disconnected) return
  36. ipcRenderer.sendToAll(this.tabId, `CHROME_PORT_DISCONNECT_${this.portId}`)
  37. this._onDisconnect()
  38. }
  39. postMessage (message) {
  40. ipcRenderer.sendToAll(this.tabId, `CHROME_PORT_POSTMESSAGE_${this.portId}`, message)
  41. }
  42. _onDisconnect () {
  43. this.disconnected = true
  44. ipcRenderer.removeAllListeners(`CHROME_PORT_POSTMESSAGE_${this.portId}`)
  45. this.onDisconnect.emit()
  46. }
  47. }
  48. // Inject chrome API to the |context|
  49. exports.injectTo = function (extensionId, isBackgroundPage, context) {
  50. const chrome = context.chrome = context.chrome || {}
  51. ipcRenderer.on(`CHROME_RUNTIME_ONCONNECT_${extensionId}`, (event, tabId, portId, connectInfo) => {
  52. chrome.runtime.onConnect.emit(new Port(tabId, portId, extensionId, connectInfo.name))
  53. })
  54. ipcRenderer.on(`CHROME_RUNTIME_ONMESSAGE_${extensionId}`, (event, tabId, message) => {
  55. chrome.runtime.onMessage.emit(message, new MessageSender(tabId, extensionId))
  56. })
  57. ipcRenderer.on('CHROME_TABS_ONCREATED', (event, tabId) => {
  58. chrome.tabs.onCreated.emit(new Tab(tabId))
  59. })
  60. ipcRenderer.on('CHROME_TABS_ONREMOVED', (event, tabId) => {
  61. chrome.tabs.onRemoved.emit(tabId)
  62. })
  63. chrome.runtime = {
  64. id: extensionId,
  65. getURL: function (path) {
  66. return url.format({
  67. protocol: 'chrome-extension',
  68. slashes: true,
  69. hostname: extensionId,
  70. pathname: path
  71. })
  72. },
  73. connect (...args) {
  74. if (isBackgroundPage) {
  75. console.error('chrome.runtime.connect is not supported in background page')
  76. return
  77. }
  78. // Parse the optional args.
  79. let targetExtensionId = extensionId
  80. let connectInfo = {name: ''}
  81. if (args.length === 1) {
  82. connectInfo = args[0]
  83. } else if (args.length === 2) {
  84. [targetExtensionId, connectInfo] = args
  85. }
  86. const {tabId, portId} = ipcRenderer.sendSync('CHROME_RUNTIME_CONNECT', targetExtensionId, connectInfo)
  87. return new Port(tabId, portId, extensionId, connectInfo.name)
  88. },
  89. sendMessage (...args) {
  90. if (isBackgroundPage) {
  91. console.error('chrome.runtime.sendMessage is not supported in background page')
  92. return
  93. }
  94. // Parse the optional args.
  95. let targetExtensionId = extensionId
  96. let message
  97. if (args.length === 1) {
  98. message = args[0]
  99. } else if (args.length === 2) {
  100. // A case of not provide extension-id: (message, responseCallback)
  101. if (typeof args[1] === 'function') {
  102. console.error('responseCallback is not supported')
  103. message = args[0]
  104. } else {
  105. [targetExtensionId, message] = args
  106. }
  107. } else {
  108. console.error('options and responseCallback are not supported')
  109. }
  110. ipcRenderer.send('CHROME_RUNTIME_SENDMESSAGE', targetExtensionId, message)
  111. },
  112. onConnect: new Event(),
  113. onMessage: new Event(),
  114. onInstalled: new Event()
  115. }
  116. chrome.tabs = {
  117. executeScript (tabId, details, callback) {
  118. const requestId = ++nextId
  119. ipcRenderer.once(`CHROME_TABS_EXECUTESCRIPT_RESULT_${requestId}`, (event, result) => {
  120. callback([event.result])
  121. })
  122. ipcRenderer.send('CHROME_TABS_EXECUTESCRIPT', requestId, tabId, extensionId, details)
  123. },
  124. sendMessage (tabId, message, options, responseCallback) {
  125. if (responseCallback) {
  126. console.error('responseCallback is not supported')
  127. }
  128. ipcRenderer.send('CHROME_TABS_SEND_MESSAGE', tabId, extensionId, isBackgroundPage, message)
  129. },
  130. onUpdated: new Event(),
  131. onCreated: new Event(),
  132. onRemoved: new Event()
  133. }
  134. chrome.extension = {
  135. getURL: chrome.runtime.getURL,
  136. connect: chrome.runtime.connect,
  137. onConnect: chrome.runtime.onConnect,
  138. sendMessage: chrome.runtime.sendMessage,
  139. onMessage: chrome.runtime.onMessage
  140. }
  141. chrome.storage = require('./extensions/storage')
  142. chrome.pageAction = {
  143. show () {},
  144. hide () {},
  145. setTitle () {},
  146. getTitle () {},
  147. setIcon () {},
  148. setPopup () {},
  149. getPopup () {}
  150. }
  151. chrome.i18n = require('./extensions/i18n').setup(extensionId)
  152. chrome.webNavigation = require('./extensions/web-navigation').setup()
  153. }