From a310d57d847b008dba3b0772085840dde3343c64 Mon Sep 17 00:00:00 2001 From: Andrey Lushnikov Date: Thu, 3 May 2018 01:51:45 +0300 Subject: [PATCH] feat(Page): add option to run 'beforeunload' when closing the page (#2478) Today, `page.close()` method doesn't run page's beforeunload listeners. This way users can be sure that `page.close()` actually closes the page. This patch adds an optional `runBeforeUnload` option to the `page.close()` method that would run beforeunload listeners. Note: running beforeunload handlers might cancel page closing. Fixes #2386. --- docs/api.md | 13 +++++++++++-- lib/Page.js | 14 +++++++++++--- test/assets/beforeunload.html | 5 +++++ test/page.spec.js | 14 ++++++++++++++ 4 files changed, 41 insertions(+), 5 deletions(-) create mode 100644 test/assets/beforeunload.html diff --git a/docs/api.md b/docs/api.md index 350bc150805e9..9e42d1933b0e4 100644 --- a/docs/api.md +++ b/docs/api.md @@ -69,7 +69,7 @@ * [page.bringToFront()](#pagebringtofront) * [page.browser()](#pagebrowser) * [page.click(selector[, options])](#pageclickselector-options) - * [page.close()](#pageclose) + * [page.close(options)](#pagecloseoptions) * [page.content()](#pagecontent) * [page.cookies(...urls)](#pagecookiesurls) * [page.coverage](#pagecoverage) @@ -747,9 +747,18 @@ const [response] = await Promise.all([ Shortcut for [page.mainFrame().click(selector[, options])](#frameclickselector-options). -#### page.close() +#### page.close(options) +- `options` <[Object]> + - `runBeforeUnload` <[boolean]> Defaults to `false`. Whether to run the + [before unload](https://developer.mozilla.org/en-US/docs/Web/Events/beforeunload) + page handlers. - returns: <[Promise]> +By default, `page.close()` **does not** run beforeunload handlers. + +> **NOTE** if `runBeforeUnload` is passed as true, a `beforeunload` dialog might be summoned +> and should be handled manually via page's ['dialog'](#event-dialog) event. + #### page.content() - returns: <[Promise]<[String]>> diff --git a/lib/Page.js b/lib/Page.js index c1cfe935e5af7..f3d9dbeef4b3b 100644 --- a/lib/Page.js +++ b/lib/Page.js @@ -823,10 +823,18 @@ class Page extends EventEmitter { return this.mainFrame().title(); } - async close() { + /** + * @param {!{runBeforeUnload: (boolean|undefined)}=} options + */ + async close(options = {runBeforeUnload: undefined}) { console.assert(!!this._client._connection, 'Protocol error: Connection closed. Most likely the page has been closed.'); - await this._client._connection.send('Target.closeTarget', { targetId: this._target._targetId }); - await this._target._isClosedPromise; + const runBeforeUnload = !!options.runBeforeUnload; + if (runBeforeUnload) { + await this._client.send('Page.close'); + } else { + await this._client._connection.send('Target.closeTarget', { targetId: this._target._targetId }); + await this._target._isClosedPromise; + } } /** diff --git a/test/assets/beforeunload.html b/test/assets/beforeunload.html new file mode 100644 index 0000000000000..6c3075b475edd --- /dev/null +++ b/test/assets/beforeunload.html @@ -0,0 +1,5 @@ + diff --git a/test/page.spec.js b/test/page.spec.js index efabb2c8484ac..515af17204c4a 100644 --- a/test/page.spec.js +++ b/test/page.spec.js @@ -40,6 +40,20 @@ module.exports.addTests = function({testRunner, expect, puppeteer, DeviceDescrip await newPage.close(); expect(await browser.pages()).not.toContain(newPage); }); + it('should run beforeunload if asked for', async({browser, server}) => { + const newPage = await browser.newPage(); + await newPage.goto(server.PREFIX + '/beforeunload.html'); + // We have to interact with a page so that 'beforeunload' handlers + // fire. + await newPage.click('body'); + newPage.close({ runBeforeUnload: true }); + const dialog = await waitEvent(newPage, 'dialog'); + expect(dialog.type()).toBe('beforeunload'); + expect(dialog.defaultValue()).toBe(''); + expect(dialog.message()).toBe(''); + dialog.accept(); + await waitEvent(newPage, 'close'); + }); }); describe('Page.Events.error', function() {