Skip to content

Commit 824312f

Browse files
committed
fix: fix foruma parsing with negative A, closes #28
1 parent 40fb434 commit 824312f

File tree

4 files changed

+75
-19
lines changed

4 files changed

+75
-19
lines changed

src/parser.ts

+34-18
Original file line numberDiff line numberDiff line change
@@ -413,51 +413,67 @@ export function createParser(
413413
return attr;
414414
}
415415

416-
function parseNumber(signed: boolean) {
416+
function parseNumber() {
417417
let result = '';
418-
if (signed && is('-')) {
419-
result = readAndNext();
420-
}
421418
while (digitsChars[chr]) {
422419
result += readAndNext();
423420
}
424-
assert(result !== '' && result !== '-', 'Formula parse error.');
421+
assert(result !== '', 'Formula parse error.');
425422
return parseInt(result);
426423
}
427424

428425
const isNumberStart = () => is('-') || is('+') || digitsChars[chr];
429426

430427
function parseFormula(): [number, number] {
428+
if (is('e') || is('o')) {
429+
const ident = parseIdentifier();
430+
if (ident === 'even') {
431+
skipWhitespace();
432+
return [2, 0];
433+
}
434+
if (ident === 'odd') {
435+
skipWhitespace();
436+
return [2, 1];
437+
}
438+
}
439+
431440
let firstNumber: null | number = null;
441+
let firstNumberMultiplier = 1;
442+
if (is('-')) {
443+
next();
444+
firstNumberMultiplier = -1;
445+
}
432446
if (isNumberStart()) {
433447
if (is('+')) {
434448
next();
435-
assert(!is('-'), 'Formula parse error.');
436449
}
437-
firstNumber = parseNumber(true);
450+
firstNumber = parseNumber();
438451
if (!is('\\') && !is('n')) {
439-
return [0, firstNumber];
452+
return [0, firstNumber * firstNumberMultiplier];
440453
}
441454
}
442455
if (firstNumber === null) {
443456
firstNumber = 1;
444457
}
445-
const ident = parseIdentifier();
446-
if (ident === 'even') {
447-
skipWhitespace();
448-
return [2, 0];
449-
}
450-
if (ident === 'odd') {
451-
skipWhitespace();
452-
return [2, 1];
458+
firstNumber *= firstNumberMultiplier;
459+
let identifier;
460+
if (is('\\')) {
461+
next();
462+
if (isHex(chr)) {
463+
identifier = parseHex();
464+
} else {
465+
identifier = readAndNext();
466+
}
467+
} else {
468+
identifier = readAndNext();
453469
}
454-
assert(ident === 'n', 'Formula parse error');
470+
assert(identifier === 'n', 'Formula parse error: expected "n".');
455471
skipWhitespace();
456472
if (is('+') || is('-')) {
457473
const sign = is('+') ? 1 : -1;
458474
next();
459475
skipWhitespace();
460-
return [firstNumber, sign * parseNumber(false)];
476+
return [firstNumber, sign * parseNumber()];
461477
} else {
462478
return [firstNumber, 0];
463479
}

src/render.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ function renderSubstitution(sub: AstSubstitution) {
1818

1919
function renderFormula(a: number, b: number) {
2020
if (a) {
21-
let result = `${a === 1 ? '' : a}n`;
21+
let result = `${a === 1 ? '' : a === -1 ? '-' : a}n`;
2222
if (b) {
2323
result += `${b > 0 ? '+' : ''}${b}`;
2424
}

test/parser.test.ts

+26
Original file line numberDiff line numberDiff line change
@@ -921,6 +921,32 @@ describe('parse()', () => {
921921
);
922922
}
923923
});
924+
it('should properly parse with 0n and negative B', () => {
925+
const sameFormulas = [
926+
':nth-child(0n-5)',
927+
':nth-child( 0n - 5 )',
928+
':nth-child( 0n-5 )',
929+
':nth-child(-5)',
930+
':nth-child( -5 )',
931+
':nth-child( -5 )'
932+
];
933+
for (const formula of sameFormulas) {
934+
expect(parse(formula)).toEqual(
935+
ast.selector({
936+
rules: [
937+
ast.rule({
938+
pseudoClasses: [
939+
ast.pseudoClass({
940+
name: 'nth-child',
941+
argument: ast.formula({a: 0, b: -5})
942+
})
943+
]
944+
})
945+
]
946+
})
947+
);
948+
}
949+
});
924950
it('should properly parse with 0 B', () => {
925951
const sameFormulas = [
926952
':nth-child(3n+0)',

test/render.test.ts

+14
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,20 @@ const testCases = {
4343
'tag1:lang(en\\))': 'tag1:lang(en\\))',
4444
'tag1:nth-child(odd)': 'tag1:nth-child(2n+1)',
4545
'tag1:nth-child(even)': 'tag1:nth-child(2n)',
46+
'tag1:nth-child( odd )': 'tag1:nth-child(2n+1)',
47+
'tag1:nth-child( even )': 'tag1:nth-child(2n)',
48+
'tag1:nth-child(-n+3)': 'tag1:nth-child(-n+3)',
49+
'tag1:nth-child(-1n+3)': 'tag1:nth-child(-n+3)',
50+
'tag1:nth-child(-5n+3)': 'tag1:nth-child(-5n+3)',
51+
'tag1:nth-child(-5n-3)': 'tag1:nth-child(-5n-3)',
52+
'tag1:nth-child(-5\\n-3)': 'tag1:nth-child(-5n-3)',
53+
'tag1:nth-child(-5\\6e-3)': 'tag1:nth-child(-5n-3)',
54+
'tag1:nth-child(-5n)': 'tag1:nth-child(-5n)',
55+
'tag1:nth-child(5)': 'tag1:nth-child(5)',
56+
'tag1:nth-child(-5)': 'tag1:nth-child(-5)',
57+
'tag1:nth-child(0)': 'tag1:nth-child(0)',
58+
'tag1:nth-child(n)': 'tag1:nth-child(n)',
59+
'tag1:nth-child(-n)': 'tag1:nth-child(-n)',
4660
'tag1:has(.class)': 'tag1:has(.class)',
4761
'tag1:has(.class,.class2)': 'tag1:has(.class, .class2)',
4862
'tag1:has(.class:has(.subcls),.class2)': 'tag1:has(.class:has(.subcls), .class2)',

0 commit comments

Comments
 (0)