css-select
Version:
a CSS selector compiler/engine
233 lines (214 loc) • 7.77 kB
text/typescript
import * as boolbase from "boolbase";
import { parse, type Selector } from "css-what";
import type {
Element as DomHandlerElement,
AnyNode as DomHandlerNode,
} from "domhandler";
import * as DomUtils from "domutils";
import { compileToken } from "./compile.js";
import { findAll, findOne, getNextSiblings } from "./helpers/querying.js";
import type {
Adapter,
CompiledQuery,
InternalOptions,
Options,
Predicate,
Query,
} from "./types.js";
export type { Options };
const defaultEquals = <Node>(a: Node, b: Node) => a === b;
const defaultOptions: InternalOptions<DomHandlerNode, DomHandlerElement> = {
adapter: DomUtils,
equals: defaultEquals,
};
function convertOptionFormats<Node, ElementNode extends Node>(
options?: Options<Node, ElementNode>,
): InternalOptions<Node, ElementNode> {
/*
* We force one format of options to the other one.
*/
// @ts-expect-error Default options may have incompatible `Node` / `ElementNode`.
const opts: Options<Node, ElementNode> = options ?? defaultOptions;
// @ts-expect-error Same as above.
opts.adapter ??= DomUtils;
// @ts-expect-error `equals` does not exist on `Options`
opts.equals ??= opts.adapter?.equals ?? defaultEquals;
return opts as InternalOptions<Node, ElementNode>;
}
/**
* Compiles a selector to an executable function.
*
* The returned function checks if each passed node is an element. Use
* `_compileUnsafe` to skip this check.
*
* @param selector Selector to compile.
* @param options Compilation options.
* @param context Optional context for the selector.
*/
export function compile<Node, ElementNode extends Node>(
selector: string | Selector[][],
options?: Options<Node, ElementNode>,
context?: Node[] | Node,
): CompiledQuery<Node> {
const opts = convertOptionFormats(options);
const next = _compileUnsafe(selector, opts, context);
return next === boolbase.falseFunc
? boolbase.falseFunc
: (elem: Node) => opts.adapter.isTag(elem) && next(elem);
}
/**
* Like `compile`, but does not add a check if elements are tags.
*/
export function _compileUnsafe<Node, ElementNode extends Node>(
selector: string | Selector[][],
options?: Options<Node, ElementNode>,
context?: Node[] | Node,
): CompiledQuery<ElementNode> {
return _compileToken<Node, ElementNode>(
typeof selector === "string" ? parse(selector) : selector,
options,
context,
);
}
/**
* @deprecated Use `_compileUnsafe` instead.
*/
export function _compileToken<Node, ElementNode extends Node>(
selector: Selector[][],
options?: Options<Node, ElementNode>,
context?: Node[] | Node,
): CompiledQuery<ElementNode> {
return compileToken<Node, ElementNode>(
selector,
convertOptionFormats(options),
context,
);
}
function getSelectorFunc<Node, ElementNode extends Node, T>(
searchFunc: (
query: Predicate<ElementNode>,
elems: Array<Node>,
options: InternalOptions<Node, ElementNode>,
) => T,
) {
return function select(
query: Query<ElementNode>,
elements: Node[] | Node,
options?: Options<Node, ElementNode>,
): T {
const opts = convertOptionFormats(options);
if (typeof query !== "function") {
query = _compileUnsafe<Node, ElementNode>(query, opts, elements);
}
const filteredElements = prepareContext(
elements,
opts.adapter,
query.shouldTestNextSiblings,
);
return searchFunc(query, filteredElements, opts);
};
}
export function prepareContext<Node, ElementNode extends Node>(
elems: Node | Node[],
adapter: Adapter<Node, ElementNode>,
shouldTestNextSiblings = false,
): Node[] {
/*
* Add siblings if the query requires them.
* See https://github.com/fb55/css-select/pull/43#issuecomment-225414692
*/
if (shouldTestNextSiblings) {
elems = appendNextSiblings(elems, adapter);
}
return Array.isArray(elems)
? adapter.removeSubsets(elems)
: adapter.getChildren(elems);
}
function appendNextSiblings<Node, ElementNode extends Node>(
elem: Node | Node[],
adapter: Adapter<Node, ElementNode>,
): Node[] {
// Order matters because jQuery seems to check the children before the siblings
const elems = Array.isArray(elem) ? elem.slice(0) : [elem];
const elemsLength = elems.length;
for (let i = 0; i < elemsLength; i++) {
const nextSiblings = getNextSiblings(elems[i], adapter);
elems.push(...nextSiblings);
}
return elems;
}
/**
* @template Node The generic Node type for the DOM adapter being used.
* @template ElementNode The Node type for elements for the DOM adapter being used.
* @param elems Elements to query. If it is an element, its children will be queried.
* @param query can be either a CSS selector string or a compiled query function.
* @param [options] options for querying the document.
* @see compile for supported selector queries.
* @returns All matching elements.
*
*/
export const selectAll: <Node, ElementNode extends Node>(
query: Query<ElementNode>,
elements: Node | Node[],
options?: Options<Node, ElementNode> | undefined,
) => ElementNode[] = getSelectorFunc(
<Node, ElementNode extends Node>(
query: Predicate<ElementNode>,
elems: Node[] | null,
options: InternalOptions<Node, ElementNode>,
): ElementNode[] =>
query === boolbase.falseFunc || !elems || elems.length === 0
? []
: findAll(query, elems, options),
);
/**
* @template Node The generic Node type for the DOM adapter being used.
* @template ElementNode The Node type for elements for the DOM adapter being used.
* @param elems Elements to query. If it is an element, its children will be queried.
* @param query can be either a CSS selector string or a compiled query function.
* @param [options] options for querying the document.
* @see compile for supported selector queries.
* @returns the first match, or null if there was no match.
*/
export const selectOne: <Node, ElementNode extends Node>(
query: Query<ElementNode>,
elements: Node | Node[],
options?: Options<Node, ElementNode> | undefined,
) => ElementNode | null = getSelectorFunc(
<Node, ElementNode extends Node>(
query: Predicate<ElementNode>,
elems: Node[] | null,
options: InternalOptions<Node, ElementNode>,
): ElementNode | null =>
query === boolbase.falseFunc || !elems || elems.length === 0
? null
: findOne(query, elems, options),
);
/**
* Tests whether or not an element is matched by query.
*
* @template Node The generic Node type for the DOM adapter being used.
* @template ElementNode The Node type for elements for the DOM adapter being used.
* @param elem The element to test if it matches the query.
* @param query can be either a CSS selector string or a compiled query function.
* @param [options] options for querying the document.
* @see compile for supported selector queries.
* @returns
*/
export function is<Node, ElementNode extends Node>(
elem: ElementNode,
query: Query<ElementNode>,
options?: Options<Node, ElementNode>,
): boolean {
return (typeof query === "function" ? query : compile(query, options))(
elem,
);
}
/**
* Alias for selectAll(query, elems, options).
* @see [compile] for supported selector queries.
*/
export default selectAll;
// Export filters, pseudos and aliases to allow users to supply their own.
/** @deprecated Use the `pseudos` option instead. */
export { aliases, filters, pseudos } from "./pseudo-selectors/index.js";