fix: promise resolved to early when browser initiated in-page navigation (#39260)
This commit is contained in:
parent
d9329042e2
commit
a0effaf9b8
3 changed files with 32 additions and 2 deletions
|
@ -446,6 +446,7 @@ WebContents.prototype.loadURL = function (url, options) {
|
||||||
};
|
};
|
||||||
|
|
||||||
let navigationStarted = false;
|
let navigationStarted = false;
|
||||||
|
let browserInitiatedInPageNavigation = false;
|
||||||
const navigationListener = (event: Electron.Event, url: string, isSameDocument: boolean, isMainFrame: boolean) => {
|
const navigationListener = (event: Electron.Event, url: string, isSameDocument: boolean, isMainFrame: boolean) => {
|
||||||
if (isMainFrame) {
|
if (isMainFrame) {
|
||||||
if (navigationStarted && !isSameDocument) {
|
if (navigationStarted && !isSameDocument) {
|
||||||
|
@ -460,6 +461,7 @@ WebContents.prototype.loadURL = function (url, options) {
|
||||||
// as the routing does not leave the document
|
// as the routing does not leave the document
|
||||||
return rejectAndCleanup(-3, 'ERR_ABORTED', url);
|
return rejectAndCleanup(-3, 'ERR_ABORTED', url);
|
||||||
}
|
}
|
||||||
|
browserInitiatedInPageNavigation = navigationStarted && isSameDocument;
|
||||||
navigationStarted = true;
|
navigationStarted = true;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -474,17 +476,22 @@ WebContents.prototype.loadURL = function (url, options) {
|
||||||
// would be more appropriate.
|
// would be more appropriate.
|
||||||
rejectAndCleanup(-2, 'ERR_FAILED', url);
|
rejectAndCleanup(-2, 'ERR_FAILED', url);
|
||||||
};
|
};
|
||||||
|
const finishListenerWhenUserInitiatedNavigation = () => {
|
||||||
|
if (!browserInitiatedInPageNavigation) {
|
||||||
|
finishListener();
|
||||||
|
}
|
||||||
|
};
|
||||||
const removeListeners = () => {
|
const removeListeners = () => {
|
||||||
this.removeListener('did-finish-load', finishListener);
|
this.removeListener('did-finish-load', finishListener);
|
||||||
this.removeListener('did-fail-load', failListener);
|
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-start-navigation', navigationListener);
|
||||||
this.removeListener('did-stop-loading', stopLoadingListener);
|
this.removeListener('did-stop-loading', stopLoadingListener);
|
||||||
this.removeListener('destroyed', stopLoadingListener);
|
this.removeListener('destroyed', stopLoadingListener);
|
||||||
};
|
};
|
||||||
this.on('did-finish-load', finishListener);
|
this.on('did-finish-load', finishListener);
|
||||||
this.on('did-fail-load', failListener);
|
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-start-navigation', navigationListener);
|
||||||
this.on('did-stop-loading', stopLoadingListener);
|
this.on('did-stop-loading', stopLoadingListener);
|
||||||
this.on('destroyed', stopLoadingListener);
|
this.on('destroyed', stopLoadingListener);
|
||||||
|
|
|
@ -375,6 +375,16 @@ describe('webContents module', () => {
|
||||||
await expect(w.loadURL(w.getURL() + '#foo')).to.eventually.be.fulfilled();
|
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 () => {
|
it('rejects when failing to load a file URL', async () => {
|
||||||
await expect(w.loadURL('file:non-existent')).to.eventually.be.rejected()
|
await expect(w.loadURL('file:non-existent')).to.eventually.be.rejected()
|
||||||
.and.have.property('code', 'ERR_FILE_NOT_FOUND');
|
.and.have.property('code', 'ERR_FILE_NOT_FOUND');
|
||||||
|
|
13
spec/fixtures/pages/navigate_in_page_and_wait.html
vendored
Normal file
13
spec/fixtures/pages/navigate_in_page_and_wait.html
vendored
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
<html>
|
||||||
|
<header>
|
||||||
|
<script type="text/javascript">
|
||||||
|
window.history.replaceState(window.location.href, "Sample Title", window.location.href);
|
||||||
|
// Simulate that we load web page.
|
||||||
|
for (let i = 0; i < 10000; i++) {
|
||||||
|
console.log('wait : ' + i);
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</header>
|
||||||
|
<body>
|
||||||
|
</body>
|
||||||
|
</html>
|
Loading…
Reference in a new issue