2023-02-14 21:31:30 +00:00
/ * *
* Copyright 2023 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 .
* /
2023-02-15 15:33:18 +00:00
import type PuppeteerUtil from '../injected/injected.js' ;
import { assert } from '../util/assert.js' ;
import { interpolateFunction , stringifyFunction } from '../util/Function.js' ;
2023-02-15 23:09:31 +00:00
2023-09-15 11:00:20 +00:00
import {
QueryHandler ,
type QuerySelector ,
type QuerySelectorAll ,
} from './QueryHandler.js' ;
2023-02-15 15:33:18 +00:00
import { scriptInjector } from './ScriptInjector.js' ;
2023-02-14 21:31:30 +00:00
/ * *
* @public
* /
export interface CustomQueryHandler {
/ * *
2023-03-30 11:54:00 +00:00
* Searches for a { @link https : //developer.mozilla.org/en-US/docs/Web/API/Node | Node} matching the given `selector` from {@link https://developer.mozilla.org/en-US/docs/Web/API/Node | node}.
2023-02-14 21:31:30 +00:00
* /
queryOne ? : ( node : Node , selector : string ) = > Node | null ;
/ * *
2023-03-30 11:54:00 +00:00
* Searches for some { @link https : //developer.mozilla.org/en-US/docs/Web/API/Node | Nodes} matching the given `selector` from {@link https://developer.mozilla.org/en-US/docs/Web/API/Node | node}.
2023-02-14 21:31:30 +00:00
* /
queryAll ? : ( node : Node , selector : string ) = > Iterable < Node > ;
}
2023-02-15 15:33:18 +00:00
/ * *
* The registry of { @link CustomQueryHandler | custom query handlers } .
*
* @example
*
* ` ` ` ts
* Puppeteer . customQueryHandlers . register ( 'lit' , { … } ) ;
* const aHandle = await page . $ ( 'lit/…' ) ;
* ` ` `
*
* @internal
* /
export class CustomQueryHandlerRegistry {
# handlers = new Map <
string ,
[ registerScript : string , Handler : typeof QueryHandler ]
> ( ) ;
get ( name : string ) : typeof QueryHandler | undefined {
const handler = this . # handlers . get ( name ) ;
return handler ? handler [ 1 ] : undefined ;
}
/ * *
* Registers a { @link CustomQueryHandler | custom query handler } .
*
* @remarks
* After registration , the handler can be used everywhere where a selector is
* expected by prepending the selection string with ` <name>/ ` . The name is
* only allowed to consist of lower - and upper case latin letters .
*
* @example
*
* ` ` ` ts
* Puppeteer . customQueryHandlers . register ( 'lit' , { … } ) ;
* const aHandle = await page . $ ( 'lit/…' ) ;
* ` ` `
*
* @param name - Name to register under .
* @param queryHandler - { @link CustomQueryHandler | Custom query handler } to
* register .
* /
register ( name : string , handler : CustomQueryHandler ) : void {
assert (
! this . # handlers . has ( name ) ,
` Cannot register over existing handler: ${ name } `
) ;
assert (
/^[a-zA-Z]+$/ . test ( name ) ,
` Custom query handler names may only contain [a-zA-Z] `
) ;
assert (
handler . queryAll || handler . queryOne ,
` At least one query method must be implemented. `
) ;
const Handler = class extends QueryHandler {
static override querySelectorAll : QuerySelectorAll = interpolateFunction (
( node , selector , PuppeteerUtil ) = > {
return PuppeteerUtil . customQuerySelectors
. get ( PLACEHOLDER ( 'name' ) ) !
. querySelectorAll ( node , selector ) ;
} ,
{ name : JSON.stringify ( name ) }
) ;
static override querySelector : QuerySelector = interpolateFunction (
( node , selector , PuppeteerUtil ) = > {
return PuppeteerUtil . customQuerySelectors
. get ( PLACEHOLDER ( 'name' ) ) !
. querySelector ( node , selector ) ;
} ,
{ name : JSON.stringify ( name ) }
) ;
} ;
const registerScript = interpolateFunction (
( PuppeteerUtil : PuppeteerUtil ) = > {
PuppeteerUtil . customQuerySelectors . register ( PLACEHOLDER ( 'name' ) , {
queryAll : PLACEHOLDER ( 'queryAll' ) ,
queryOne : PLACEHOLDER ( 'queryOne' ) ,
} ) ;
} ,
{
name : JSON.stringify ( name ) ,
queryAll : handler.queryAll
? stringifyFunction ( handler . queryAll )
: String ( undefined ) ,
queryOne : handler.queryOne
? stringifyFunction ( handler . queryOne )
: String ( undefined ) ,
}
) . toString ( ) ;
this . # handlers . set ( name , [ registerScript , Handler ] ) ;
scriptInjector . append ( registerScript ) ;
}
/ * *
* Unregisters the { @link CustomQueryHandler | custom query handler } for the
* given name .
*
* @throws ` Error ` if there is no handler under the given name .
* /
unregister ( name : string ) : void {
const handler = this . # handlers . get ( name ) ;
if ( ! handler ) {
throw new Error ( ` Cannot unregister unknown handler: ${ name } ` ) ;
}
scriptInjector . pop ( handler [ 0 ] ) ;
this . # handlers . delete ( name ) ;
}
/ * *
* Gets the names of all { @link CustomQueryHandler | custom query handlers } .
* /
names ( ) : string [ ] {
return [ . . . this . # handlers . keys ( ) ] ;
}
/ * *
* Unregisters all custom query handlers .
* /
clear ( ) : void {
for ( const [ registerScript ] of this . # handlers ) {
scriptInjector . pop ( registerScript ) ;
}
this . # handlers . clear ( ) ;
}
}
/ * *
* @internal
* /
export const customQueryHandlers = new CustomQueryHandlerRegistry ( ) ;
2023-02-14 21:31:30 +00:00
/ * *
* @deprecated Import { @link Puppeteer } and use the static method
* { @link Puppeteer . registerCustomQueryHandler }
*
* @public
* /
export function registerCustomQueryHandler (
name : string ,
handler : CustomQueryHandler
) : void {
2023-02-15 15:33:18 +00:00
customQueryHandlers . register ( name , handler ) ;
2023-02-14 21:31:30 +00:00
}
/ * *
* @deprecated Import { @link Puppeteer } and use the static method
* { @link Puppeteer . unregisterCustomQueryHandler }
*
* @public
* /
export function unregisterCustomQueryHandler ( name : string ) : void {
2023-02-15 15:33:18 +00:00
customQueryHandlers . unregister ( name ) ;
2023-02-14 21:31:30 +00:00
}
/ * *
* @deprecated Import { @link Puppeteer } and use the static method
* { @link Puppeteer . customQueryHandlerNames }
*
* @public
* /
export function customQueryHandlerNames ( ) : string [ ] {
2023-02-15 15:33:18 +00:00
return customQueryHandlers . names ( ) ;
2023-02-14 21:31:30 +00:00
}
/ * *
* @deprecated Import { @link Puppeteer } and use the static method
* { @link Puppeteer . clearCustomQueryHandlers }
*
* @public
* /
export function clearCustomQueryHandlers ( ) : void {
customQueryHandlers . clear ( ) ;
}