2019-11-18 18:18:28 -08:00
|
|
|
/**
|
|
|
|
* Copyright 2017 Google Inc. All rights reserved.
|
|
|
|
* Modifications copyright (c) Microsoft Corporation.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2019-11-27 12:38:26 -08:00
|
|
|
import * as types from './types';
|
2019-11-26 08:57:53 -08:00
|
|
|
import * as fs from 'fs';
|
2019-11-27 12:39:53 -08:00
|
|
|
import * as js from './javascript';
|
2019-11-27 16:02:31 -08:00
|
|
|
import * as dom from './dom';
|
2019-11-27 12:44:12 -08:00
|
|
|
import * as network from './network';
|
2019-11-27 12:38:26 -08:00
|
|
|
import { helper, assert } from './helper';
|
|
|
|
import { ClickOptions, MultiClickOptions, PointerActionOptions, SelectOption } from './input';
|
2019-12-03 10:43:13 -08:00
|
|
|
import { WaitTaskParams, WaitTask } from './waitTask';
|
2019-11-27 12:38:26 -08:00
|
|
|
import { TimeoutSettings } from './TimeoutSettings';
|
2019-11-26 08:57:53 -08:00
|
|
|
|
|
|
|
const readFileAsync = helper.promisify(fs.readFile);
|
2019-11-18 18:18:28 -08:00
|
|
|
|
2019-11-26 15:37:25 -08:00
|
|
|
type WorldType = 'main' | 'utility';
|
2019-11-27 16:02:31 -08:00
|
|
|
type World = {
|
|
|
|
contextPromise: Promise<js.ExecutionContext>;
|
|
|
|
contextResolveCallback: (c: js.ExecutionContext) => void;
|
|
|
|
context: js.ExecutionContext | null;
|
|
|
|
waitTasks: Set<WaitTask>;
|
2019-11-26 15:37:25 -08:00
|
|
|
};
|
|
|
|
|
2019-11-26 16:19:43 -08:00
|
|
|
export type NavigateOptions = {
|
|
|
|
timeout?: number,
|
|
|
|
waitUntil?: string | string[],
|
|
|
|
};
|
|
|
|
|
|
|
|
export type GotoOptions = NavigateOptions & {
|
|
|
|
referer?: string,
|
|
|
|
};
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
export interface FrameDelegate {
|
|
|
|
navigateFrame(frame: Frame, url: string, options?: GotoOptions): Promise<network.Response | null>;
|
|
|
|
waitForFrameNavigation(frame: Frame, options?: NavigateOptions): Promise<network.Response | null>;
|
|
|
|
setFrameContent(frame: Frame, html: string, options?: NavigateOptions): Promise<void>;
|
2019-11-26 16:19:43 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
export class Frame {
|
|
|
|
_delegate: FrameDelegate;
|
2019-11-27 16:03:51 -08:00
|
|
|
private _timeoutSettings: TimeoutSettings;
|
2019-11-27 16:02:31 -08:00
|
|
|
private _parentFrame: Frame;
|
2019-11-18 18:18:28 -08:00
|
|
|
private _url = '';
|
|
|
|
private _detached = false;
|
2019-11-27 16:02:31 -08:00
|
|
|
private _worlds = new Map<WorldType, World>();
|
|
|
|
private _childFrames = new Set<Frame>();
|
2019-11-18 18:18:28 -08:00
|
|
|
private _name: string;
|
|
|
|
|
2019-11-27 16:03:51 -08:00
|
|
|
constructor(delegate: FrameDelegate, timeoutSettings: TimeoutSettings, parentFrame: Frame | null) {
|
2019-11-26 16:19:43 -08:00
|
|
|
this._delegate = delegate;
|
2019-11-27 16:03:51 -08:00
|
|
|
this._timeoutSettings = timeoutSettings;
|
2019-11-18 18:18:28 -08:00
|
|
|
this._parentFrame = parentFrame;
|
|
|
|
|
2019-11-26 15:37:25 -08:00
|
|
|
this._worlds.set('main', { contextPromise: new Promise(() => {}), contextResolveCallback: () => {}, context: null, waitTasks: new Set() });
|
|
|
|
this._worlds.set('utility', { contextPromise: new Promise(() => {}), contextResolveCallback: () => {}, context: null, waitTasks: new Set() });
|
|
|
|
this._setContext('main', null);
|
|
|
|
this._setContext('utility', null);
|
2019-11-18 18:18:28 -08:00
|
|
|
|
|
|
|
if (this._parentFrame)
|
|
|
|
this._parentFrame._childFrames.add(this);
|
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
goto(url: string, options?: GotoOptions): Promise<network.Response | null> {
|
2019-11-26 16:19:43 -08:00
|
|
|
return this._delegate.navigateFrame(this, url, options);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
waitForNavigation(options?: NavigateOptions): Promise<network.Response | null> {
|
2019-11-26 16:19:43 -08:00
|
|
|
return this._delegate.waitForFrameNavigation(this, options);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
_mainContext(): Promise<js.ExecutionContext> {
|
2019-11-26 15:37:25 -08:00
|
|
|
if (this._detached)
|
|
|
|
throw new Error(`Execution Context is not available in detached frame "${this.url()}" (are you trying to evaluate?)`);
|
|
|
|
return this._worlds.get('main').contextPromise;
|
|
|
|
}
|
|
|
|
|
2019-11-28 12:50:52 -08:00
|
|
|
async _mainDOMWorld(): Promise<dom.DOMWorld> {
|
|
|
|
const context = await this._mainContext();
|
|
|
|
if (!context._domWorld)
|
|
|
|
throw new Error(`Execution Context does not belong to frame`);
|
|
|
|
return context._domWorld;
|
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
_utilityContext(): Promise<js.ExecutionContext> {
|
2019-11-26 15:37:25 -08:00
|
|
|
if (this._detached)
|
|
|
|
throw new Error(`Execution Context is not available in detached frame "${this.url()}" (are you trying to evaluate?)`);
|
|
|
|
return this._worlds.get('utility').contextPromise;
|
|
|
|
}
|
|
|
|
|
2019-11-28 12:50:52 -08:00
|
|
|
async _utilityDOMWorld(): Promise<dom.DOMWorld> {
|
|
|
|
const context = await this._utilityContext();
|
|
|
|
if (!context._domWorld)
|
|
|
|
throw new Error(`Execution Context does not belong to frame`);
|
|
|
|
return context._domWorld;
|
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
executionContext(): Promise<js.ExecutionContext> {
|
2019-11-26 15:37:25 -08:00
|
|
|
return this._mainContext();
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:03:51 -08:00
|
|
|
evaluateHandle: types.EvaluateHandle = async (pageFunction, ...args) => {
|
2019-11-26 15:37:25 -08:00
|
|
|
const context = await this._mainContext();
|
2019-11-26 08:57:53 -08:00
|
|
|
return context.evaluateHandle(pageFunction, ...args as any);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:03:51 -08:00
|
|
|
evaluate: types.Evaluate = async (pageFunction, ...args) => {
|
2019-11-26 15:37:25 -08:00
|
|
|
const context = await this._mainContext();
|
2019-11-26 08:57:53 -08:00
|
|
|
return context.evaluate(pageFunction, ...args as any);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
async $(selector: string): Promise<dom.ElementHandle | null> {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._mainDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
return domWorld.$(selector);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
async $x(expression: string): Promise<dom.ElementHandle[]> {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._mainDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
return domWorld.$$('xpath=' + expression);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:03:51 -08:00
|
|
|
$eval: types.$Eval = async (selector, pageFunction, ...args) => {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._mainDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
return domWorld.$eval(selector, pageFunction, ...args as any);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:03:51 -08:00
|
|
|
$$eval: types.$$Eval = async (selector, pageFunction, ...args) => {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._mainDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
return domWorld.$$eval(selector, pageFunction, ...args as any);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
async $$(selector: string): Promise<dom.ElementHandle[]> {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._mainDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
return domWorld.$$(selector);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
async content(): Promise<string> {
|
2019-11-26 15:37:25 -08:00
|
|
|
const context = await this._utilityContext();
|
2019-11-26 08:57:53 -08:00
|
|
|
return context.evaluate(() => {
|
|
|
|
let retVal = '';
|
|
|
|
if (document.doctype)
|
|
|
|
retVal = new XMLSerializer().serializeToString(document.doctype);
|
|
|
|
if (document.documentElement)
|
|
|
|
retVal += document.documentElement.outerHTML;
|
|
|
|
return retVal;
|
|
|
|
});
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-26 16:19:43 -08:00
|
|
|
setContent(html: string, options?: NavigateOptions) {
|
|
|
|
return this._delegate.setFrameContent(this, html, options);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
name(): string {
|
|
|
|
return this._name || '';
|
|
|
|
}
|
|
|
|
|
|
|
|
url(): string {
|
|
|
|
return this._url;
|
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
parentFrame(): Frame | null {
|
2019-11-18 18:18:28 -08:00
|
|
|
return this._parentFrame;
|
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
childFrames(): Frame[] {
|
2019-11-18 18:18:28 -08:00
|
|
|
return Array.from(this._childFrames);
|
|
|
|
}
|
|
|
|
|
|
|
|
isDetached(): boolean {
|
|
|
|
return this._detached;
|
|
|
|
}
|
|
|
|
|
|
|
|
async addScriptTag(options: {
|
|
|
|
url?: string; path?: string;
|
|
|
|
content?: string;
|
2019-11-26 08:57:53 -08:00
|
|
|
type?: string;
|
2019-11-27 16:02:31 -08:00
|
|
|
}): Promise<dom.ElementHandle> {
|
2019-11-26 08:57:53 -08:00
|
|
|
const {
|
|
|
|
url = null,
|
|
|
|
path = null,
|
|
|
|
content = null,
|
|
|
|
type = ''
|
|
|
|
} = options;
|
|
|
|
if (url !== null) {
|
|
|
|
try {
|
2019-11-26 15:37:25 -08:00
|
|
|
const context = await this._mainContext();
|
2019-11-26 08:57:53 -08:00
|
|
|
return (await context.evaluateHandle(addScriptUrl, url, type)).asElement();
|
|
|
|
} catch (error) {
|
|
|
|
throw new Error(`Loading script from ${url} failed`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (path !== null) {
|
|
|
|
let contents = await readFileAsync(path, 'utf8');
|
|
|
|
contents += '//# sourceURL=' + path.replace(/\n/g, '');
|
2019-11-26 15:37:25 -08:00
|
|
|
const context = await this._mainContext();
|
2019-11-26 08:57:53 -08:00
|
|
|
return (await context.evaluateHandle(addScriptContent, contents, type)).asElement();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (content !== null) {
|
2019-11-26 15:37:25 -08:00
|
|
|
const context = await this._mainContext();
|
2019-11-26 08:57:53 -08:00
|
|
|
return (await context.evaluateHandle(addScriptContent, content, type)).asElement();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error('Provide an object with a `url`, `path` or `content` property');
|
|
|
|
|
|
|
|
async function addScriptUrl(url: string, type: string): Promise<HTMLElement> {
|
|
|
|
const script = document.createElement('script');
|
|
|
|
script.src = url;
|
|
|
|
if (type)
|
|
|
|
script.type = type;
|
|
|
|
const promise = new Promise((res, rej) => {
|
|
|
|
script.onload = res;
|
|
|
|
script.onerror = rej;
|
|
|
|
});
|
|
|
|
document.head.appendChild(script);
|
|
|
|
await promise;
|
|
|
|
return script;
|
|
|
|
}
|
|
|
|
|
|
|
|
function addScriptContent(content: string, type: string = 'text/javascript'): HTMLElement {
|
|
|
|
const script = document.createElement('script');
|
|
|
|
script.type = type;
|
|
|
|
script.text = content;
|
|
|
|
let error = null;
|
|
|
|
script.onerror = e => error = e;
|
|
|
|
document.head.appendChild(script);
|
|
|
|
if (error)
|
|
|
|
throw error;
|
|
|
|
return script;
|
|
|
|
}
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
async addStyleTag(options: { url?: string; path?: string; content?: string; }): Promise<dom.ElementHandle> {
|
2019-11-26 08:57:53 -08:00
|
|
|
const {
|
|
|
|
url = null,
|
|
|
|
path = null,
|
|
|
|
content = null
|
|
|
|
} = options;
|
|
|
|
if (url !== null) {
|
|
|
|
try {
|
2019-11-26 15:37:25 -08:00
|
|
|
const context = await this._mainContext();
|
2019-11-26 08:57:53 -08:00
|
|
|
return (await context.evaluateHandle(addStyleUrl, url)).asElement();
|
|
|
|
} catch (error) {
|
|
|
|
throw new Error(`Loading style from ${url} failed`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (path !== null) {
|
|
|
|
let contents = await readFileAsync(path, 'utf8');
|
|
|
|
contents += '/*# sourceURL=' + path.replace(/\n/g, '') + '*/';
|
2019-11-26 15:37:25 -08:00
|
|
|
const context = await this._mainContext();
|
2019-11-26 08:57:53 -08:00
|
|
|
return (await context.evaluateHandle(addStyleContent, contents)).asElement();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (content !== null) {
|
2019-11-26 15:37:25 -08:00
|
|
|
const context = await this._mainContext();
|
2019-11-26 08:57:53 -08:00
|
|
|
return (await context.evaluateHandle(addStyleContent, content)).asElement();
|
|
|
|
}
|
|
|
|
|
|
|
|
throw new Error('Provide an object with a `url`, `path` or `content` property');
|
|
|
|
|
|
|
|
async function addStyleUrl(url: string): Promise<HTMLElement> {
|
|
|
|
const link = document.createElement('link');
|
|
|
|
link.rel = 'stylesheet';
|
|
|
|
link.href = url;
|
|
|
|
const promise = new Promise((res, rej) => {
|
|
|
|
link.onload = res;
|
|
|
|
link.onerror = rej;
|
|
|
|
});
|
|
|
|
document.head.appendChild(link);
|
|
|
|
await promise;
|
|
|
|
return link;
|
|
|
|
}
|
|
|
|
|
|
|
|
async function addStyleContent(content: string): Promise<HTMLElement> {
|
|
|
|
const style = document.createElement('style');
|
|
|
|
style.type = 'text/css';
|
|
|
|
style.appendChild(document.createTextNode(content));
|
|
|
|
const promise = new Promise((res, rej) => {
|
|
|
|
style.onload = res;
|
|
|
|
style.onerror = rej;
|
|
|
|
});
|
|
|
|
document.head.appendChild(style);
|
|
|
|
await promise;
|
|
|
|
return style;
|
|
|
|
}
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
async click(selector: string, options?: ClickOptions) {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._utilityDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
const handle = await domWorld.$(selector);
|
2019-11-22 14:46:34 -08:00
|
|
|
assert(handle, 'No node found for selector: ' + selector);
|
|
|
|
await handle.click(options);
|
|
|
|
await handle.dispose();
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
async dblclick(selector: string, options?: MultiClickOptions) {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._utilityDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
const handle = await domWorld.$(selector);
|
2019-11-22 14:46:34 -08:00
|
|
|
assert(handle, 'No node found for selector: ' + selector);
|
|
|
|
await handle.dblclick(options);
|
|
|
|
await handle.dispose();
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
async tripleclick(selector: string, options?: MultiClickOptions) {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._utilityDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
const handle = await domWorld.$(selector);
|
2019-11-22 14:46:34 -08:00
|
|
|
assert(handle, 'No node found for selector: ' + selector);
|
|
|
|
await handle.tripleclick(options);
|
|
|
|
await handle.dispose();
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
async fill(selector: string, value: string) {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._utilityDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
const handle = await domWorld.$(selector);
|
2019-11-22 14:46:34 -08:00
|
|
|
assert(handle, 'No node found for selector: ' + selector);
|
|
|
|
await handle.fill(value);
|
|
|
|
await handle.dispose();
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
async focus(selector: string) {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._utilityDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
const handle = await domWorld.$(selector);
|
2019-11-22 14:46:34 -08:00
|
|
|
assert(handle, 'No node found for selector: ' + selector);
|
|
|
|
await handle.focus();
|
|
|
|
await handle.dispose();
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
async hover(selector: string, options?: PointerActionOptions) {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._utilityDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
const handle = await domWorld.$(selector);
|
2019-11-22 14:46:34 -08:00
|
|
|
assert(handle, 'No node found for selector: ' + selector);
|
|
|
|
await handle.hover(options);
|
|
|
|
await handle.dispose();
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
async select(selector: string, ...values: (string | dom.ElementHandle | SelectOption)[]): Promise<string[]> {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._utilityDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
const handle = await domWorld.$(selector);
|
2019-11-22 14:46:34 -08:00
|
|
|
assert(handle, 'No node found for selector: ' + selector);
|
2019-12-02 17:33:44 -08:00
|
|
|
const toDispose: Promise<dom.ElementHandle>[] = [];
|
2019-11-27 12:38:26 -08:00
|
|
|
const adoptedValues = await Promise.all(values.map(async value => {
|
2019-12-02 17:33:44 -08:00
|
|
|
if (value instanceof dom.ElementHandle && value.executionContext() !== domWorld.context) {
|
|
|
|
const adopted = domWorld.adoptElementHandle(value);
|
|
|
|
toDispose.push(adopted);
|
|
|
|
return adopted;
|
|
|
|
}
|
2019-11-27 12:38:26 -08:00
|
|
|
return value;
|
|
|
|
}));
|
2019-11-22 14:46:34 -08:00
|
|
|
const result = await handle.select(...adoptedValues);
|
|
|
|
await handle.dispose();
|
2019-12-02 17:33:44 -08:00
|
|
|
await Promise.all(toDispose.map(handlePromise => handlePromise.then(handle => handle.dispose())));
|
2019-11-22 14:46:34 -08:00
|
|
|
return result;
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
async type(selector: string, text: string, options: { delay: (number | undefined); } | undefined) {
|
2019-11-28 12:50:52 -08:00
|
|
|
const domWorld = await this._utilityDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
const handle = await domWorld.$(selector);
|
2019-11-22 14:46:34 -08:00
|
|
|
assert(handle, 'No node found for selector: ' + selector);
|
|
|
|
await handle.type(text, options);
|
|
|
|
await handle.dispose();
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
waitFor(selectorOrFunctionOrTimeout: (string | number | Function), options: any = {}, ...args: any[]): Promise<js.JSHandle | null> {
|
2019-12-03 10:43:13 -08:00
|
|
|
if (helper.isString(selectorOrFunctionOrTimeout))
|
|
|
|
return this.waitForSelector(selectorOrFunctionOrTimeout as string, options) as any;
|
2019-11-18 18:18:28 -08:00
|
|
|
if (helper.isNumber(selectorOrFunctionOrTimeout))
|
|
|
|
return new Promise(fulfill => setTimeout(fulfill, selectorOrFunctionOrTimeout as number));
|
|
|
|
if (typeof selectorOrFunctionOrTimeout === 'function')
|
|
|
|
return this.waitForFunction(selectorOrFunctionOrTimeout, options, ...args);
|
|
|
|
return Promise.reject(new Error('Unsupported target type: ' + (typeof selectorOrFunctionOrTimeout)));
|
|
|
|
}
|
|
|
|
|
2019-12-03 10:43:13 -08:00
|
|
|
async waitForSelector(selector: string, options: dom.WaitForSelectorOptions = {}): Promise<dom.ElementHandle | null> {
|
|
|
|
const params = dom.waitForSelectorTask(selector, { timeout: this._timeoutSettings.timeout(), ...options });
|
|
|
|
const handle = await this._scheduleWaitTask(params, 'utility');
|
2019-11-26 15:37:25 -08:00
|
|
|
if (!handle.asElement()) {
|
|
|
|
await handle.dispose();
|
|
|
|
return null;
|
2019-11-26 11:16:20 -08:00
|
|
|
}
|
2019-11-28 12:50:52 -08:00
|
|
|
const mainDOMWorld = await this._mainDOMWorld();
|
2019-12-02 17:33:44 -08:00
|
|
|
if (handle.executionContext() === mainDOMWorld.context)
|
|
|
|
return handle.asElement();
|
|
|
|
const adopted = await mainDOMWorld.adoptElementHandle(handle.asElement());
|
|
|
|
await handle.dispose();
|
|
|
|
return adopted;
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-12-03 10:43:13 -08:00
|
|
|
async waitForXPath(xpath: string, options: dom.WaitForSelectorOptions = {}): Promise<dom.ElementHandle | null> {
|
|
|
|
return this.waitForSelector('xpath=' + xpath, options);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
waitForFunction(
|
|
|
|
pageFunction: Function | string,
|
|
|
|
options: { polling?: string | number; timeout?: number; } = {},
|
2019-11-27 16:02:31 -08:00
|
|
|
...args): Promise<js.JSHandle> {
|
2019-11-26 11:16:20 -08:00
|
|
|
const {
|
|
|
|
polling = 'raf',
|
2019-11-27 16:03:51 -08:00
|
|
|
timeout = this._timeoutSettings.timeout(),
|
2019-11-26 11:16:20 -08:00
|
|
|
} = options;
|
|
|
|
const params: WaitTaskParams = {
|
|
|
|
predicateBody: pageFunction,
|
|
|
|
title: 'function',
|
|
|
|
polling,
|
|
|
|
timeout,
|
|
|
|
args
|
|
|
|
};
|
2019-12-03 10:43:13 -08:00
|
|
|
return this._scheduleWaitTask(params, 'main');
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
async title(): Promise<string> {
|
2019-11-26 15:37:25 -08:00
|
|
|
const context = await this._utilityContext();
|
2019-11-26 08:57:53 -08:00
|
|
|
return context.evaluate(() => document.title);
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
2019-11-26 16:19:43 -08:00
|
|
|
_navigated(url: string, name: string) {
|
2019-11-18 18:18:28 -08:00
|
|
|
this._url = url;
|
2019-11-26 16:19:43 -08:00
|
|
|
this._name = name;
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
_detach() {
|
|
|
|
this._detached = true;
|
2019-11-26 15:37:25 -08:00
|
|
|
for (const world of this._worlds.values()) {
|
|
|
|
for (const waitTask of world.waitTasks)
|
|
|
|
waitTask.terminate(new Error('waitForFunction failed: frame got detached.'));
|
|
|
|
}
|
2019-11-18 18:18:28 -08:00
|
|
|
if (this._parentFrame)
|
|
|
|
this._parentFrame._childFrames.delete(this);
|
|
|
|
this._parentFrame = null;
|
|
|
|
}
|
2019-11-26 15:37:25 -08:00
|
|
|
|
2019-12-03 10:43:13 -08:00
|
|
|
private _scheduleWaitTask(params: WaitTaskParams, worldType: WorldType): Promise<js.JSHandle> {
|
|
|
|
const world = this._worlds.get(worldType);
|
2019-11-26 15:37:25 -08:00
|
|
|
const task = new WaitTask(params, () => world.waitTasks.delete(task));
|
|
|
|
world.waitTasks.add(task);
|
|
|
|
if (world.context)
|
|
|
|
task.rerun(world.context);
|
|
|
|
return task.promise;
|
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
private _setContext(worldType: WorldType, context: js.ExecutionContext | null) {
|
2019-11-26 15:37:25 -08:00
|
|
|
const world = this._worlds.get(worldType);
|
|
|
|
world.context = context;
|
|
|
|
if (context) {
|
|
|
|
world.contextResolveCallback.call(null, context);
|
|
|
|
for (const waitTask of world.waitTasks)
|
|
|
|
waitTask.rerun(context);
|
|
|
|
} else {
|
|
|
|
world.contextPromise = new Promise(fulfill => {
|
|
|
|
world.contextResolveCallback = fulfill;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
_contextCreated(worldType: WorldType, context: js.ExecutionContext) {
|
2019-11-26 15:37:25 -08:00
|
|
|
const world = this._worlds.get(worldType);
|
|
|
|
// In case of multiple sessions to the same target, there's a race between
|
|
|
|
// connections so we might end up creating multiple isolated worlds.
|
|
|
|
// We can use either.
|
2019-12-03 11:47:02 -07:00
|
|
|
if (world.context)
|
|
|
|
this._setContext(worldType, null);
|
|
|
|
this._setContext(worldType, context);
|
2019-11-26 15:37:25 -08:00
|
|
|
}
|
|
|
|
|
2019-11-27 16:02:31 -08:00
|
|
|
_contextDestroyed(context: js.ExecutionContext) {
|
2019-11-26 15:37:25 -08:00
|
|
|
for (const [worldType, world] of this._worlds) {
|
|
|
|
if (world.context === context)
|
|
|
|
this._setContext(worldType, null);
|
|
|
|
}
|
|
|
|
}
|
2019-11-18 18:18:28 -08:00
|
|
|
}
|