You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			115 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			JavaScript
		
	
			
		
		
	
	
			115 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			JavaScript
		
	
| 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
 |