From f30fbebb984816d71ab6723c04dd40b81337f397 Mon Sep 17 00:00:00 2001 From: Tomasz <30628471+TomaszMa@users.noreply.github.com> Date: Mon, 28 Aug 2023 18:37:28 +0200 Subject: [PATCH] fix: promise resolved to early when browser initiated in-page navigation v2 (#39597) --- lib/browser/api/web-contents.ts | 11 +++++++++-- spec/api-web-contents-spec.ts | 10 ++++++++++ .../fixtures/pages/navigate_in_page_and_wait.html | 15 +++++++++++++++ 3 files changed, 34 insertions(+), 2 deletions(-) create mode 100644 spec/fixtures/pages/navigate_in_page_and_wait.html diff --git a/lib/browser/api/web-contents.ts b/lib/browser/api/web-contents.ts index 44ceb6b8c43e..d1d74d15603b 100644 --- a/lib/browser/api/web-contents.ts +++ b/lib/browser/api/web-contents.ts @@ -454,6 +454,7 @@ WebContents.prototype.loadURL = function (url, options) { }; let navigationStarted = false; + let browserInitiatedInPageNavigation = false; const navigationListener = (event: Electron.Event, url: string, isSameDocument: boolean, isMainFrame: boolean) => { if (isMainFrame) { if (navigationStarted && !isSameDocument) { @@ -468,6 +469,7 @@ WebContents.prototype.loadURL = function (url, options) { // as the routing does not leave the document return rejectAndCleanup(-3, 'ERR_ABORTED', url); } + browserInitiatedInPageNavigation = navigationStarted && isSameDocument; navigationStarted = true; } }; @@ -482,17 +484,22 @@ WebContents.prototype.loadURL = function (url, options) { // would be more appropriate. rejectAndCleanup(-2, 'ERR_FAILED', url); }; + const finishListenerWhenUserInitiatedNavigation = () => { + if (!browserInitiatedInPageNavigation) { + finishListener(); + } + }; const removeListeners = () => { this.removeListener('did-finish-load', finishListener); this.removeListener('did-fail-load', failListener); - this.removeListener('did-navigate-in-page', finishListener); + this.removeListener('did-navigate-in-page', finishListenerWhenUserInitiatedNavigation); this.removeListener('did-start-navigation', navigationListener); this.removeListener('did-stop-loading', stopLoadingListener); this.removeListener('destroyed', stopLoadingListener); }; this.on('did-finish-load', finishListener); this.on('did-fail-load', failListener); - this.on('did-navigate-in-page', finishListener); + this.on('did-navigate-in-page', finishListenerWhenUserInitiatedNavigation); this.on('did-start-navigation', navigationListener); this.on('did-stop-loading', stopLoadingListener); this.on('destroyed', stopLoadingListener); diff --git a/spec/api-web-contents-spec.ts b/spec/api-web-contents-spec.ts index d0f560b8bcf1..161f515eb694 100644 --- a/spec/api-web-contents-spec.ts +++ b/spec/api-web-contents-spec.ts @@ -375,6 +375,16 @@ describe('webContents module', () => { await expect(w.loadURL(w.getURL() + '#foo')).to.eventually.be.fulfilled(); }); + it('resolves after browser initiated navigation', async () => { + let finishedLoading = false; + w.webContents.on('did-finish-load', function () { + finishedLoading = true; + }); + + await w.loadFile(path.join(fixturesPath, 'pages', 'navigate_in_page_and_wait.html')); + expect(finishedLoading).to.be.true(); + }); + it('rejects when failing to load a file URL', async () => { await expect(w.loadURL('file:non-existent')).to.eventually.be.rejected() .and.have.property('code', 'ERR_FILE_NOT_FOUND'); diff --git a/spec/fixtures/pages/navigate_in_page_and_wait.html b/spec/fixtures/pages/navigate_in_page_and_wait.html new file mode 100644 index 000000000000..f582af55a583 --- /dev/null +++ b/spec/fixtures/pages/navigate_in_page_and_wait.html @@ -0,0 +1,15 @@ + +
+ +
+ + +