Skip to content

Commit cd56683

Browse files
committed
feat: Add "modules" property to SyntaxDefinition and include latest CSS modules
1 parent 6dbe132 commit cd56683

File tree

3 files changed

+139
-2
lines changed

3 files changed

+139
-2
lines changed

src/parser.ts

+11-1
Original file line numberDiff line numberDiff line change
@@ -91,7 +91,7 @@ export function createParser(
9191
syntaxDefinition = extendSyntaxDefinition(cssSyntaxDefinitions[syntaxDefinition.baseSyntax], syntaxDefinition);
9292
}
9393

94-
// Apply additional modules if specified
94+
// Apply additional modules if specified from options
9595
if (modules && modules.length > 0) {
9696
for (const module of modules) {
9797
const moduleSyntax = cssModules[module];
@@ -100,6 +100,16 @@ export function createParser(
100100
}
101101
}
102102
}
103+
104+
// Apply modules from syntax definition
105+
if (syntaxDefinition.modules && syntaxDefinition.modules.length > 0) {
106+
for (const module of syntaxDefinition.modules) {
107+
const moduleSyntax = cssModules[module];
108+
if (moduleSyntax) {
109+
syntaxDefinition = extendSyntaxDefinition(syntaxDefinition, moduleSyntax);
110+
}
111+
}
112+
}
103113

104114
const [tagNameEnabled, tagNameWildcardEnabled] = syntaxDefinition.tag
105115
? [true, Boolean(getXmlOptions(syntaxDefinition.tag).wildcard)]

src/syntax-definitions.ts

+15-1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,12 @@ export interface SyntaxDefinition {
1313
* If not specified, syntax will be defined from scratch.
1414
*/
1515
baseSyntax?: CssLevel;
16+
/**
17+
* Additional CSS modules to include in the syntax definition.
18+
* These are specific CSS modules that add new selectors or modify existing ones.
19+
* @example ['css-position-4', 'css-scoping-1']
20+
*/
21+
modules?: CssModule[];
1622
/**
1723
* CSS Tag (type).
1824
* @example div
@@ -243,6 +249,7 @@ function mergeDefinitions<T>(
243249
export const extendSyntaxDefinition: MergeMethod<SyntaxDefinition> = withNoNegative(
244250
mergeSection<SyntaxDefinition>({
245251
baseSyntax: replaceValueIfSpecified,
252+
modules: concatArray,
246253
tag: withPositive(
247254
defaultXmlOptions,
248255
mergeSection({
@@ -398,7 +405,14 @@ const selectors4SyntaxDefinition = extendSyntaxDefinition(selectors3SyntaxDefini
398405
NoArgument: ['marker'],
399406
Selector: ['part']
400407
}
401-
}
408+
},
409+
// Include all the latest modules
410+
modules: [
411+
'css-position-4',
412+
'css-scoping-1',
413+
'css-pseudo-4',
414+
'css-shadow-parts-1'
415+
]
402416
});
403417

404418
const progressiveSyntaxDefinition = extendSyntaxDefinition(selectors4SyntaxDefinition, {

test/modules.test.ts

+113
Original file line numberDiff line numberDiff line change
@@ -600,4 +600,117 @@ describe('CSS Modules', () => {
600600
);
601601
});
602602
});
603+
604+
describe('Syntax definition with modules', () => {
605+
it('should support modules defined in syntax definition', () => {
606+
const parse = createParser({
607+
syntax: {
608+
pseudoClasses: {
609+
unknown: 'reject'
610+
},
611+
pseudoElements: {
612+
unknown: 'reject'
613+
},
614+
modules: ['css-position-4', 'css-shadow-parts-1']
615+
}
616+
});
617+
618+
// Should parse position-4 pseudo-classes
619+
expect(parse(':initial')).toEqual(
620+
ast.selector({
621+
rules: [
622+
ast.rule({
623+
items: [ast.pseudoClass({name: 'initial'})]
624+
})
625+
]
626+
})
627+
);
628+
629+
// Should parse shadow-parts-1 pseudo-elements
630+
expect(parse('::part(button)')).toEqual(
631+
ast.selector({
632+
rules: [
633+
ast.rule({
634+
items: [
635+
ast.pseudoElement({
636+
name: 'part',
637+
argument: ast.selector({
638+
rules: [
639+
ast.rule({
640+
items: [ast.tagName({name: 'button'})]
641+
})
642+
]
643+
})
644+
})
645+
]
646+
})
647+
]
648+
})
649+
);
650+
651+
// Should reject pseudo-classes not in the modules
652+
expect(() => parse(':focus-visible')).toThrow('Unknown pseudo-class: "focus-visible".');
653+
});
654+
655+
it('should support latest syntax with all latest modules', () => {
656+
const parse = createParser({
657+
syntax: 'latest'
658+
});
659+
660+
// Should parse position-4 pseudo-classes
661+
expect(parse(':initial')).toEqual(
662+
ast.selector({
663+
rules: [
664+
ast.rule({
665+
items: [ast.pseudoClass({name: 'initial'})]
666+
})
667+
]
668+
})
669+
);
670+
671+
// Should parse shadow-parts-1 pseudo-elements
672+
expect(parse('::part(button)')).toEqual(
673+
ast.selector({
674+
rules: [
675+
ast.rule({
676+
items: [
677+
ast.pseudoElement({
678+
name: 'part',
679+
argument: ast.selector({
680+
rules: [
681+
ast.rule({
682+
items: [ast.tagName({name: 'button'})]
683+
})
684+
]
685+
})
686+
})
687+
]
688+
})
689+
]
690+
})
691+
);
692+
693+
// Should parse pseudo-4 pseudo-elements
694+
expect(parse('::marker')).toEqual(
695+
ast.selector({
696+
rules: [
697+
ast.rule({
698+
items: [ast.pseudoElement({name: 'marker'})]
699+
})
700+
]
701+
})
702+
);
703+
704+
// Should parse scoping-1 pseudo-classes
705+
expect(parse(':host')).toEqual(
706+
ast.selector({
707+
rules: [
708+
ast.rule({
709+
items: [ast.pseudoClass({name: 'host'})]
710+
})
711+
]
712+
})
713+
);
714+
});
715+
});
603716
});

0 commit comments

Comments
 (0)