Spaces:
Configuration error
Configuration error
import boolbase from "boolbase"; | |
import { isTraversal } from "../sort.js"; | |
/** Used as a placeholder for :has. Will be replaced with the actual element. */ | |
export const PLACEHOLDER_ELEMENT = {}; | |
export function ensureIsTag(next, adapter) { | |
if (next === boolbase.falseFunc) | |
return boolbase.falseFunc; | |
return (elem) => adapter.isTag(elem) && next(elem); | |
} | |
export function getNextSiblings(elem, adapter) { | |
const siblings = adapter.getSiblings(elem); | |
if (siblings.length <= 1) | |
return []; | |
const elemIndex = siblings.indexOf(elem); | |
if (elemIndex < 0 || elemIndex === siblings.length - 1) | |
return []; | |
return siblings.slice(elemIndex + 1).filter(adapter.isTag); | |
} | |
function copyOptions(options) { | |
// Not copied: context, rootFunc | |
return { | |
xmlMode: !!options.xmlMode, | |
lowerCaseAttributeNames: !!options.lowerCaseAttributeNames, | |
lowerCaseTags: !!options.lowerCaseTags, | |
quirksMode: !!options.quirksMode, | |
cacheResults: !!options.cacheResults, | |
pseudos: options.pseudos, | |
adapter: options.adapter, | |
equals: options.equals, | |
}; | |
} | |
const is = (next, token, options, context, compileToken) => { | |
const func = compileToken(token, copyOptions(options), context); | |
return func === boolbase.trueFunc | |
? next | |
: func === boolbase.falseFunc | |
? boolbase.falseFunc | |
: (elem) => func(elem) && next(elem); | |
}; | |
/* | |
* :not, :has, :is, :matches and :where have to compile selectors | |
* doing this in src/pseudos.ts would lead to circular dependencies, | |
* so we add them here | |
*/ | |
export const subselects = { | |
is, | |
/** | |
* `:matches` and `:where` are aliases for `:is`. | |
*/ | |
matches: is, | |
where: is, | |
not(next, token, options, context, compileToken) { | |
const func = compileToken(token, copyOptions(options), context); | |
return func === boolbase.falseFunc | |
? next | |
: func === boolbase.trueFunc | |
? boolbase.falseFunc | |
: (elem) => !func(elem) && next(elem); | |
}, | |
has(next, subselect, options, _context, compileToken) { | |
const { adapter } = options; | |
const opts = copyOptions(options); | |
opts.relativeSelector = true; | |
const context = subselect.some((s) => s.some(isTraversal)) | |
? // Used as a placeholder. Will be replaced with the actual element. | |
[PLACEHOLDER_ELEMENT] | |
: undefined; | |
const compiled = compileToken(subselect, opts, context); | |
if (compiled === boolbase.falseFunc) | |
return boolbase.falseFunc; | |
const hasElement = ensureIsTag(compiled, adapter); | |
// If `compiled` is `trueFunc`, we can skip this. | |
if (context && compiled !== boolbase.trueFunc) { | |
/* | |
* `shouldTestNextSiblings` will only be true if the query starts with | |
* a traversal (sibling or adjacent). That means we will always have a context. | |
*/ | |
const { shouldTestNextSiblings = false } = compiled; | |
return (elem) => { | |
if (!next(elem)) | |
return false; | |
context[0] = elem; | |
const childs = adapter.getChildren(elem); | |
const nextElements = shouldTestNextSiblings | |
? [...childs, ...getNextSiblings(elem, adapter)] | |
: childs; | |
return adapter.existsOne(hasElement, nextElements); | |
}; | |
} | |
return (elem) => next(elem) && | |
adapter.existsOne(hasElement, adapter.getChildren(elem)); | |
}, | |
}; | |
//# sourceMappingURL=subselects.js.map |