From 2b9dae4b065e9fea59c90ddee2acf2c01b9ebd59 Mon Sep 17 00:00:00 2001 From: Will Anderson Date: Tue, 28 Mar 2023 07:55:41 -0700 Subject: [PATCH] feat: add `will-frame-navigate` event (#34418) * feat: add will-navigate-in-frame event to webContents * docs: add documentation for webview will-frame-navigate event * feat: Eliminate isInPlace argument from will-frame-navigate event * fix: Fire will-frame-navigate before will-navigate * feat: send will-frame-navigate with a WebFrameMain in the event details * docs: Update WebContents docs for new API signature * feat: Add custom event forwarding for will-frame-navigate * fix: wrap WebFrameMain so it can be sent as an event * test: update webContents and tests to match new signatures * chore: undo unnecessary change * fix: don't switch will-navigate to use EmitNavigationEventDetails * test: clean up will-navigate and will-frame-navigate tests for * chore: apply lint fixes * chore: move GetRenderFrameHost helper into anonymous namespace * docs: auto-generate WillFrameNavigateDetails rather than defining it manually * test: Update tests to actually pass under new spec runner * docs: Add section explaining relationship between various nav events * test: Add some tests to ensure navigation event order doesn't silently change * test: Always monitor all nav events to ensure unexpected ones don't fire * test: Add test to verify in-page navigation event order * feat: Change to new style where extra params are exposed as event props * fix: Remove unused EmitNavigationEventDetails * fix: Update tests to use new async helpers * docs: Rename and reorder sections documenting navigation events --------- Co-authored-by: Milan Burda --- docs/api/web-contents.md | 60 ++- docs/api/webview-tag.md | 22 ++ lib/browser/guest-view-manager.ts | 10 + .../browser/api/electron_api_web_contents.cc | 29 +- shell/browser/electron_navigation_throttle.cc | 4 + spec/api-browser-window-spec.ts | 374 +++++++++++++++++- .../pages/webview-will-navigate-in-frame.html | 12 + spec/webview-spec.ts | 43 +- 8 files changed, 540 insertions(+), 14 deletions(-) create mode 100644 spec/fixtures/pages/webview-will-navigate-in-frame.html diff --git a/docs/api/web-contents.md b/docs/api/web-contents.md index 47f6e8a41c0c..456893603091 100644 --- a/docs/api/web-contents.md +++ b/docs/api/web-contents.md @@ -19,6 +19,36 @@ const contents = win.webContents console.log(contents) ``` +## Navigation Events + +Several events can be used to monitor navigations as they occur within a `webContents`. + +### Document Navigations + +When a `webContents` navigates to another page (as opposed to an [in-page navigation](web-contents.md#in-page-navigation)), the following events will be fired. + +* [`did-start-navigation`](web-contents.md#event-did-start-navigation) +* [`will-frame-navigate`](web-contents.md#event-will-frame-navigate) +* [`will-navigate`](web-contents.md#event-will-navigate) (only fired when main frame navigates) +* [`will-redirect`](web-contents.md#event-will-redirect) (only fired when a redirect happens during navigation) +* [`did-redirect-navigation`](web-contents.md#event-did-redirect-navigation) (only fired when a redirect happens during navigation) +* [`did-frame-navigate`](web-contents.md#event-did-frame-navigate) +* [`did-navigate`](web-contents.md#event-did-navigate) (only fired when main frame navigates) + +Subsequent events will not fire if `event.preventDefault()` is called on any of the cancellable events. + +### In-page Navigation + +In-page navigations don't cause the page to reload, but instead navigate to a location within the current page. These events are not cancellable. For an in-page navigations, the following events will fire in this order: + +* [`did-start-navigation`](web-contents.md#event-did-start-navigation) +* [`did-navigate-in-page`](web-contents.md#event-did-navigate-in-page) + +### Frame Navigation + +The [`will-navigate`](web-contents.md#event-will-navigate) and [`did-navigate`](web-contents.md#event-did-navigate) events only fire when the [mainFrame](web-contents.md#contentsmainframe-readonly) navigates. +If you want to also observe navigations in ``); + }); + + it('is triggered when navigating from file: to http:', async () => { + await w.loadFile(path.join(fixtures, 'api', 'blank.html')); + w.webContents.executeJavaScript(`location.href = ${JSON.stringify(url)}`); + const navigatedTo = await new Promise(resolve => { + w.webContents.once('will-frame-navigate', (e) => { + e.preventDefault(); + resolve(e.url); + }); + }); + expect(navigatedTo).to.equal(url); + expect(w.webContents.getURL()).to.match(/^file:/); + }); + + it('is triggered when navigating from about:blank to http:', async () => { + await w.loadURL('about:blank'); + w.webContents.executeJavaScript(`location.href = ${JSON.stringify(url)}`); + const navigatedTo = await new Promise(resolve => { + w.webContents.once('will-frame-navigate', (e) => { + e.preventDefault(); + resolve(e.url); + }); + }); + expect(navigatedTo).to.equal(url); + expect(w.webContents.getURL()).to.equal('about:blank'); + }); + + it('is triggered when a cross-origin iframe navigates _top', async () => { + await w.loadURL(`data:text/html,`); + await setTimeout(1000); + + let willFrameNavigateEmitted = false; + let isMainFrameValue; + w.webContents.on('will-frame-navigate', (event) => { + willFrameNavigateEmitted = true; + isMainFrameValue = event.isMainFrame; + }); + const didNavigatePromise = once(w.webContents, 'did-navigate'); + + w.webContents.debugger.attach('1.1'); + const targets = await w.webContents.debugger.sendCommand('Target.getTargets'); + const iframeTarget = targets.targetInfos.find((t: any) => t.type === 'iframe'); + const { sessionId } = await w.webContents.debugger.sendCommand('Target.attachToTarget', { + targetId: iframeTarget.targetId, + flatten: true + }); + await w.webContents.debugger.sendCommand('Input.dispatchMouseEvent', { + type: 'mousePressed', + x: 10, + y: 10, + clickCount: 1, + button: 'left' + }, sessionId); + await w.webContents.debugger.sendCommand('Input.dispatchMouseEvent', { + type: 'mouseReleased', + x: 10, + y: 10, + clickCount: 1, + button: 'left' + }, sessionId); + + await didNavigatePromise; + + expect(willFrameNavigateEmitted).to.be.true(); + expect(isMainFrameValue).to.be.true(); + }); + + it('is triggered when a cross-origin iframe navigates itself', async () => { + await w.loadURL(`data:text/html,`); + await setTimeout(1000); + + let willNavigateEmitted = false; + let isMainFrameValue; + w.webContents.on('will-frame-navigate', (event) => { + willNavigateEmitted = true; + isMainFrameValue = event.isMainFrame; + }); + const didNavigatePromise = once(w.webContents, 'did-frame-navigate'); + + w.webContents.debugger.attach('1.1'); + const targets = await w.webContents.debugger.sendCommand('Target.getTargets'); + const iframeTarget = targets.targetInfos.find((t: any) => t.type === 'iframe'); + const { sessionId } = await w.webContents.debugger.sendCommand('Target.attachToTarget', { + targetId: iframeTarget.targetId, + flatten: true + }); + await w.webContents.debugger.sendCommand('Input.dispatchMouseEvent', { + type: 'mousePressed', + x: 10, + y: 10, + clickCount: 1, + button: 'left' + }, sessionId); + await w.webContents.debugger.sendCommand('Input.dispatchMouseEvent', { + type: 'mouseReleased', + x: 10, + y: 10, + clickCount: 1, + button: 'left' + }, sessionId); + + await didNavigatePromise; + + expect(willNavigateEmitted).to.be.true(); + expect(isMainFrameValue).to.be.false(); + }); + + it('can cancel when a cross-origin iframe navigates itself', async () => { + + }); + }); + describe('will-redirect event', () => { let server: http.Server; let url: string; @@ -652,6 +851,179 @@ describe('BrowserWindow module', () => { w.loadURL(`${url}/navigate-302`); }); }); + + describe('ordering', () => { + let server = null as unknown as http.Server; + let url = null as unknown as string; + const navigationEvents = [ + 'did-start-navigation', + 'did-navigate-in-page', + 'will-frame-navigate', + 'will-navigate', + 'will-redirect', + 'did-redirect-navigation', + 'did-frame-navigate', + 'did-navigate' + ]; + before((done) => { + server = http.createServer((req, res) => { + if (req.url === '/navigate') { + res.end('navigate'); + } else if (req.url === '/redirect') { + res.end('redirect'); + } else if (req.url === '/redirect2') { + res.statusCode = 302; + res.setHeader('location', url); + res.end(); + } else if (req.url === '/in-page') { + res.end('redirect
'); + } else { + res.end(''); + } + }); + server.listen(0, '127.0.0.1', () => { + url = `http://127.0.0.1:${(server.address() as AddressInfo).port}/`; + done(); + }); + }); + it('for initial navigation, event order is consistent', async () => { + const firedEvents: string[] = []; + const expectedEventOrder = [ + 'did-start-navigation', + 'did-frame-navigate', + 'did-navigate' + ]; + const allEvents = Promise.all(navigationEvents.map(event => + once(w.webContents, event).then(() => firedEvents.push(event)) + )); + const timeout = setTimeout(1000); + w.loadURL(url); + await Promise.race([allEvents, timeout]); + expect(firedEvents).to.deep.equal(expectedEventOrder); + }); + + it('for second navigation, event order is consistent', async () => { + const firedEvents: string[] = []; + const expectedEventOrder = [ + 'did-start-navigation', + 'will-frame-navigate', + 'will-navigate', + 'did-frame-navigate', + 'did-navigate' + ]; + w.loadURL(`${url}navigate`); + await once(w.webContents, 'did-navigate'); + await setTimeout(1000); + navigationEvents.forEach(event => + once(w.webContents, event).then(() => firedEvents.push(event)) + ); + const navigationFinished = once(w.webContents, 'did-navigate'); + w.webContents.debugger.attach('1.1'); + const targets = await w.webContents.debugger.sendCommand('Target.getTargets'); + const pageTarget = targets.targetInfos.find((t: any) => t.type === 'page'); + const { sessionId } = await w.webContents.debugger.sendCommand('Target.attachToTarget', { + targetId: pageTarget.targetId, + flatten: true + }); + await w.webContents.debugger.sendCommand('Input.dispatchMouseEvent', { + type: 'mousePressed', + x: 10, + y: 10, + clickCount: 1, + button: 'left' + }, sessionId); + await w.webContents.debugger.sendCommand('Input.dispatchMouseEvent', { + type: 'mouseReleased', + x: 10, + y: 10, + clickCount: 1, + button: 'left' + }, sessionId); + await navigationFinished; + expect(firedEvents).to.deep.equal(expectedEventOrder); + }); + + it('when navigating with redirection, event order is consistent', async () => { + const firedEvents: string[] = []; + const expectedEventOrder = [ + 'did-start-navigation', + 'will-frame-navigate', + 'will-navigate', + 'will-redirect', + 'did-redirect-navigation', + 'did-frame-navigate', + 'did-navigate' + ]; + w.loadURL(`${url}redirect`); + await once(w.webContents, 'did-navigate'); + await setTimeout(1000); + navigationEvents.forEach(event => + once(w.webContents, event).then(() => firedEvents.push(event)) + ); + const navigationFinished = once(w.webContents, 'did-navigate'); + w.webContents.debugger.attach('1.1'); + const targets = await w.webContents.debugger.sendCommand('Target.getTargets'); + const pageTarget = targets.targetInfos.find((t: any) => t.type === 'page'); + const { sessionId } = await w.webContents.debugger.sendCommand('Target.attachToTarget', { + targetId: pageTarget.targetId, + flatten: true + }); + await w.webContents.debugger.sendCommand('Input.dispatchMouseEvent', { + type: 'mousePressed', + x: 10, + y: 10, + clickCount: 1, + button: 'left' + }, sessionId); + await w.webContents.debugger.sendCommand('Input.dispatchMouseEvent', { + type: 'mouseReleased', + x: 10, + y: 10, + clickCount: 1, + button: 'left' + }, sessionId); + await navigationFinished; + expect(firedEvents).to.deep.equal(expectedEventOrder); + }); + + it('when navigating in-page, event order is consistent', async () => { + const firedEvents: string[] = []; + const expectedEventOrder = [ + 'did-start-navigation', + 'did-navigate-in-page' + ]; + w.loadURL(`${url}in-page`); + await once(w.webContents, 'did-navigate'); + await setTimeout(1000); + navigationEvents.forEach(event => + once(w.webContents, event).then(() => firedEvents.push(event)) + ); + const navigationFinished = once(w.webContents, 'did-navigate-in-page'); + w.webContents.debugger.attach('1.1'); + const targets = await w.webContents.debugger.sendCommand('Target.getTargets'); + const pageTarget = targets.targetInfos.find((t: any) => t.type === 'page'); + const { sessionId } = await w.webContents.debugger.sendCommand('Target.attachToTarget', { + targetId: pageTarget.targetId, + flatten: true + }); + await w.webContents.debugger.sendCommand('Input.dispatchMouseEvent', { + type: 'mousePressed', + x: 10, + y: 10, + clickCount: 1, + button: 'left' + }, sessionId); + await w.webContents.debugger.sendCommand('Input.dispatchMouseEvent', { + type: 'mouseReleased', + x: 10, + y: 10, + clickCount: 1, + button: 'left' + }, sessionId); + await navigationFinished; + expect(firedEvents).to.deep.equal(expectedEventOrder); + }); + }); }); } diff --git a/spec/fixtures/pages/webview-will-navigate-in-frame.html b/spec/fixtures/pages/webview-will-navigate-in-frame.html new file mode 100644 index 000000000000..63f921e72427 --- /dev/null +++ b/spec/fixtures/pages/webview-will-navigate-in-frame.html @@ -0,0 +1,12 @@ + + + + + diff --git a/spec/webview-spec.ts b/spec/webview-spec.ts index c6a4d514d91e..8d0f78a5c800 100644 --- a/spec/webview-spec.ts +++ b/spec/webview-spec.ts @@ -1480,7 +1480,7 @@ describe(' tag', function () { }); describe('will-navigate event', () => { - it('emits when a url that leads to outside of the page is clicked', async () => { + it('emits when a url that leads to outside of the page is loaded', async () => { const { url } = await loadWebViewAndWaitForEvent(w, { src: `file://${fixtures}/pages/webview-will-navigate.html` }, 'will-navigate'); @@ -1489,6 +1489,47 @@ describe(' tag', function () { }); }); + describe('will-frame-navigate event', () => { + it('emits when a link that leads to outside of the page is loaded', async () => { + const { url, isMainFrame } = await loadWebViewAndWaitForEvent(w, { + src: `file://${fixtures}/pages/webview-will-navigate.html` + }, 'will-frame-navigate'); + expect(url).to.equal('http://host/'); + expect(isMainFrame).to.be.true(); + }); + + it('emits when a link within an iframe, which leads to outside of the page, is loaded', async () => { + await loadWebView(w, { + src: `file://${fixtures}/pages/webview-will-navigate-in-frame.html`, + nodeIntegration: '' + }); + + const { url, frameProcessId, frameRoutingId } = await w.executeJavaScript(` + new Promise((resolve, reject) => { + let hasFrameNavigatedOnce = false; + const webview = document.getElementById('webview'); + webview.addEventListener('will-frame-navigate', ({url, isMainFrame, frameProcessId, frameRoutingId}) => { + if (isMainFrame) return; + if (hasFrameNavigatedOnce) resolve({ + url, + isMainFrame, + frameProcessId, + frameRoutingId, + }); + + // First navigation is the initial iframe load within the + hasFrameNavigatedOnce = true; + }); + webview.executeJavaScript('loadSubframe()'); + }); + `); + + expect(url).to.equal('http://host/'); + expect(frameProcessId).to.be.a('number'); + expect(frameRoutingId).to.be.a('number'); + }); + }); + describe('did-navigate event', () => { it('emits when a url that leads to outside of the page is clicked', async () => { const pageUrl = url.pathToFileURL(path.join(fixtures, 'pages', 'webview-will-navigate.html')).toString();