feat: make ipc-message and ipc-message-sync events public (#16468)

This commit is contained in:
Milan Burda 2019-01-22 17:47:58 +01:00 committed by Michelle Tilley
parent dbc5f67dca
commit 6cba2c50a2
7 changed files with 82 additions and 26 deletions

View file

@ -673,6 +673,26 @@ Returns:
Emitted when the preload script `preloadPath` throws an unhandled exception `error`. Emitted when the preload script `preloadPath` throws an unhandled exception `error`.
#### Event: 'ipc-message'
Returns:
* `event` Event
* `channel` String
* `...args` any[]
Emitted when the renderer process sends an asynchronous message via `ipcRenderer.send()`.
#### Event: 'ipc-message-sync'
Returns:
* `event` Event
* `channel` String
* `...args` any[]
Emitted when the renderer process sends a synchronous message via `ipcRenderer.sendSync()`.
#### Event: 'desktop-capturer-get-sources' #### Event: 'desktop-capturer-get-sources'
Returns: Returns:

View file

@ -342,17 +342,19 @@ WebContents.prototype._init = function () {
this.capturePage = deprecate.promisify(this.capturePage, 2) this.capturePage = deprecate.promisify(this.capturePage, 2)
// Dispatch IPC messages to the ipc module. // Dispatch IPC messages to the ipc module.
this.on('ipc-message', function (event, [channel, ...args]) { this.on('-ipc-message', function (event, [channel, ...args]) {
this.emit('ipc-message', event, channel, ...args)
ipcMain.emit(channel, event, ...args) ipcMain.emit(channel, event, ...args)
}) })
this.on('ipc-message-sync', function (event, [channel, ...args]) { this.on('-ipc-message-sync', function (event, [channel, ...args]) {
Object.defineProperty(event, 'returnValue', { Object.defineProperty(event, 'returnValue', {
set: function (value) { set: function (value) {
return event.sendReply([value]) return event.sendReply([value])
}, },
get: function () {} get: function () {}
}) })
this.emit('ipc-message-sync', event, channel, ...args)
ipcMain.emit(channel, event, ...args) ipcMain.emit(channel, event, ...args)
}) })

View file

@ -8,11 +8,11 @@ const ipcRenderer = v8Util.getHiddenValue(global, 'ipc')
const internal = false const internal = false
ipcRenderer.send = function (...args) { ipcRenderer.send = function (...args) {
return binding.send('ipc-message', args) return binding.send('-ipc-message', args)
} }
ipcRenderer.sendSync = function (...args) { ipcRenderer.sendSync = function (...args) {
return binding.sendSync('ipc-message-sync', args)[0] return binding.sendSync('-ipc-message-sync', args)[0]
} }
ipcRenderer.sendToHost = function (...args) { ipcRenderer.sendToHost = function (...args) {

View file

@ -995,6 +995,34 @@ describe('webContents module', () => {
}) })
}) })
describe('ipc-message event', () => {
it('emits when the renderer process sends an asynchronous message', async () => {
const webContents = remote.getCurrentWebContents()
const promise = emittedOnce(webContents, 'ipc-message')
ipcRenderer.send('message', 'Hello World!')
const [, channel, message] = await promise
expect(channel).to.equal('message')
expect(message).to.equal('Hello World!')
})
})
describe('ipc-message-sync event', () => {
it('emits when the renderer process sends a synchronous message', async () => {
const webContents = remote.getCurrentWebContents()
const promise = emittedOnce(webContents, 'ipc-message-sync')
ipcRenderer.send('handle-next-ipc-message-sync', 'foobar')
const result = ipcRenderer.sendSync('message', 'Hello World!')
const [, channel, message] = await promise
expect(channel).to.equal('message')
expect(message).to.equal('Hello World!')
expect(result).to.equal('foobar')
})
})
describe('referrer', () => { describe('referrer', () => {
it('propagates referrer information to new target=_blank windows', (done) => { it('propagates referrer information to new target=_blank windows', (done) => {
const server = http.createServer((req, res) => { const server = http.createServer((req, res) => {

View file

@ -175,8 +175,8 @@ describe('chromium feature', () => {
session: ses session: ses
} }
}) })
w.webContents.on('ipc-message', (event, args) => { w.webContents.on('ipc-message', (event, channel, deviceId) => {
if (args[0] === 'deviceIds') deviceIds.push(args[1]) if (channel === 'deviceIds') deviceIds.push(deviceId)
if (deviceIds.length === 2) { if (deviceIds.length === 2) {
assert.notDeepStrictEqual(deviceIds[0], deviceIds[1]) assert.notDeepStrictEqual(deviceIds[0], deviceIds[1])
closeWindow(w).then(() => { closeWindow(w).then(() => {
@ -216,13 +216,13 @@ describe('chromium feature', () => {
partition: 'sw-file-scheme-spec' partition: 'sw-file-scheme-spec'
} }
}) })
w.webContents.on('ipc-message', (event, args) => { w.webContents.on('ipc-message', (event, channel, message) => {
if (args[0] === 'reload') { if (channel === 'reload') {
w.webContents.reload() w.webContents.reload()
} else if (args[0] === 'error') { } else if (channel === 'error') {
done(args[1]) done(message)
} else if (args[0] === 'response') { } else if (channel === 'response') {
assert.strictEqual(args[1], 'Hello from serviceWorker!') assert.strictEqual(message, 'Hello from serviceWorker!')
session.fromPartition('sw-file-scheme-spec').clearStorageData({ session.fromPartition('sw-file-scheme-spec').clearStorageData({
storages: ['serviceworkers'] storages: ['serviceworkers']
}, () => done()) }, () => done())
@ -255,13 +255,13 @@ describe('chromium feature', () => {
session: customSession session: customSession
} }
}) })
w.webContents.on('ipc-message', (event, args) => { w.webContents.on('ipc-message', (event, channel, message) => {
if (args[0] === 'reload') { if (channel === 'reload') {
w.webContents.reload() w.webContents.reload()
} else if (args[0] === 'error') { } else if (channel === 'error') {
done(`unexpected error : ${args[1]}`) done(`unexpected error : ${message}`)
} else if (args[0] === 'response') { } else if (channel === 'response') {
assert.strictEqual(args[1], 'Hello from serviceWorker!') assert.strictEqual(message, 'Hello from serviceWorker!')
customSession.clearStorageData({ customSession.clearStorageData({
storages: ['serviceworkers'] storages: ['serviceworkers']
}, () => { }, () => {
@ -298,8 +298,8 @@ describe('chromium feature', () => {
partition: 'geolocation-spec' partition: 'geolocation-spec'
} }
}) })
w.webContents.on('ipc-message', (event, args) => { w.webContents.on('ipc-message', (event, channel) => {
if (args[0] === 'success') { if (channel === 'success') {
done() done()
} else { } else {
done('unexpected response from geolocation api') done('unexpected response from geolocation api')
@ -584,18 +584,18 @@ describe('chromium feature', () => {
describe('window.opener', () => { describe('window.opener', () => {
const url = `file://${fixtures}/pages/window-opener.html` const url = `file://${fixtures}/pages/window-opener.html`
it('is null for main window', (done) => { it('is null for main window', async () => {
w = new BrowserWindow({ w = new BrowserWindow({
show: false, show: false,
webPreferences: { webPreferences: {
nodeIntegration: true nodeIntegration: true
} }
}) })
w.webContents.once('ipc-message', (event, args) => { const promise = emittedOnce(w.webContents, 'ipc-message')
assert.deepStrictEqual(args, ['opener', null])
done()
})
w.loadFile(path.join(fixtures, 'pages', 'window-opener.html')) w.loadFile(path.join(fixtures, 'pages', 'window-opener.html'))
const [, channel, opener] = await promise
expect(channel).to.equal('opener')
expect(opener).to.equal(null)
}) })
it('is not null for window opened by window.open', (done) => { it('is not null for window opened by window.open', (done) => {

View file

@ -235,6 +235,12 @@ app.on('ready', function () {
}) })
}) })
ipcMain.on('handle-next-ipc-message-sync', function (event, returnValue) {
event.sender.once('ipc-message-sync', (event, channel, args) => {
event.returnValue = returnValue
})
})
for (const eventName of [ for (const eventName of [
'remote-require', 'remote-require',
'remote-get-global', 'remote-get-global',

View file

@ -651,7 +651,7 @@ describe('<webview> tag', function () {
}) })
describe('ipc-message event', () => { describe('ipc-message event', () => {
it('emits when guest sends a ipc message to browser', async () => { it('emits when guest sends an ipc message to browser', async () => {
loadWebView(webview, { loadWebView(webview, {
nodeintegration: 'on', nodeintegration: 'on',
src: `file://${fixtures}/pages/ipc-message.html` src: `file://${fixtures}/pages/ipc-message.html`