From a00a25376d339ba78c4a1efbae2de05fd534332e Mon Sep 17 00:00:00 2001 From: Shelley Vohr Date: Wed, 21 Jun 2023 21:20:54 +0200 Subject: [PATCH] fix: crash calling `BrowserWindow.removeBrowserView()` with destroyed `webContents` (#38842) fix: crash calling removeBrowserView() with destroyed webContents https://github.com/electron/electron/issues/37642 --- shell/browser/api/electron_api_browser_view.cc | 4 ++++ shell/browser/api/electron_api_browser_view.h | 3 +++ spec/api-browser-view-spec.ts | 14 ++++++++++++++ 3 files changed, 21 insertions(+) diff --git a/shell/browser/api/electron_api_browser_view.cc b/shell/browser/api/electron_api_browser_view.cc index cc7a1d717ef3..2049664c6abf 100644 --- a/shell/browser/api/electron_api_browser_view.cc +++ b/shell/browser/api/electron_api_browser_view.cc @@ -131,6 +131,10 @@ void BrowserView::WebContentsDestroyed() { Unpin(); } +void BrowserView::OnCloseContents() { + api_web_contents_ = nullptr; +} + // static gin::Handle BrowserView::New(gin_helper::ErrorThrower thrower, gin::Arguments* args) { diff --git a/shell/browser/api/electron_api_browser_view.h b/shell/browser/api/electron_api_browser_view.h index 409b5dda0c0b..caf31bae155a 100644 --- a/shell/browser/api/electron_api_browser_view.h +++ b/shell/browser/api/electron_api_browser_view.h @@ -71,6 +71,9 @@ class BrowserView : public gin::Wrappable, // content::WebContentsObserver: void WebContentsDestroyed() override; + // ExtendedWebContentsObserver: + void OnCloseContents() override; + private: void SetAutoResize(AutoResizeFlags flags); void SetBounds(const gfx::Rect& bounds); diff --git a/spec/api-browser-view-spec.ts b/spec/api-browser-view-spec.ts index 69084034d260..80bb56478d41 100644 --- a/spec/api-browser-view-spec.ts +++ b/spec/api-browser-view-spec.ts @@ -257,6 +257,20 @@ describe('BrowserView module', () => { w.removeBrowserView(view); }).to.not.throw(); }); + + it('can be called on a BrowserView with a destroyed webContents', (done) => { + view = new BrowserView(); + w.addBrowserView(view); + + view.webContents.on('destroyed', () => { + w.removeBrowserView(view); + done(); + }); + + view.webContents.loadURL('data:text/html,hello there').then(() => { + view.webContents.close(); + }); + }); }); describe('BrowserWindow.getBrowserViews()', () => {