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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
export interface QueryHandler {
|
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
|
|
|
}
|
|
|
|
|
|
|
|
const _customQueryHandlers = new Map<string, QueryHandler>();
|
|
|
|
|
2020-05-07 10:54:55 +00:00
|
|
|
export function registerCustomQueryHandler(
|
|
|
|
name: string,
|
2020-07-17 05:29:42 +00:00
|
|
|
handler: QueryHandler
|
2020-05-07 10:54:55 +00:00
|
|
|
): void {
|
2020-04-30 11:45:52 +00:00
|
|
|
if (_customQueryHandlers.get(name))
|
|
|
|
throw new Error(`A custom query handler named "${name}" already exists`);
|
|
|
|
|
|
|
|
const isValidName = /^[a-zA-Z]+$/.test(name);
|
|
|
|
if (!isValidName)
|
|
|
|
throw new Error(`Custom query handler names may only contain [a-zA-Z]`);
|
|
|
|
|
2020-07-17 05:29:42 +00:00
|
|
|
_customQueryHandlers.set(name, handler);
|
2020-04-30 11:45:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} name
|
|
|
|
*/
|
|
|
|
export function unregisterCustomQueryHandler(name: string): void {
|
|
|
|
_customQueryHandlers.delete(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function customQueryHandlers(): Map<string, QueryHandler> {
|
|
|
|
return _customQueryHandlers;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function clearQueryHandlers(): void {
|
|
|
|
_customQueryHandlers.clear();
|
|
|
|
}
|
|
|
|
|
2020-05-07 10:54:55 +00:00
|
|
|
export function getQueryHandlerAndSelector(
|
2020-07-17 05:29:42 +00:00
|
|
|
selector: string
|
2020-05-07 10:54:55 +00:00
|
|
|
): { updatedSelector: string; queryHandler: QueryHandler } {
|
2020-07-17 05:29:42 +00:00
|
|
|
const defaultHandler = {
|
|
|
|
queryOne: (element: Element, selector: string) =>
|
|
|
|
element.querySelector(selector),
|
|
|
|
queryAll: (element: Element, selector: string) =>
|
|
|
|
element.querySelectorAll(selector),
|
|
|
|
};
|
2020-04-30 11:45:52 +00:00
|
|
|
const hasCustomQueryHandler = /^[a-zA-Z]+\//.test(selector);
|
|
|
|
if (!hasCustomQueryHandler)
|
2020-07-17 05:29:42 +00:00
|
|
|
return { updatedSelector: selector, queryHandler: defaultHandler };
|
2020-04-30 11:45:52 +00:00
|
|
|
|
|
|
|
const index = selector.indexOf('/');
|
|
|
|
const name = selector.slice(0, index);
|
|
|
|
const updatedSelector = selector.slice(index + 1);
|
|
|
|
const queryHandler = customQueryHandlers().get(name);
|
|
|
|
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`
|
|
|
|
);
|
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
|
|
|
};
|
|
|
|
}
|