| import { parse, SelectorType } from "css-what"; |
| import boolbase from "boolbase"; |
| import sortRules, { isTraversal } from "./sort.js"; |
| import { compileGeneralSelector } from "./general.js"; |
| import { ensureIsTag, PLACEHOLDER_ELEMENT, } from "./pseudo-selectors/subselects.js"; |
| /** |
| * Compiles a selector to an executable function. |
| * |
| * @param selector Selector to compile. |
| * @param options Compilation options. |
| * @param context Optional context for the selector. |
| */ |
| export function compile(selector, options, context) { |
| const next = compileUnsafe(selector, options, context); |
| return ensureIsTag(next, options.adapter); |
| } |
| export function compileUnsafe(selector, options, context) { |
| const token = typeof selector === "string" ? parse(selector) : selector; |
| return compileToken(token, options, context); |
| } |
| function includesScopePseudo(t) { |
| return (t.type === SelectorType.Pseudo && |
| (t.name === "scope" || |
| (Array.isArray(t.data) && |
| t.data.some((data) => data.some(includesScopePseudo))))); |
| } |
| const DESCENDANT_TOKEN = { type: SelectorType.Descendant }; |
| const FLEXIBLE_DESCENDANT_TOKEN = { |
| type: "_flexibleDescendant", |
| }; |
| const SCOPE_TOKEN = { |
| type: SelectorType.Pseudo, |
| name: "scope", |
| data: null, |
| }; |
| /* |
| * CSS 4 Spec (Draft): 3.4.1. Absolutizing a Relative Selector |
| * http://www.w3.org/TR/selectors4/#absolutizing |
| */ |
| function absolutize(token, { adapter }, context) { |
| // TODO Use better check if the context is a document |
| const hasContext = !!(context === null || context === void 0 ? void 0 : context.every((e) => { |
| const parent = adapter.isTag(e) && adapter.getParent(e); |
| return e === PLACEHOLDER_ELEMENT || (parent && adapter.isTag(parent)); |
| })); |
| for (const t of token) { |
| if (t.length > 0 && |
| isTraversal(t[0]) && |
| t[0].type !== SelectorType.Descendant) { |
| // Don't continue in else branch |
| } |
| else if (hasContext && !t.some(includesScopePseudo)) { |
| t.unshift(DESCENDANT_TOKEN); |
| } |
| else { |
| continue; |
| } |
| t.unshift(SCOPE_TOKEN); |
| } |
| } |
| export function compileToken(token, options, context) { |
| var _a; |
| token.forEach(sortRules); |
| context = (_a = options.context) !== null && _a !== void 0 ? _a : context; |
| const isArrayContext = Array.isArray(context); |
| const finalContext = context && (Array.isArray(context) ? context : [context]); |
| // Check if the selector is relative |
| if (options.relativeSelector !== false) { |
| absolutize(token, options, finalContext); |
| } |
| else if (token.some((t) => t.length > 0 && isTraversal(t[0]))) { |
| throw new Error("Relative selectors are not allowed when the `relativeSelector` option is disabled"); |
| } |
| let shouldTestNextSiblings = false; |
| const query = token |
| .map((rules) => { |
| if (rules.length >= 2) { |
| const [first, second] = rules; |
| if (first.type !== SelectorType.Pseudo || |
| first.name !== "scope") { |
| // Ignore |
| } |
| else if (isArrayContext && |
| second.type === SelectorType.Descendant) { |
| rules[1] = FLEXIBLE_DESCENDANT_TOKEN; |
| } |
| else if (second.type === SelectorType.Adjacent || |
| second.type === SelectorType.Sibling) { |
| shouldTestNextSiblings = true; |
| } |
| } |
| return compileRules(rules, options, finalContext); |
| }) |
| .reduce(reduceRules, boolbase.falseFunc); |
| query.shouldTestNextSiblings = shouldTestNextSiblings; |
| return query; |
| } |
| function compileRules(rules, options, context) { |
| var _a; |
| return rules.reduce((previous, rule) => previous === boolbase.falseFunc |
| ? boolbase.falseFunc |
| : compileGeneralSelector(previous, rule, options, context, compileToken), (_a = options.rootFunc) !== null && _a !== void 0 ? _a : boolbase.trueFunc); |
| } |
| function reduceRules(a, b) { |
| if (b === boolbase.falseFunc || a === boolbase.trueFunc) { |
| return a; |
| } |
| if (a === boolbase.falseFunc || b === boolbase.trueFunc) { |
| return b; |
| } |
| return function combine(elem) { |
| return a(elem) || b(elem); |
| }; |
| } |
| //# sourceMappingURL=compile.js.map |