2020-04-30 11:45:52 +00:00
|
|
|
/**
|
|
|
|
* Copyright 2020 Google Inc. All rights reserved.
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2020-09-23 14:02:22 +00:00
|
|
|
import { WaitForSelectorOptions, DOMWorld } from './DOMWorld.js';
|
|
|
|
import { ElementHandle, JSHandle } from './JSHandle.js';
|
2022-06-13 09:16:25 +00:00
|
|
|
import { _ariaHandler } from './AriaQueryHandler.js';
|
2020-09-23 14:02:22 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2020-10-05 06:25:55 +00:00
|
|
|
export interface InternalQueryHandler {
|
2020-09-23 14:02:22 +00:00
|
|
|
queryOne?: (
|
|
|
|
element: ElementHandle,
|
|
|
|
selector: string
|
|
|
|
) => Promise<ElementHandle | null>;
|
|
|
|
waitFor?: (
|
|
|
|
domWorld: DOMWorld,
|
|
|
|
selector: string,
|
|
|
|
options: WaitForSelectorOptions
|
|
|
|
) => Promise<ElementHandle | null>;
|
|
|
|
queryAll?: (
|
|
|
|
element: ElementHandle,
|
|
|
|
selector: string
|
|
|
|
) => Promise<ElementHandle[]>;
|
|
|
|
queryAllArray?: (
|
|
|
|
element: ElementHandle,
|
|
|
|
selector: string
|
2022-05-31 14:34:16 +00:00
|
|
|
) => Promise<JSHandle<Element[]>>;
|
2020-09-23 14:02:22 +00:00
|
|
|
}
|
|
|
|
|
2020-10-07 08:43:46 +00:00
|
|
|
/**
|
|
|
|
* Contains two functions `queryOne` and `queryAll` that can
|
2020-10-23 14:28:38 +00:00
|
|
|
* be {@link Puppeteer.registerCustomQueryHandler | registered}
|
2020-10-07 08:43:46 +00:00
|
|
|
* as alternative querying strategies. The functions `queryOne` and `queryAll`
|
|
|
|
* are executed in the page context. `queryOne` should take an `Element` and a
|
|
|
|
* selector string as argument and return a single `Element` or `null` if no
|
|
|
|
* element is found. `queryAll` takes the same arguments but should instead
|
|
|
|
* return a `NodeListOf<Element>` or `Array<Element>` with all the elements
|
|
|
|
* that match the given query selector.
|
|
|
|
* @public
|
|
|
|
*/
|
2020-09-23 14:02:22 +00:00
|
|
|
export interface CustomQueryHandler {
|
2020-07-17 05:29:42 +00:00
|
|
|
queryOne?: (element: Element | Document, selector: string) => Element | null;
|
|
|
|
queryAll?: (
|
|
|
|
element: Element | Document,
|
|
|
|
selector: string
|
|
|
|
) => Element[] | NodeListOf<Element>;
|
2020-04-30 11:45:52 +00:00
|
|
|
}
|
|
|
|
|
2020-09-23 14:02:22 +00:00
|
|
|
function makeQueryHandler(handler: CustomQueryHandler): InternalQueryHandler {
|
|
|
|
const internalHandler: InternalQueryHandler = {};
|
|
|
|
|
|
|
|
if (handler.queryOne) {
|
2022-05-31 14:34:16 +00:00
|
|
|
const queryOne = handler.queryOne;
|
2020-09-23 14:02:22 +00:00
|
|
|
internalHandler.queryOne = async (element, selector) => {
|
2022-05-31 14:34:16 +00:00
|
|
|
const jsHandle = await element.evaluateHandle(queryOne, selector);
|
2020-09-23 14:02:22 +00:00
|
|
|
const elementHandle = jsHandle.asElement();
|
2022-06-14 11:55:35 +00:00
|
|
|
if (elementHandle) {
|
|
|
|
return elementHandle;
|
|
|
|
}
|
2020-09-23 14:02:22 +00:00
|
|
|
await jsHandle.dispose();
|
|
|
|
return null;
|
|
|
|
};
|
|
|
|
internalHandler.waitFor = (
|
|
|
|
domWorld: DOMWorld,
|
|
|
|
selector: string,
|
|
|
|
options: WaitForSelectorOptions
|
2022-06-15 10:42:21 +00:00
|
|
|
) => {
|
|
|
|
return domWorld._waitForSelectorInPage(queryOne, selector, options);
|
|
|
|
};
|
2020-09-23 14:02:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (handler.queryAll) {
|
2022-05-31 14:34:16 +00:00
|
|
|
const queryAll = handler.queryAll;
|
2020-09-23 14:02:22 +00:00
|
|
|
internalHandler.queryAll = async (element, selector) => {
|
2022-05-31 14:34:16 +00:00
|
|
|
const jsHandle = await element.evaluateHandle(queryAll, selector);
|
2020-09-23 14:02:22 +00:00
|
|
|
const properties = await jsHandle.getProperties();
|
|
|
|
await jsHandle.dispose();
|
|
|
|
const result = [];
|
|
|
|
for (const property of properties.values()) {
|
|
|
|
const elementHandle = property.asElement();
|
2022-06-14 11:55:35 +00:00
|
|
|
if (elementHandle) {
|
|
|
|
result.push(elementHandle);
|
|
|
|
}
|
2020-09-23 14:02:22 +00:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
};
|
|
|
|
internalHandler.queryAllArray = async (element, selector) => {
|
2022-05-31 14:34:16 +00:00
|
|
|
const resultHandle = await element.evaluateHandle(queryAll, selector);
|
2020-09-23 14:02:22 +00:00
|
|
|
const arrayHandle = await resultHandle.evaluateHandle(
|
2022-06-15 10:42:21 +00:00
|
|
|
(res: Element[] | NodeListOf<Element>) => {
|
|
|
|
return Array.from(res);
|
|
|
|
}
|
2020-09-23 14:02:22 +00:00
|
|
|
);
|
|
|
|
return arrayHandle;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return internalHandler;
|
|
|
|
}
|
|
|
|
|
|
|
|
const _defaultHandler = makeQueryHandler({
|
2022-06-15 10:42:21 +00:00
|
|
|
queryOne: (element: Element | Document, selector: string) => {
|
|
|
|
return element.querySelector(selector);
|
|
|
|
},
|
|
|
|
queryAll: (element: Element | Document, selector: string) => {
|
|
|
|
return element.querySelectorAll(selector);
|
|
|
|
},
|
2020-09-23 14:02:22 +00:00
|
|
|
});
|
2020-04-30 11:45:52 +00:00
|
|
|
|
2020-10-13 09:05:47 +00:00
|
|
|
const pierceHandler = makeQueryHandler({
|
|
|
|
queryOne: (element, selector) => {
|
|
|
|
let found: Element | null = null;
|
|
|
|
const search = (root: Element | ShadowRoot) => {
|
|
|
|
const iter = document.createTreeWalker(root, NodeFilter.SHOW_ELEMENT);
|
|
|
|
do {
|
|
|
|
const currentNode = iter.currentNode as HTMLElement;
|
|
|
|
if (currentNode.shadowRoot) {
|
|
|
|
search(currentNode.shadowRoot);
|
|
|
|
}
|
|
|
|
if (currentNode instanceof ShadowRoot) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-05-11 12:17:02 +00:00
|
|
|
if (currentNode !== root && !found && currentNode.matches(selector)) {
|
2020-10-13 09:05:47 +00:00
|
|
|
found = currentNode;
|
|
|
|
}
|
|
|
|
} while (!found && iter.nextNode());
|
|
|
|
};
|
|
|
|
if (element instanceof Document) {
|
|
|
|
element = element.documentElement;
|
|
|
|
}
|
|
|
|
search(element);
|
|
|
|
return found;
|
|
|
|
},
|
|
|
|
|
|
|
|
queryAll: (element, selector) => {
|
|
|
|
const result: Element[] = [];
|
|
|
|
const collect = (root: Element | ShadowRoot) => {
|
|
|
|
const iter = document.createTreeWalker(root, NodeFilter.SHOW_ELEMENT);
|
|
|
|
do {
|
|
|
|
const currentNode = iter.currentNode as HTMLElement;
|
|
|
|
if (currentNode.shadowRoot) {
|
|
|
|
collect(currentNode.shadowRoot);
|
|
|
|
}
|
|
|
|
if (currentNode instanceof ShadowRoot) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-05-11 12:17:02 +00:00
|
|
|
if (currentNode !== root && currentNode.matches(selector)) {
|
2020-10-13 09:05:47 +00:00
|
|
|
result.push(currentNode);
|
|
|
|
}
|
|
|
|
} while (iter.nextNode());
|
|
|
|
};
|
|
|
|
if (element instanceof Document) {
|
|
|
|
element = element.documentElement;
|
|
|
|
}
|
|
|
|
collect(element);
|
|
|
|
return result;
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2022-06-13 09:16:25 +00:00
|
|
|
const builtInHandlers = new Map([
|
|
|
|
['aria', _ariaHandler],
|
2020-10-13 09:05:47 +00:00
|
|
|
['pierce', pierceHandler],
|
|
|
|
]);
|
2022-06-13 09:16:25 +00:00
|
|
|
const queryHandlers = new Map(builtInHandlers);
|
2020-10-05 06:25:55 +00:00
|
|
|
|
2020-10-07 08:43:46 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-06-13 09:16:25 +00:00
|
|
|
export function _registerCustomQueryHandler(
|
2020-05-07 10:54:55 +00:00
|
|
|
name: string,
|
2020-09-23 14:02:22 +00:00
|
|
|
handler: CustomQueryHandler
|
2020-05-07 10:54:55 +00:00
|
|
|
): void {
|
2022-06-14 11:55:35 +00:00
|
|
|
if (queryHandlers.get(name)) {
|
2020-04-30 11:45:52 +00:00
|
|
|
throw new Error(`A custom query handler named "${name}" already exists`);
|
2022-06-14 11:55:35 +00:00
|
|
|
}
|
2020-04-30 11:45:52 +00:00
|
|
|
|
|
|
|
const isValidName = /^[a-zA-Z]+$/.test(name);
|
2022-06-14 11:55:35 +00:00
|
|
|
if (!isValidName) {
|
2020-04-30 11:45:52 +00:00
|
|
|
throw new Error(`Custom query handler names may only contain [a-zA-Z]`);
|
2022-06-14 11:55:35 +00:00
|
|
|
}
|
2020-04-30 11:45:52 +00:00
|
|
|
|
2020-09-23 14:02:22 +00:00
|
|
|
const internalHandler = makeQueryHandler(handler);
|
|
|
|
|
2022-06-13 09:16:25 +00:00
|
|
|
queryHandlers.set(name, internalHandler);
|
2020-04-30 11:45:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-10-07 08:43:46 +00:00
|
|
|
* @internal
|
2020-04-30 11:45:52 +00:00
|
|
|
*/
|
2022-06-13 09:16:25 +00:00
|
|
|
export function _unregisterCustomQueryHandler(name: string): void {
|
|
|
|
if (queryHandlers.has(name) && !builtInHandlers.has(name)) {
|
|
|
|
queryHandlers.delete(name);
|
2020-09-23 14:02:22 +00:00
|
|
|
}
|
2020-04-30 11:45:52 +00:00
|
|
|
}
|
|
|
|
|
2020-10-07 08:43:46 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-06-13 09:16:25 +00:00
|
|
|
export function _customQueryHandlerNames(): string[] {
|
2022-06-15 10:42:21 +00:00
|
|
|
return [...queryHandlers.keys()].filter((name) => {
|
|
|
|
return !builtInHandlers.has(name);
|
|
|
|
});
|
2020-04-30 11:45:52 +00:00
|
|
|
}
|
|
|
|
|
2020-10-07 08:43:46 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-06-13 09:16:25 +00:00
|
|
|
export function _clearCustomQueryHandlers(): void {
|
|
|
|
_customQueryHandlerNames().forEach(_unregisterCustomQueryHandler);
|
2020-04-30 11:45:52 +00:00
|
|
|
}
|
|
|
|
|
2020-10-07 08:43:46 +00:00
|
|
|
/**
|
|
|
|
* @internal
|
|
|
|
*/
|
2022-06-13 09:16:25 +00:00
|
|
|
export function _getQueryHandlerAndSelector(selector: string): {
|
2021-05-12 14:48:30 +00:00
|
|
|
updatedSelector: string;
|
|
|
|
queryHandler: InternalQueryHandler;
|
|
|
|
} {
|
2020-04-30 11:45:52 +00:00
|
|
|
const hasCustomQueryHandler = /^[a-zA-Z]+\//.test(selector);
|
2022-06-14 11:55:35 +00:00
|
|
|
if (!hasCustomQueryHandler) {
|
2020-09-23 14:02:22 +00:00
|
|
|
return { updatedSelector: selector, queryHandler: _defaultHandler };
|
2022-06-14 11:55:35 +00:00
|
|
|
}
|
2020-04-30 11:45:52 +00:00
|
|
|
|
|
|
|
const index = selector.indexOf('/');
|
|
|
|
const name = selector.slice(0, index);
|
|
|
|
const updatedSelector = selector.slice(index + 1);
|
2022-06-13 09:16:25 +00:00
|
|
|
const queryHandler = queryHandlers.get(name);
|
2022-06-14 11:55:35 +00:00
|
|
|
if (!queryHandler) {
|
2020-05-07 10:54:55 +00:00
|
|
|
throw new Error(
|
|
|
|
`Query set to use "${name}", but no query handler of that name was found`
|
|
|
|
);
|
2022-06-14 11:55:35 +00:00
|
|
|
}
|
2020-04-30 11:45:52 +00:00
|
|
|
|
|
|
|
return {
|
|
|
|
updatedSelector,
|
2020-05-07 10:54:55 +00:00
|
|
|
queryHandler,
|
2020-04-30 11:45:52 +00:00
|
|
|
};
|
|
|
|
}
|