2020-04-30 11:45:52 +00:00
|
|
|
/**
|
2023-02-14 21:31:30 +00:00
|
|
|
* Copyright 2023 Google Inc. All rights reserved.
|
2020-04-30 11:45:52 +00:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2023-02-09 17:04:06 +00:00
|
|
|
import {ElementHandle} from '../api/ElementHandle.js';
|
2023-02-14 21:31:30 +00:00
|
|
|
import type PuppeteerUtil from '../injected/injected.js';
|
|
|
|
import {assert} from '../util/assert.js';
|
|
|
|
import {createFunction} from '../util/Function.js';
|
|
|
|
import {transposeIterableHandle} from './HandleIterator.js';
|
|
|
|
import type {Frame} from './Frame.js';
|
|
|
|
import type {WaitForSelectorOptions} from './IsolatedWorld.js';
|
2022-11-10 16:11:18 +00:00
|
|
|
import {MAIN_WORLD, PUPPETEER_WORLD} from './IsolatedWorlds.js';
|
2023-01-27 19:58:40 +00:00
|
|
|
import {LazyArg} from './LazyArg.js';
|
2023-02-14 21:31:30 +00:00
|
|
|
import type {Awaitable, AwaitableIterable} from './types.js';
|
2020-09-23 14:02:22 +00:00
|
|
|
|
2022-08-04 13:45:21 +00:00
|
|
|
/**
|
2023-02-14 21:31:30 +00:00
|
|
|
* @internal
|
2022-08-04 13:45:21 +00:00
|
|
|
*/
|
2023-02-14 21:31:30 +00:00
|
|
|
export type QuerySelectorAll = (
|
|
|
|
node: Node,
|
|
|
|
selector: string,
|
|
|
|
PuppeteerUtil: PuppeteerUtil
|
|
|
|
) => AwaitableIterable<Node>;
|
2022-08-04 13:45:21 +00:00
|
|
|
|
2022-09-15 11:12:13 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2023-02-14 21:31:30 +00:00
|
|
|
export type QuerySelector = (
|
|
|
|
node: Node,
|
|
|
|
selector: string,
|
|
|
|
PuppeteerUtil: PuppeteerUtil
|
|
|
|
) => Awaitable<Node | null>;
|
2022-09-15 11:12:13 +00:00
|
|
|
|
2020-09-23 14:02:22 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2023-02-14 21:31:30 +00:00
|
|
|
export class QueryHandler {
|
|
|
|
// Either one of these may be implemented, but at least one must be.
|
|
|
|
static querySelectorAll?: QuerySelectorAll;
|
|
|
|
static querySelector?: QuerySelector;
|
|
|
|
|
|
|
|
static get _querySelector(): QuerySelector {
|
|
|
|
if (this.querySelector) {
|
|
|
|
return this.querySelector;
|
|
|
|
}
|
|
|
|
if (!this.querySelectorAll) {
|
|
|
|
throw new Error('Cannot create default query selector');
|
|
|
|
}
|
|
|
|
|
|
|
|
const querySelector: QuerySelector = async (
|
|
|
|
node,
|
|
|
|
selector,
|
|
|
|
PuppeteerUtil
|
|
|
|
) => {
|
|
|
|
const querySelectorAll =
|
|
|
|
'FUNCTION_DEFINITION' as unknown as QuerySelectorAll;
|
|
|
|
const results = querySelectorAll(node, selector, PuppeteerUtil);
|
|
|
|
for await (const result of results) {
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
};
|
|
|
|
|
|
|
|
return (this.querySelector = createFunction(
|
|
|
|
querySelector
|
|
|
|
.toString()
|
|
|
|
.replace("'FUNCTION_DEFINITION'", this.querySelectorAll.toString())
|
|
|
|
) as typeof querySelector);
|
|
|
|
}
|
|
|
|
|
|
|
|
static get _querySelectorAll(): QuerySelectorAll {
|
|
|
|
if (this.querySelectorAll) {
|
|
|
|
return this.querySelectorAll;
|
|
|
|
}
|
|
|
|
if (!this.querySelector) {
|
|
|
|
throw new Error('Cannot create default query selector');
|
|
|
|
}
|
|
|
|
|
|
|
|
const querySelectorAll: QuerySelectorAll = async function* (
|
|
|
|
node,
|
|
|
|
selector,
|
|
|
|
PuppeteerUtil
|
|
|
|
) {
|
|
|
|
const querySelector = 'FUNCTION_DEFINITION' as unknown as QuerySelector;
|
|
|
|
const result = await querySelector(node, selector, PuppeteerUtil);
|
|
|
|
if (result) {
|
|
|
|
yield result;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return (this.querySelectorAll = createFunction(
|
|
|
|
querySelectorAll
|
|
|
|
.toString()
|
|
|
|
.replace("'FUNCTION_DEFINITION'", this.querySelector.toString())
|
|
|
|
) as typeof querySelectorAll);
|
|
|
|
}
|
|
|
|
|
2022-08-04 13:45:21 +00:00
|
|
|
/**
|
2023-02-14 21:31:30 +00:00
|
|
|
* Queries for multiple nodes given a selector and {@link ElementHandle}.
|
2022-08-04 13:45:21 +00:00
|
|
|
*
|
2023-02-14 21:31:30 +00:00
|
|
|
* Akin to {@link Window.prototype.querySelectorAll}.
|
2022-08-04 13:45:21 +00:00
|
|
|
*/
|
2023-02-14 21:31:30 +00:00
|
|
|
static async *queryAll(
|
2022-07-06 07:05:37 +00:00
|
|
|
element: ElementHandle<Node>,
|
|
|
|
selector: string
|
2023-02-14 21:31:30 +00:00
|
|
|
): AwaitableIterable<ElementHandle<Node>> {
|
|
|
|
const world = element.executionContext()._world;
|
|
|
|
assert(world);
|
|
|
|
const handle = await element.evaluateHandle(
|
|
|
|
this._querySelectorAll,
|
|
|
|
selector,
|
|
|
|
LazyArg.create(context => {
|
|
|
|
return context.puppeteerUtil;
|
|
|
|
})
|
|
|
|
);
|
|
|
|
yield* transposeIterableHandle(handle);
|
|
|
|
}
|
|
|
|
|
2022-08-04 13:45:21 +00:00
|
|
|
/**
|
2023-02-14 21:31:30 +00:00
|
|
|
* Queries for a single node given a selector and {@link ElementHandle}.
|
2022-08-04 13:45:21 +00:00
|
|
|
*
|
2023-02-14 21:31:30 +00:00
|
|
|
* Akin to {@link Window.prototype.querySelector}.
|
2022-08-04 13:45:21 +00:00
|
|
|
*/
|
2023-02-14 21:31:30 +00:00
|
|
|
static async queryOne(
|
2022-07-06 07:05:37 +00:00
|
|
|
element: ElementHandle<Node>,
|
2020-09-23 14:02:22 +00:00
|
|
|
selector: string
|
2023-02-14 21:31:30 +00:00
|
|
|
): Promise<ElementHandle<Node> | null> {
|
|
|
|
const world = element.executionContext()._world;
|
|
|
|
assert(world);
|
|
|
|
const result = await element.evaluateHandle(
|
|
|
|
this._querySelector,
|
|
|
|
selector,
|
|
|
|
LazyArg.create(context => {
|
|
|
|
return context.puppeteerUtil;
|
|
|
|
})
|
|
|
|
);
|
|
|
|
if (!(result instanceof ElementHandle)) {
|
|
|
|
await result.dispose();
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2022-08-10 21:34:29 +00:00
|
|
|
|
2022-08-04 13:45:21 +00:00
|
|
|
/**
|
|
|
|
* Waits until a single node appears for a given selector and
|
|
|
|
* {@link ElementHandle}.
|
|
|
|
*/
|
2023-02-14 21:31:30 +00:00
|
|
|
static async waitFor(
|
2022-08-26 10:55:30 +00:00
|
|
|
elementOrFrame: ElementHandle<Node> | Frame,
|
2020-09-23 14:02:22 +00:00
|
|
|
selector: string,
|
2023-02-14 21:31:30 +00:00
|
|
|
options: WaitForSelectorOptions,
|
|
|
|
bindings = new Map<string, (...args: never[]) => unknown>()
|
|
|
|
): Promise<ElementHandle<Node> | null> {
|
|
|
|
let frame: Frame;
|
|
|
|
let element: ElementHandle<Node> | undefined;
|
|
|
|
if (!(elementOrFrame instanceof ElementHandle)) {
|
|
|
|
frame = elementOrFrame;
|
|
|
|
} else {
|
|
|
|
frame = elementOrFrame.frame;
|
|
|
|
element = await frame.worlds[PUPPETEER_WORLD].adoptHandle(elementOrFrame);
|
2022-07-06 07:05:37 +00:00
|
|
|
}
|
2023-02-14 21:31:30 +00:00
|
|
|
const result = await frame.worlds[PUPPETEER_WORLD]._waitForSelectorInPage(
|
|
|
|
this._querySelector,
|
|
|
|
element,
|
|
|
|
selector,
|
|
|
|
options,
|
|
|
|
bindings
|
|
|
|
);
|
|
|
|
if (element) {
|
|
|
|
await element.dispose();
|
2022-07-06 07:05:37 +00:00
|
|
|
}
|
2023-02-14 21:31:30 +00:00
|
|
|
if (!(result instanceof ElementHandle)) {
|
|
|
|
await result?.dispose();
|
|
|
|
return null;
|
2022-08-04 13:45:21 +00:00
|
|
|
}
|
2023-02-14 21:31:30 +00:00
|
|
|
return frame.worlds[MAIN_WORLD].transferHandle(result);
|
2022-06-14 11:55:35 +00:00
|
|
|
}
|
2020-04-30 11:45:52 +00:00
|
|
|
}
|