|
import { |
|
isHexDigit, |
|
cmpChar, |
|
Ident, |
|
Delim, |
|
Number as NumberToken, |
|
Dimension |
|
} from '../tokenizer/index.js'; |
|
|
|
const PLUSSIGN = 0x002B; |
|
const HYPHENMINUS = 0x002D; |
|
const QUESTIONMARK = 0x003F; |
|
const U = 0x0075; |
|
|
|
function isDelim(token, code) { |
|
return token !== null && token.type === Delim && token.value.charCodeAt(0) === code; |
|
} |
|
|
|
function startsWith(token, code) { |
|
return token.value.charCodeAt(0) === code; |
|
} |
|
|
|
function hexSequence(token, offset, allowDash) { |
|
let hexlen = 0; |
|
|
|
for (let pos = offset; pos < token.value.length; pos++) { |
|
const code = token.value.charCodeAt(pos); |
|
|
|
if (code === HYPHENMINUS && allowDash && hexlen !== 0) { |
|
hexSequence(token, offset + hexlen + 1, false); |
|
return 6; |
|
} |
|
|
|
if (!isHexDigit(code)) { |
|
return 0; |
|
} |
|
|
|
if (++hexlen > 6) { |
|
return 0; |
|
}; |
|
} |
|
|
|
return hexlen; |
|
} |
|
|
|
function withQuestionMarkSequence(consumed, length, getNextToken) { |
|
if (!consumed) { |
|
return 0; |
|
} |
|
|
|
while (isDelim(getNextToken(length), QUESTIONMARK)) { |
|
if (++consumed > 6) { |
|
return 0; |
|
} |
|
|
|
length++; |
|
} |
|
|
|
return length; |
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
export default function urange(token, getNextToken) { |
|
let length = 0; |
|
|
|
|
|
if (token === null || token.type !== Ident || !cmpChar(token.value, 0, U)) { |
|
return 0; |
|
} |
|
|
|
token = getNextToken(++length); |
|
if (token === null) { |
|
return 0; |
|
} |
|
|
|
|
|
|
|
if (isDelim(token, PLUSSIGN)) { |
|
token = getNextToken(++length); |
|
if (token === null) { |
|
return 0; |
|
} |
|
|
|
if (token.type === Ident) { |
|
|
|
return withQuestionMarkSequence(hexSequence(token, 0, true), ++length, getNextToken); |
|
} |
|
|
|
if (isDelim(token, QUESTIONMARK)) { |
|
|
|
return withQuestionMarkSequence(1, ++length, getNextToken); |
|
} |
|
|
|
|
|
return 0; |
|
} |
|
|
|
|
|
|
|
|
|
if (token.type === NumberToken) { |
|
const consumedHexLength = hexSequence(token, 1, true); |
|
if (consumedHexLength === 0) { |
|
return 0; |
|
} |
|
|
|
token = getNextToken(++length); |
|
if (token === null) { |
|
|
|
return length; |
|
} |
|
|
|
if (token.type === Dimension || token.type === NumberToken) { |
|
|
|
|
|
if (!startsWith(token, HYPHENMINUS) || !hexSequence(token, 1, false)) { |
|
return 0; |
|
} |
|
|
|
return length + 1; |
|
} |
|
|
|
|
|
return withQuestionMarkSequence(consumedHexLength, length, getNextToken); |
|
} |
|
|
|
|
|
if (token.type === Dimension) { |
|
return withQuestionMarkSequence(hexSequence(token, 1, true), ++length, getNextToken); |
|
} |
|
|
|
return 0; |
|
}; |
|
|