@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
67 lines (54 loc) • 1.69 kB
JavaScript
import { Ident } from '../../tokenizer/index.js';
const ASTERISK = 0x002A; // U+002A ASTERISK (*)
const VERTICALLINE = 0x007C; // U+007C VERTICAL LINE (|)
function eatIdentifierOrAsterisk() {
if (this.tokenType !== Ident &&
this.isDelim(ASTERISK) === false) {
this.error('Identifier or asterisk is expected');
}
// Check if asterisk is followed immediately by an ident (no whitespace)
// This is invalid in selectors (e.g., "*foo" should error, "* foo" or "*:hover" is ok)
if (this.isDelim(ASTERISK)) {
const currentTokenEnd = this.tokenStart + 1; // asterisk is 1 char
this.next();
// If next token is an ident and starts immediately after asterisk, it's an error
if (this.tokenType === Ident && this.tokenStart === currentTokenEnd) {
this.error('Whitespace is required between universal selector and type selector');
}
return;
}
this.next();
}
export const name = 'TypeSelector';
export const structure = {
name: String
};
// ident
// ident|ident
// ident|*
// *
// *|ident
// *|*
// |ident
// |*
export function parse() {
const start = this.tokenStart;
if (this.isDelim(VERTICALLINE)) {
this.next();
eatIdentifierOrAsterisk.call(this);
} else {
eatIdentifierOrAsterisk.call(this);
if (this.isDelim(VERTICALLINE)) {
this.next();
eatIdentifierOrAsterisk.call(this);
}
}
return {
type: 'TypeSelector',
loc: this.getLocation(start, this.tokenStart),
name: this.substrToCursor(start)
};
}
export function generate(node) {
this.tokenize(node.name);
}