UNPKG

@eslint/css-tree

Version:

A tool set for CSS: fast detailed parser (CSS → AST), walker (AST traversal), generator (AST → CSS) and lexer (validation and matching) based on specs and browser implementations

133 lines (113 loc) 3.8 kB
import { WhiteSpace, Comment, Semicolon, AtKeyword, Delim, Hash, LeftSquareBracket, Colon, Ident, RightCurlyBracket, LeftCurlyBracket, Comma } from '../../tokenizer/index.js'; const AMPERSAND = 0x0026; // U+0026 AMPERSAND (&) const DOT = 0x002E; // U+002E FULL STOP (.) const STAR = 0x002A; // U+002A ASTERISK (*); const PLUSSIGN = 0x002B; // U+002B PLUS SIGN (+) const GREATERTHANSIGN = 0x003E; // U+003E GREATER-THAN SIGN (>) const TILDE = 0x007E; // U+007E TILDE (~) const selectorStarts = new Set([ AMPERSAND, DOT, STAR, PLUSSIGN, GREATERTHANSIGN, TILDE ]); function consumeRaw() { return this.Raw(this.consumeUntilSemicolonIncluded, true); } function isElementSelectorStart() { if (this.tokenType !== Ident) { return false; } const nextTokenType = this.lookupTypeNonSC(1); // If next token is a left curly bracket, it's definitely a selector (e.g., "div { ... }") if (nextTokenType === LeftCurlyBracket) { return true; } // If next token is semicolon, comma, or closing brace, it's definitely a declaration if (nextTokenType === Semicolon || nextTokenType === Comma || nextTokenType === RightCurlyBracket) { return false; } // Special handling for colon case - could be pseudo-class/pseudo-element or property if (nextTokenType === Colon) { // Look ahead further to see what follows the colon const afterColonType = this.lookupTypeNonSC(2); // If after colon there's an identifier (pseudo-class/pseudo-element name), // check what comes after that if (afterColonType === Ident) { const afterPseudoType = this.lookupTypeNonSC(3); // If it's followed by {, it's definitely a selector (e.g., "div:hover { ... }") if (afterPseudoType === LeftCurlyBracket) { return true; } // Otherwise, it's a property (e.g., "transition: opacity 1s") return false; } // If after colon there's not an identifier, it's a property return false; } // For other token types (dimensions, numbers, strings, etc.), assume it's a declaration // This handles cases like multi-value properties return false; } function isSelectorStart() { return this.tokenType === Delim && selectorStarts.has(this.source.charCodeAt(this.tokenStart)) || this.tokenType === Hash || this.tokenType === LeftSquareBracket || this.tokenType === Colon || isElementSelectorStart.call(this); } export const name = 'DeclarationList'; export const structure = { children: [[ 'Declaration', 'Atrule', 'Rule' ]] }; export function parse() { const children = this.createList(); scan: while (!this.eof) { switch (this.tokenType) { case WhiteSpace: case Comment: case Semicolon: this.next(); break; case AtKeyword: children.push(this.parseWithFallback(this.Atrule.bind(this, true), consumeRaw)); break; default: if (isSelectorStart.call(this)) { children.push(this.parseWithFallback(this.Rule, consumeRaw)); } else { children.push(this.parseWithFallback(this.Declaration, consumeRaw)); } } } return { type: 'DeclarationList', loc: this.getLocationFromList(children), children }; } export function generate(node) { this.children(node, prev => { if (prev.type === 'Declaration') { this.token(Semicolon, ';'); } }); }