mirror of
https://github.com/puppeteer/puppeteer
synced 2024-06-14 14:02:48 +00:00
refactor: unify tab target handling in page (#11115)
This commit is contained in:
parent
78c335e611
commit
bb9ef6ee8b
@ -68,7 +68,8 @@ export class CdpCDPSession extends CDPSession {
|
|||||||
|
|
||||||
override parentSession(): CDPSession | undefined {
|
override parentSession(): CDPSession | undefined {
|
||||||
if (!this.#parentSessionId) {
|
if (!this.#parentSessionId) {
|
||||||
return;
|
// To make it work in Firefox that does not have parent (tab) sessions.
|
||||||
|
return this;
|
||||||
}
|
}
|
||||||
const parent = this.#connection?.session(this.#parentSessionId);
|
const parent = this.#connection?.session(this.#parentSessionId);
|
||||||
return parent ?? undefined;
|
return parent ?? undefined;
|
||||||
|
@ -22,6 +22,7 @@ import {EventEmitter} from '../common/EventEmitter.js';
|
|||||||
import {assert} from '../util/assert.js';
|
import {assert} from '../util/assert.js';
|
||||||
import {Deferred} from '../util/Deferred.js';
|
import {Deferred} from '../util/Deferred.js';
|
||||||
|
|
||||||
|
import type {CdpCDPSession} from './CDPSession.js';
|
||||||
import type {Connection} from './Connection.js';
|
import type {Connection} from './Connection.js';
|
||||||
import type {CdpTarget} from './Target.js';
|
import type {CdpTarget} from './Target.js';
|
||||||
import {
|
import {
|
||||||
@ -205,6 +206,7 @@ export class FirefoxTargetManager
|
|||||||
|
|
||||||
assert(target, `Target ${targetInfo.targetId} is missing`);
|
assert(target, `Target ${targetInfo.targetId} is missing`);
|
||||||
|
|
||||||
|
(session as CdpCDPSession)._setTarget(target);
|
||||||
this.setupAttachmentListeners(session);
|
this.setupAttachmentListeners(session);
|
||||||
|
|
||||||
this.#availableTargetsBySessionId.set(
|
this.#availableTargetsBySessionId.set(
|
||||||
|
@ -80,6 +80,7 @@ import {CdpKeyboard, CdpMouse, CdpTouchscreen} from './Input.js';
|
|||||||
import {MAIN_WORLD} from './IsolatedWorlds.js';
|
import {MAIN_WORLD} from './IsolatedWorlds.js';
|
||||||
import type {Credentials, NetworkConditions} from './NetworkManager.js';
|
import type {Credentials, NetworkConditions} from './NetworkManager.js';
|
||||||
import type {CdpTarget} from './Target.js';
|
import type {CdpTarget} from './Target.js';
|
||||||
|
import type {TargetManager} from './TargetManager.js';
|
||||||
import {TargetManagerEvent} from './TargetManager.js';
|
import {TargetManagerEvent} from './TargetManager.js';
|
||||||
import {Tracing} from './Tracing.js';
|
import {Tracing} from './Tracing.js';
|
||||||
import {WebWorker} from './WebWorker.js';
|
import {WebWorker} from './WebWorker.js';
|
||||||
@ -111,9 +112,12 @@ export class CdpPage extends Page {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#closed = false;
|
#closed = false;
|
||||||
#client: CDPSession;
|
readonly #targetManager: TargetManager;
|
||||||
#tabSession: CDPSession | undefined;
|
|
||||||
#target: CdpTarget;
|
#primaryTargetClient: CDPSession;
|
||||||
|
#primaryTarget: CdpTarget;
|
||||||
|
#tabTargetClient: CDPSession;
|
||||||
|
#tabTarget: CdpTarget;
|
||||||
#keyboard: CdpKeyboard;
|
#keyboard: CdpKeyboard;
|
||||||
#mouse: CdpMouse;
|
#mouse: CdpMouse;
|
||||||
#touchscreen: CdpTouchscreen;
|
#touchscreen: CdpTouchscreen;
|
||||||
@ -222,9 +226,13 @@ export class CdpPage extends Page {
|
|||||||
ignoreHTTPSErrors: boolean
|
ignoreHTTPSErrors: boolean
|
||||||
) {
|
) {
|
||||||
super();
|
super();
|
||||||
this.#client = client;
|
this.#primaryTargetClient = client;
|
||||||
this.#tabSession = client.parentSession();
|
this.#tabTargetClient = client.parentSession()!;
|
||||||
this.#target = target;
|
assert(this.#tabTargetClient, 'Tab target session is not defined.');
|
||||||
|
this.#tabTarget = (this.#tabTargetClient as CdpCDPSession)._target();
|
||||||
|
assert(this.#tabTarget, 'Tab target is not defined.');
|
||||||
|
this.#primaryTarget = target;
|
||||||
|
this.#targetManager = target._targetManager();
|
||||||
this.#keyboard = new CdpKeyboard(client);
|
this.#keyboard = new CdpKeyboard(client);
|
||||||
this.#mouse = new CdpMouse(client, this.#keyboard);
|
this.#mouse = new CdpMouse(client, this.#keyboard);
|
||||||
this.#touchscreen = new CdpTouchscreen(client, this.#keyboard);
|
this.#touchscreen = new CdpTouchscreen(client, this.#keyboard);
|
||||||
@ -249,16 +257,45 @@ export class CdpPage extends Page {
|
|||||||
this.#frameManager.networkManager.on(eventName, handler as any);
|
this.#frameManager.networkManager.on(eventName, handler as any);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.#setupPrimaryTargetListeners();
|
this.#tabTargetClient.on(
|
||||||
|
CDPSessionEvent.Swapped,
|
||||||
|
this.#onActivation.bind(this)
|
||||||
|
);
|
||||||
|
|
||||||
this.#tabSession?.on(CDPSessionEvent.Swapped, async newSession => {
|
this.#tabTargetClient.on(
|
||||||
this.#client = newSession;
|
CDPSessionEvent.Ready,
|
||||||
|
this.#onSecondaryTarget.bind(this)
|
||||||
|
);
|
||||||
|
|
||||||
|
this.#targetManager.on(
|
||||||
|
TargetManagerEvent.TargetGone,
|
||||||
|
this.#onDetachedFromTarget
|
||||||
|
);
|
||||||
|
|
||||||
|
this.#tabTarget._isClosedDeferred
|
||||||
|
.valueOrThrow()
|
||||||
|
.then(() => {
|
||||||
|
this.#targetManager.off(
|
||||||
|
TargetManagerEvent.TargetGone,
|
||||||
|
this.#onDetachedFromTarget
|
||||||
|
);
|
||||||
|
|
||||||
|
this.emit(PageEvent.Close, undefined);
|
||||||
|
this.#closed = true;
|
||||||
|
})
|
||||||
|
.catch(debugError);
|
||||||
|
|
||||||
|
this.#setupPrimaryTargetListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
async #onActivation(newSession: CDPSession): Promise<void> {
|
||||||
|
this.#primaryTargetClient = newSession;
|
||||||
assert(
|
assert(
|
||||||
this.#client instanceof CdpCDPSession,
|
this.#primaryTargetClient instanceof CdpCDPSession,
|
||||||
'CDPSession is not instance of CDPSessionImpl'
|
'CDPSession is not instance of CDPSessionImpl'
|
||||||
);
|
);
|
||||||
this.#target = this.#client._target();
|
this.#primaryTarget = this.#primaryTargetClient._target();
|
||||||
assert(this.#target, 'Missing target on swap');
|
assert(this.#primaryTarget, 'Missing target on swap');
|
||||||
this.#keyboard.updateClient(newSession);
|
this.#keyboard.updateClient(newSession);
|
||||||
this.#mouse.updateClient(newSession);
|
this.#mouse.updateClient(newSession);
|
||||||
this.#touchscreen.updateClient(newSession);
|
this.#touchscreen.updateClient(newSession);
|
||||||
@ -268,8 +305,9 @@ export class CdpPage extends Page {
|
|||||||
this.#coverage.updateClient(newSession);
|
this.#coverage.updateClient(newSession);
|
||||||
await this.#frameManager.swapFrameTree(newSession);
|
await this.#frameManager.swapFrameTree(newSession);
|
||||||
this.#setupPrimaryTargetListeners();
|
this.#setupPrimaryTargetListeners();
|
||||||
});
|
}
|
||||||
this.#tabSession?.on(CDPSessionEvent.Ready, session => {
|
|
||||||
|
async #onSecondaryTarget(session: CDPSession): Promise<void> {
|
||||||
assert(session instanceof CdpCDPSession);
|
assert(session instanceof CdpCDPSession);
|
||||||
if (session._target()._subtype() !== 'prerender') {
|
if (session._target()._subtype() !== 'prerender') {
|
||||||
return;
|
return;
|
||||||
@ -278,7 +316,6 @@ export class CdpPage extends Page {
|
|||||||
this.#emulationManager
|
this.#emulationManager
|
||||||
.registerSpeculativeSession(session)
|
.registerSpeculativeSession(session)
|
||||||
.catch(debugError);
|
.catch(debugError);
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -286,30 +323,15 @@ export class CdpPage extends Page {
|
|||||||
* during a navigation to a prerended page.
|
* during a navigation to a prerended page.
|
||||||
*/
|
*/
|
||||||
#setupPrimaryTargetListeners() {
|
#setupPrimaryTargetListeners() {
|
||||||
this.#client.on(CDPSessionEvent.Ready, this.#onAttachedToTarget);
|
this.#primaryTargetClient.on(
|
||||||
|
CDPSessionEvent.Ready,
|
||||||
|
this.#onAttachedToTarget
|
||||||
|
);
|
||||||
|
|
||||||
for (const [eventName, handler] of this.#sessionHandlers) {
|
for (const [eventName, handler] of this.#sessionHandlers) {
|
||||||
// TODO: Remove any.
|
// TODO: Remove any.
|
||||||
this.#client.on(eventName, handler as any);
|
this.#primaryTargetClient.on(eventName, handler as any);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.#target
|
|
||||||
._targetManager()
|
|
||||||
.on(TargetManagerEvent.TargetGone, this.#onDetachedFromTarget);
|
|
||||||
|
|
||||||
this.#target._isClosedDeferred
|
|
||||||
.valueOrThrow()
|
|
||||||
.then(() => {
|
|
||||||
this.#client.off(CDPSessionEvent.Ready, this.#onAttachedToTarget);
|
|
||||||
|
|
||||||
this.#target
|
|
||||||
._targetManager()
|
|
||||||
.off(TargetManagerEvent.TargetGone, this.#onDetachedFromTarget);
|
|
||||||
|
|
||||||
this.emit(PageEvent.Close, undefined);
|
|
||||||
this.#closed = true;
|
|
||||||
})
|
|
||||||
.catch(debugError);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#onDetachedFromTarget = (target: CdpTarget) => {
|
#onDetachedFromTarget = (target: CdpTarget) => {
|
||||||
@ -341,9 +363,9 @@ export class CdpPage extends Page {
|
|||||||
async #initialize(): Promise<void> {
|
async #initialize(): Promise<void> {
|
||||||
try {
|
try {
|
||||||
await Promise.all([
|
await Promise.all([
|
||||||
this.#frameManager.initialize(this.#client),
|
this.#frameManager.initialize(this.#primaryTargetClient),
|
||||||
this.#client.send('Performance.enable'),
|
this.#primaryTargetClient.send('Performance.enable'),
|
||||||
this.#client.send('Log.enable'),
|
this.#primaryTargetClient.send('Log.enable'),
|
||||||
]);
|
]);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
if (isErrorLike(err) && isTargetClosedError(err)) {
|
if (isErrorLike(err) && isTargetClosedError(err)) {
|
||||||
@ -377,7 +399,7 @@ export class CdpPage extends Page {
|
|||||||
}
|
}
|
||||||
|
|
||||||
_client(): CDPSession {
|
_client(): CDPSession {
|
||||||
return this.#client;
|
return this.#primaryTargetClient;
|
||||||
}
|
}
|
||||||
|
|
||||||
override isServiceWorkerBypassed(): boolean {
|
override isServiceWorkerBypassed(): boolean {
|
||||||
@ -404,9 +426,12 @@ export class CdpPage extends Page {
|
|||||||
this.#fileChooserDeferreds.add(deferred);
|
this.#fileChooserDeferreds.add(deferred);
|
||||||
let enablePromise: Promise<void> | undefined;
|
let enablePromise: Promise<void> | undefined;
|
||||||
if (needsEnable) {
|
if (needsEnable) {
|
||||||
enablePromise = this.#client.send('Page.setInterceptFileChooserDialog', {
|
enablePromise = this.#primaryTargetClient.send(
|
||||||
|
'Page.setInterceptFileChooserDialog',
|
||||||
|
{
|
||||||
enabled: true,
|
enabled: true,
|
||||||
});
|
}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
const [result] = await Promise.all([
|
const [result] = await Promise.all([
|
||||||
@ -425,15 +450,15 @@ export class CdpPage extends Page {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override target(): CdpTarget {
|
override target(): CdpTarget {
|
||||||
return this.#target;
|
return this.#primaryTarget;
|
||||||
}
|
}
|
||||||
|
|
||||||
override browser(): Browser {
|
override browser(): Browser {
|
||||||
return this.#target.browser();
|
return this.#primaryTarget.browser();
|
||||||
}
|
}
|
||||||
|
|
||||||
override browserContext(): BrowserContext {
|
override browserContext(): BrowserContext {
|
||||||
return this.#target.browserContext();
|
return this.#primaryTarget.browserContext();
|
||||||
}
|
}
|
||||||
|
|
||||||
#onTargetCrashed(): void {
|
#onTargetCrashed(): void {
|
||||||
@ -444,7 +469,7 @@ export class CdpPage extends Page {
|
|||||||
const {level, text, args, source, url, lineNumber} = event.entry;
|
const {level, text, args, source, url, lineNumber} = event.entry;
|
||||||
if (args) {
|
if (args) {
|
||||||
args.map(arg => {
|
args.map(arg => {
|
||||||
return releaseObject(this.#client, arg);
|
return releaseObject(this.#primaryTargetClient, arg);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
if (source !== 'worker') {
|
if (source !== 'worker') {
|
||||||
@ -495,12 +520,17 @@ export class CdpPage extends Page {
|
|||||||
|
|
||||||
override async setBypassServiceWorker(bypass: boolean): Promise<void> {
|
override async setBypassServiceWorker(bypass: boolean): Promise<void> {
|
||||||
this.#serviceWorkerBypassed = bypass;
|
this.#serviceWorkerBypassed = bypass;
|
||||||
return await this.#client.send('Network.setBypassServiceWorker', {bypass});
|
return await this.#primaryTargetClient.send(
|
||||||
|
'Network.setBypassServiceWorker',
|
||||||
|
{bypass}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
override async setDragInterception(enabled: boolean): Promise<void> {
|
override async setDragInterception(enabled: boolean): Promise<void> {
|
||||||
this.#userDragInterceptionEnabled = enabled;
|
this.#userDragInterceptionEnabled = enabled;
|
||||||
return await this.#client.send('Input.setInterceptDrags', {enabled});
|
return await this.#primaryTargetClient.send('Input.setInterceptDrags', {
|
||||||
|
enabled,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
override async setOfflineMode(enabled: boolean): Promise<void> {
|
override async setOfflineMode(enabled: boolean): Promise<void> {
|
||||||
@ -551,7 +581,7 @@ export class CdpPage extends Page {
|
|||||||
...urls: string[]
|
...urls: string[]
|
||||||
): Promise<Protocol.Network.Cookie[]> {
|
): Promise<Protocol.Network.Cookie[]> {
|
||||||
const originalCookies = (
|
const originalCookies = (
|
||||||
await this.#client.send('Network.getCookies', {
|
await this.#primaryTargetClient.send('Network.getCookies', {
|
||||||
urls: urls.length ? urls : [this.url()],
|
urls: urls.length ? urls : [this.url()],
|
||||||
})
|
})
|
||||||
).cookies;
|
).cookies;
|
||||||
@ -577,7 +607,7 @@ export class CdpPage extends Page {
|
|||||||
if (!cookie.url && pageURL.startsWith('http')) {
|
if (!cookie.url && pageURL.startsWith('http')) {
|
||||||
item.url = pageURL;
|
item.url = pageURL;
|
||||||
}
|
}
|
||||||
await this.#client.send('Network.deleteCookies', item);
|
await this.#primaryTargetClient.send('Network.deleteCookies', item);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -603,7 +633,9 @@ export class CdpPage extends Page {
|
|||||||
});
|
});
|
||||||
await this.deleteCookie(...items);
|
await this.deleteCookie(...items);
|
||||||
if (items.length) {
|
if (items.length) {
|
||||||
await this.#client.send('Network.setCookies', {cookies: items});
|
await this.#primaryTargetClient.send('Network.setCookies', {
|
||||||
|
cookies: items,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -636,8 +668,8 @@ export class CdpPage extends Page {
|
|||||||
this.#bindings.set(name, binding);
|
this.#bindings.set(name, binding);
|
||||||
|
|
||||||
const expression = pageBindingInitString('exposedFun', name);
|
const expression = pageBindingInitString('exposedFun', name);
|
||||||
await this.#client.send('Runtime.addBinding', {name});
|
await this.#primaryTargetClient.send('Runtime.addBinding', {name});
|
||||||
const {identifier} = await this.#client.send(
|
const {identifier} = await this.#primaryTargetClient.send(
|
||||||
'Page.addScriptToEvaluateOnNewDocument',
|
'Page.addScriptToEvaluateOnNewDocument',
|
||||||
{
|
{
|
||||||
source: expression,
|
source: expression,
|
||||||
@ -661,7 +693,7 @@ export class CdpPage extends Page {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
await this.#client.send('Runtime.removeBinding', {name});
|
await this.#primaryTargetClient.send('Runtime.removeBinding', {name});
|
||||||
await this.removeScriptToEvaluateOnNewDocument(exposedFun);
|
await this.removeScriptToEvaluateOnNewDocument(exposedFun);
|
||||||
|
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
@ -701,7 +733,9 @@ export class CdpPage extends Page {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override async metrics(): Promise<Metrics> {
|
override async metrics(): Promise<Metrics> {
|
||||||
const response = await this.#client.send('Performance.getMetrics');
|
const response = await this.#primaryTargetClient.send(
|
||||||
|
'Performance.getMetrics'
|
||||||
|
);
|
||||||
return this.#buildMetricsObject(response.metrics);
|
return this.#buildMetricsObject(response.metrics);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -753,7 +787,7 @@ export class CdpPage extends Page {
|
|||||||
}
|
}
|
||||||
const context = this.#frameManager.getExecutionContextById(
|
const context = this.#frameManager.getExecutionContextById(
|
||||||
event.executionContextId,
|
event.executionContextId,
|
||||||
this.#client
|
this.#primaryTargetClient
|
||||||
);
|
);
|
||||||
if (!context) {
|
if (!context) {
|
||||||
debugError(
|
debugError(
|
||||||
@ -789,7 +823,7 @@ export class CdpPage extends Page {
|
|||||||
|
|
||||||
const context = this.#frameManager.executionContextById(
|
const context = this.#frameManager.executionContextById(
|
||||||
event.executionContextId,
|
event.executionContextId,
|
||||||
this.#client
|
this.#primaryTargetClient
|
||||||
);
|
);
|
||||||
if (!context) {
|
if (!context) {
|
||||||
return;
|
return;
|
||||||
@ -843,7 +877,7 @@ export class CdpPage extends Page {
|
|||||||
#onDialog(event: Protocol.Page.JavascriptDialogOpeningEvent): void {
|
#onDialog(event: Protocol.Page.JavascriptDialogOpeningEvent): void {
|
||||||
const type = validateDialogType(event.type);
|
const type = validateDialogType(event.type);
|
||||||
const dialog = new CdpDialog(
|
const dialog = new CdpDialog(
|
||||||
this.#client,
|
this.#primaryTargetClient,
|
||||||
type,
|
type,
|
||||||
event.message,
|
event.message,
|
||||||
event.defaultPrompt
|
event.defaultPrompt
|
||||||
@ -856,7 +890,7 @@ export class CdpPage extends Page {
|
|||||||
): Promise<HTTPResponse | null> {
|
): Promise<HTTPResponse | null> {
|
||||||
const [result] = await Promise.all([
|
const [result] = await Promise.all([
|
||||||
this.waitForNavigation(options),
|
this.waitForNavigation(options),
|
||||||
this.#client.send('Page.reload'),
|
this.#primaryTargetClient.send('Page.reload'),
|
||||||
]);
|
]);
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
@ -925,20 +959,24 @@ export class CdpPage extends Page {
|
|||||||
delta: number,
|
delta: number,
|
||||||
options: WaitForOptions
|
options: WaitForOptions
|
||||||
): Promise<HTTPResponse | null> {
|
): Promise<HTTPResponse | null> {
|
||||||
const history = await this.#client.send('Page.getNavigationHistory');
|
const history = await this.#primaryTargetClient.send(
|
||||||
|
'Page.getNavigationHistory'
|
||||||
|
);
|
||||||
const entry = history.entries[history.currentIndex + delta];
|
const entry = history.entries[history.currentIndex + delta];
|
||||||
if (!entry) {
|
if (!entry) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
const result = await Promise.all([
|
const result = await Promise.all([
|
||||||
this.waitForNavigation(options),
|
this.waitForNavigation(options),
|
||||||
this.#client.send('Page.navigateToHistoryEntry', {entryId: entry.id}),
|
this.#primaryTargetClient.send('Page.navigateToHistoryEntry', {
|
||||||
|
entryId: entry.id,
|
||||||
|
}),
|
||||||
]);
|
]);
|
||||||
return result[0];
|
return result[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
override async bringToFront(): Promise<void> {
|
override async bringToFront(): Promise<void> {
|
||||||
await this.#client.send('Page.bringToFront');
|
await this.#primaryTargetClient.send('Page.bringToFront');
|
||||||
}
|
}
|
||||||
|
|
||||||
override async setJavaScriptEnabled(enabled: boolean): Promise<void> {
|
override async setJavaScriptEnabled(enabled: boolean): Promise<void> {
|
||||||
@ -946,7 +984,7 @@ export class CdpPage extends Page {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override async setBypassCSP(enabled: boolean): Promise<void> {
|
override async setBypassCSP(enabled: boolean): Promise<void> {
|
||||||
await this.#client.send('Page.setBypassCSP', {enabled});
|
await this.#primaryTargetClient.send('Page.setBypassCSP', {enabled});
|
||||||
}
|
}
|
||||||
|
|
||||||
override async emulateMediaType(type?: string): Promise<void> {
|
override async emulateMediaType(type?: string): Promise<void> {
|
||||||
@ -1000,7 +1038,7 @@ export class CdpPage extends Page {
|
|||||||
...args: Params
|
...args: Params
|
||||||
): Promise<NewDocumentScriptEvaluation> {
|
): Promise<NewDocumentScriptEvaluation> {
|
||||||
const source = evaluationString(pageFunction, ...args);
|
const source = evaluationString(pageFunction, ...args);
|
||||||
const {identifier} = await this.#client.send(
|
const {identifier} = await this.#primaryTargetClient.send(
|
||||||
'Page.addScriptToEvaluateOnNewDocument',
|
'Page.addScriptToEvaluateOnNewDocument',
|
||||||
{
|
{
|
||||||
source,
|
source,
|
||||||
@ -1013,9 +1051,12 @@ export class CdpPage extends Page {
|
|||||||
override async removeScriptToEvaluateOnNewDocument(
|
override async removeScriptToEvaluateOnNewDocument(
|
||||||
identifier: string
|
identifier: string
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
await this.#client.send('Page.removeScriptToEvaluateOnNewDocument', {
|
await this.#primaryTargetClient.send(
|
||||||
|
'Page.removeScriptToEvaluateOnNewDocument',
|
||||||
|
{
|
||||||
identifier,
|
identifier,
|
||||||
});
|
}
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
override async setCacheEnabled(enabled = true): Promise<void> {
|
override async setCacheEnabled(enabled = true): Promise<void> {
|
||||||
@ -1066,7 +1107,9 @@ export class CdpPage extends Page {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// We need to do these spreads because Firefox doesn't allow unknown options.
|
// We need to do these spreads because Firefox doesn't allow unknown options.
|
||||||
const {data} = await this.#client.send('Page.captureScreenshot', {
|
const {data} = await this.#primaryTargetClient.send(
|
||||||
|
'Page.captureScreenshot',
|
||||||
|
{
|
||||||
format: type,
|
format: type,
|
||||||
...(optimizeForSpeed ? {optimizeForSpeed} : {}),
|
...(optimizeForSpeed ? {optimizeForSpeed} : {}),
|
||||||
...(quality !== undefined ? {quality: Math.round(quality)} : {}),
|
...(quality !== undefined ? {quality: Math.round(quality)} : {}),
|
||||||
@ -1076,7 +1119,8 @@ export class CdpPage extends Page {
|
|||||||
},
|
},
|
||||||
...(!fromSurface ? {fromSurface} : {}),
|
...(!fromSurface ? {fromSurface} : {}),
|
||||||
captureBeyondViewport,
|
captureBeyondViewport,
|
||||||
});
|
}
|
||||||
|
);
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1101,7 +1145,9 @@ export class CdpPage extends Page {
|
|||||||
await this.#emulationManager.setTransparentBackgroundColor();
|
await this.#emulationManager.setTransparentBackgroundColor();
|
||||||
}
|
}
|
||||||
|
|
||||||
const printCommandPromise = this.#client.send('Page.printToPDF', {
|
const printCommandPromise = this.#primaryTargetClient.send(
|
||||||
|
'Page.printToPDF',
|
||||||
|
{
|
||||||
transferMode: 'ReturnAsStream',
|
transferMode: 'ReturnAsStream',
|
||||||
landscape,
|
landscape,
|
||||||
displayHeaderFooter,
|
displayHeaderFooter,
|
||||||
@ -1117,7 +1163,8 @@ export class CdpPage extends Page {
|
|||||||
marginRight: margin.right,
|
marginRight: margin.right,
|
||||||
pageRanges,
|
pageRanges,
|
||||||
preferCSSPageSize,
|
preferCSSPageSize,
|
||||||
});
|
}
|
||||||
|
);
|
||||||
|
|
||||||
const result = await waitWithTimeout(
|
const result = await waitWithTimeout(
|
||||||
printCommandPromise,
|
printCommandPromise,
|
||||||
@ -1130,7 +1177,10 @@ export class CdpPage extends Page {
|
|||||||
}
|
}
|
||||||
|
|
||||||
assert(result.stream, '`stream` is missing from `Page.printToPDF');
|
assert(result.stream, '`stream` is missing from `Page.printToPDF');
|
||||||
return await getReadableFromProtocolStream(this.#client, result.stream);
|
return await getReadableFromProtocolStream(
|
||||||
|
this.#primaryTargetClient,
|
||||||
|
result.stream
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
override async pdf(options: PDFOptions = {}): Promise<Buffer> {
|
override async pdf(options: PDFOptions = {}): Promise<Buffer> {
|
||||||
@ -1144,19 +1194,19 @@ export class CdpPage extends Page {
|
|||||||
override async close(
|
override async close(
|
||||||
options: {runBeforeUnload?: boolean} = {runBeforeUnload: undefined}
|
options: {runBeforeUnload?: boolean} = {runBeforeUnload: undefined}
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
const connection = this.#client.connection();
|
const connection = this.#primaryTargetClient.connection();
|
||||||
assert(
|
assert(
|
||||||
connection,
|
connection,
|
||||||
'Protocol error: Connection closed. Most likely the page has been closed.'
|
'Protocol error: Connection closed. Most likely the page has been closed.'
|
||||||
);
|
);
|
||||||
const runBeforeUnload = !!options.runBeforeUnload;
|
const runBeforeUnload = !!options.runBeforeUnload;
|
||||||
if (runBeforeUnload) {
|
if (runBeforeUnload) {
|
||||||
await this.#client.send('Page.close');
|
await this.#primaryTargetClient.send('Page.close');
|
||||||
} else {
|
} else {
|
||||||
await connection.send('Target.closeTarget', {
|
await connection.send('Target.closeTarget', {
|
||||||
targetId: this.#target._targetId,
|
targetId: this.#primaryTarget._targetId,
|
||||||
});
|
});
|
||||||
await this.#target._isClosedDeferred.valueOrThrow();
|
await this.#tabTarget._isClosedDeferred.valueOrThrow();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user