From 99ee1fc0eba003e6d5791e11880fa6c06e509a59 Mon Sep 17 00:00:00 2001 From: Milan Burda Date: Wed, 12 Jan 2022 08:41:20 +0100 Subject: [PATCH] fix: will-attach-webview handler modifying params.instanceId does not break (#32386) --- lib/browser/guest-view-manager.ts | 30 +++++++++++++++----------- lib/renderer/web-view/web-view-impl.ts | 4 +--- spec/static/main.js | 6 ++++++ spec/webview-spec.js | 8 +++++++ typings/internal-electron.d.ts | 2 +- 5 files changed, 34 insertions(+), 16 deletions(-) diff --git a/lib/browser/guest-view-manager.ts b/lib/browser/guest-view-manager.ts index fd6a0ac0d370..c973586e807d 100644 --- a/lib/browser/guest-view-manager.ts +++ b/lib/browser/guest-view-manager.ts @@ -73,6 +73,17 @@ function makeWebPreferences (embedder: Electron.WebContents, params: Record) { + const opts: Electron.LoadURLOptions = {}; + if (params.httpreferrer) { + opts.httpReferrer = params.httpreferrer; + } + if (params.useragent) { + opts.userAgent = params.useragent; + } + return opts; +} + // Create a new guest instance. const createGuest = function (embedder: Electron.WebContents, embedderFrameId: number, elementInstanceId: number, params: Record) { // eslint-disable-next-line no-undef @@ -96,7 +107,7 @@ const createGuest = function (embedder: Electron.WebContents, embedderFrameId: n // Init guest web view after attached. guest.once('did-attach' as any, function (this: Electron.WebContents, event: Electron.Event) { - params = this.attachParams!; + const params = this.attachParams!; delete this.attachParams; const previouslyAttached = this.viewInstanceId != null; @@ -108,14 +119,7 @@ const createGuest = function (embedder: Electron.WebContents, embedderFrameId: n } if (params.src) { - const opts: Electron.LoadURLOptions = {}; - if (params.httpreferrer) { - opts.httpReferrer = params.httpreferrer; - } - if (params.useragent) { - opts.userAgent = params.useragent; - } - this.loadURL(params.src, opts); + this.loadURL(params.src, params.opts); } embedder.emit('did-attach-webview', event, guest); }); @@ -204,13 +208,15 @@ const attachGuest = function (embedder: Electron.WebContents, embedderFrameId: n return false; } + const { instanceId } = params; + // If this guest is already attached to an element then remove it if (guestInstance.elementInstanceId) { const oldKey = `${guestInstance.embedder.id}-${guestInstance.elementInstanceId}`; embedderElementsMap.delete(oldKey); // Remove guest from embedder if moving across web views - if (guest.viewInstanceId !== params.instanceId) { + if (guest.viewInstanceId !== instanceId) { webViewManager.removeGuest(guestInstance.embedder, guestInstanceId); guestInstance.embedder._sendInternal(`${IPC_MESSAGES.GUEST_VIEW_INTERNAL_DESTROY_GUEST}-${guest.viewInstanceId}`); } @@ -221,12 +227,12 @@ const attachGuest = function (embedder: Electron.WebContents, embedderFrameId: n const event = eventBinding.createWithSender(embedder); embedder.emit('will-attach-webview', event, webPreferences, params); if (event.defaultPrevented) { - if (guest.viewInstanceId == null) guest.viewInstanceId = params.instanceId; + if (guest.viewInstanceId == null) guest.viewInstanceId = instanceId; guest.destroy(); return false; } - guest.attachParams = params; + guest.attachParams = { instanceId, src: params.src, opts: makeLoadURLOptions(params) }; embedderElementsMap.set(key, guestInstanceId); guest.setEmbedder(embedder); diff --git a/lib/renderer/web-view/web-view-impl.ts b/lib/renderer/web-view/web-view-impl.ts index d12eff2f69b9..691b477e1bdb 100644 --- a/lib/renderer/web-view/web-view-impl.ts +++ b/lib/renderer/web-view/web-view-impl.ts @@ -25,7 +25,6 @@ export class WebViewImpl { public hasFocus = false public internalInstanceId?: number; public resizeObserver?: ResizeObserver; - public userAgentOverride?: string; public viewInstanceId: number // on* Event handlers. @@ -180,8 +179,7 @@ export class WebViewImpl { buildParams () { const params: Record = { - instanceId: this.viewInstanceId, - userAgentOverride: this.userAgentOverride + instanceId: this.viewInstanceId }; for (const [attributeName, attribute] of this.attributes) { diff --git a/spec/static/main.js b/spec/static/main.js index f259a3612fd3..28ec06f524cc 100644 --- a/spec/static/main.js +++ b/spec/static/main.js @@ -138,6 +138,12 @@ ipcMain.on('prevent-next-will-attach-webview', (event) => { event.sender.once('will-attach-webview', event => event.preventDefault()); }); +ipcMain.on('break-next-will-attach-webview', (event, id) => { + event.sender.once('will-attach-webview', (event, webPreferences, params) => { + params.instanceId = null; + }); +}); + ipcMain.on('disable-node-on-next-will-attach-webview', (event, id) => { event.sender.once('will-attach-webview', (event, webPreferences, params) => { params.src = `file://${path.join(__dirname, '..', 'fixtures', 'pages', 'c.html')}`; diff --git a/spec/webview-spec.js b/spec/webview-spec.js index 3cff1700b72c..2ded4b0fdedc 100644 --- a/spec/webview-spec.js +++ b/spec/webview-spec.js @@ -1163,6 +1163,14 @@ describe(' tag', function () { }); }); + it('handler modifying params.instanceId does not break ', async () => { + ipcRenderer.send('break-next-will-attach-webview'); + + await startLoadingWebViewAndWaitForMessage(webview, { + src: `file://${fixtures}/pages/a.html` + }); + }); + it('supports preventing a webview from being created', async () => { ipcRenderer.send('prevent-next-will-attach-webview'); diff --git a/typings/internal-electron.d.ts b/typings/internal-electron.d.ts index 0a0ab49b8fbb..666775c16a25 100644 --- a/typings/internal-electron.d.ts +++ b/typings/internal-electron.d.ts @@ -81,7 +81,7 @@ declare namespace Electron { attachToIframe(embedderWebContents: Electron.WebContents, embedderFrameId: number): void; detachFromOuterFrame(): void; setEmbedder(embedder: Electron.WebContents): void; - attachParams?: Record; + attachParams?: { instanceId: number; src: string, opts: LoadURLOptions }; viewInstanceId: number; }