298 lines
9.3 KiB
JavaScript
298 lines
9.3 KiB
JavaScript
|
var cmpChar = require('../../tokenizer').cmpChar;
|
||
|
var isDigit = require('../../tokenizer').isDigit;
|
||
|
var TYPE = require('../../tokenizer').TYPE;
|
||
|
|
||
|
var WHITESPACE = TYPE.WhiteSpace;
|
||
|
var COMMENT = TYPE.Comment;
|
||
|
var IDENT = TYPE.Ident;
|
||
|
var NUMBER = TYPE.Number;
|
||
|
var DIMENSION = TYPE.Dimension;
|
||
|
var PLUSSIGN = 0x002B; // U+002B PLUS SIGN (+)
|
||
|
var HYPHENMINUS = 0x002D; // U+002D HYPHEN-MINUS (-)
|
||
|
var N = 0x006E; // U+006E LATIN SMALL LETTER N (n)
|
||
|
var DISALLOW_SIGN = true;
|
||
|
var ALLOW_SIGN = false;
|
||
|
|
||
|
function checkInteger(offset, disallowSign) {
|
||
|
var pos = this.scanner.tokenStart + offset;
|
||
|
var code = this.scanner.source.charCodeAt(pos);
|
||
|
|
||
|
if (code === PLUSSIGN || code === HYPHENMINUS) {
|
||
|
if (disallowSign) {
|
||
|
this.error('Number sign is not allowed');
|
||
|
}
|
||
|
pos++;
|
||
|
}
|
||
|
|
||
|
for (; pos < this.scanner.tokenEnd; pos++) {
|
||
|
if (!isDigit(this.scanner.source.charCodeAt(pos))) {
|
||
|
this.error('Integer is expected', pos);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function checkTokenIsInteger(disallowSign) {
|
||
|
return checkInteger.call(this, 0, disallowSign);
|
||
|
}
|
||
|
|
||
|
function expectCharCode(offset, code) {
|
||
|
if (!cmpChar(this.scanner.source, this.scanner.tokenStart + offset, code)) {
|
||
|
var msg = '';
|
||
|
|
||
|
switch (code) {
|
||
|
case N:
|
||
|
msg = 'N is expected';
|
||
|
break;
|
||
|
case HYPHENMINUS:
|
||
|
msg = 'HyphenMinus is expected';
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
this.error(msg, this.scanner.tokenStart + offset);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// ... <signed-integer>
|
||
|
// ... ['+' | '-'] <signless-integer>
|
||
|
function consumeB() {
|
||
|
var offset = 0;
|
||
|
var sign = 0;
|
||
|
var type = this.scanner.tokenType;
|
||
|
|
||
|
while (type === WHITESPACE || type === COMMENT) {
|
||
|
type = this.scanner.lookupType(++offset);
|
||
|
}
|
||
|
|
||
|
if (type !== NUMBER) {
|
||
|
if (this.scanner.isDelim(PLUSSIGN, offset) ||
|
||
|
this.scanner.isDelim(HYPHENMINUS, offset)) {
|
||
|
sign = this.scanner.isDelim(PLUSSIGN, offset) ? PLUSSIGN : HYPHENMINUS;
|
||
|
|
||
|
do {
|
||
|
type = this.scanner.lookupType(++offset);
|
||
|
} while (type === WHITESPACE || type === COMMENT);
|
||
|
|
||
|
if (type !== NUMBER) {
|
||
|
this.scanner.skip(offset);
|
||
|
checkTokenIsInteger.call(this, DISALLOW_SIGN);
|
||
|
}
|
||
|
} else {
|
||
|
return null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (offset > 0) {
|
||
|
this.scanner.skip(offset);
|
||
|
}
|
||
|
|
||
|
if (sign === 0) {
|
||
|
type = this.scanner.source.charCodeAt(this.scanner.tokenStart);
|
||
|
if (type !== PLUSSIGN && type !== HYPHENMINUS) {
|
||
|
this.error('Number sign is expected');
|
||
|
}
|
||
|
}
|
||
|
|
||
|
checkTokenIsInteger.call(this, sign !== 0);
|
||
|
return sign === HYPHENMINUS ? '-' + this.consume(NUMBER) : this.consume(NUMBER);
|
||
|
}
|
||
|
|
||
|
// An+B microsyntax https://www.w3.org/TR/css-syntax-3/#anb
|
||
|
module.exports = {
|
||
|
name: 'AnPlusB',
|
||
|
structure: {
|
||
|
a: [String, null],
|
||
|
b: [String, null]
|
||
|
},
|
||
|
parse: function() {
|
||
|
/* eslint-disable brace-style*/
|
||
|
var start = this.scanner.tokenStart;
|
||
|
var a = null;
|
||
|
var b = null;
|
||
|
|
||
|
// <integer>
|
||
|
if (this.scanner.tokenType === NUMBER) {
|
||
|
checkTokenIsInteger.call(this, ALLOW_SIGN);
|
||
|
b = this.consume(NUMBER);
|
||
|
}
|
||
|
|
||
|
// -n
|
||
|
// -n <signed-integer>
|
||
|
// -n ['+' | '-'] <signless-integer>
|
||
|
// -n- <signless-integer>
|
||
|
// <dashndashdigit-ident>
|
||
|
else if (this.scanner.tokenType === IDENT && cmpChar(this.scanner.source, this.scanner.tokenStart, HYPHENMINUS)) {
|
||
|
a = '-1';
|
||
|
|
||
|
expectCharCode.call(this, 1, N);
|
||
|
|
||
|
switch (this.scanner.getTokenLength()) {
|
||
|
// -n
|
||
|
// -n <signed-integer>
|
||
|
// -n ['+' | '-'] <signless-integer>
|
||
|
case 2:
|
||
|
this.scanner.next();
|
||
|
b = consumeB.call(this);
|
||
|
break;
|
||
|
|
||
|
// -n- <signless-integer>
|
||
|
case 3:
|
||
|
expectCharCode.call(this, 2, HYPHENMINUS);
|
||
|
|
||
|
this.scanner.next();
|
||
|
this.scanner.skipSC();
|
||
|
|
||
|
checkTokenIsInteger.call(this, DISALLOW_SIGN);
|
||
|
|
||
|
b = '-' + this.consume(NUMBER);
|
||
|
break;
|
||
|
|
||
|
// <dashndashdigit-ident>
|
||
|
default:
|
||
|
expectCharCode.call(this, 2, HYPHENMINUS);
|
||
|
checkInteger.call(this, 3, DISALLOW_SIGN);
|
||
|
this.scanner.next();
|
||
|
|
||
|
b = this.scanner.substrToCursor(start + 2);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// '+'? n
|
||
|
// '+'? n <signed-integer>
|
||
|
// '+'? n ['+' | '-'] <signless-integer>
|
||
|
// '+'? n- <signless-integer>
|
||
|
// '+'? <ndashdigit-ident>
|
||
|
else if (this.scanner.tokenType === IDENT || (this.scanner.isDelim(PLUSSIGN) && this.scanner.lookupType(1) === IDENT)) {
|
||
|
var sign = 0;
|
||
|
a = '1';
|
||
|
|
||
|
// just ignore a plus
|
||
|
if (this.scanner.isDelim(PLUSSIGN)) {
|
||
|
sign = 1;
|
||
|
this.scanner.next();
|
||
|
}
|
||
|
|
||
|
expectCharCode.call(this, 0, N);
|
||
|
|
||
|
switch (this.scanner.getTokenLength()) {
|
||
|
// '+'? n
|
||
|
// '+'? n <signed-integer>
|
||
|
// '+'? n ['+' | '-'] <signless-integer>
|
||
|
case 1:
|
||
|
this.scanner.next();
|
||
|
b = consumeB.call(this);
|
||
|
break;
|
||
|
|
||
|
// '+'? n- <signless-integer>
|
||
|
case 2:
|
||
|
expectCharCode.call(this, 1, HYPHENMINUS);
|
||
|
|
||
|
this.scanner.next();
|
||
|
this.scanner.skipSC();
|
||
|
|
||
|
checkTokenIsInteger.call(this, DISALLOW_SIGN);
|
||
|
|
||
|
b = '-' + this.consume(NUMBER);
|
||
|
break;
|
||
|
|
||
|
// '+'? <ndashdigit-ident>
|
||
|
default:
|
||
|
expectCharCode.call(this, 1, HYPHENMINUS);
|
||
|
checkInteger.call(this, 2, DISALLOW_SIGN);
|
||
|
this.scanner.next();
|
||
|
|
||
|
b = this.scanner.substrToCursor(start + sign + 1);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// <ndashdigit-dimension>
|
||
|
// <ndash-dimension> <signless-integer>
|
||
|
// <n-dimension>
|
||
|
// <n-dimension> <signed-integer>
|
||
|
// <n-dimension> ['+' | '-'] <signless-integer>
|
||
|
else if (this.scanner.tokenType === DIMENSION) {
|
||
|
var code = this.scanner.source.charCodeAt(this.scanner.tokenStart);
|
||
|
var sign = code === PLUSSIGN || code === HYPHENMINUS;
|
||
|
|
||
|
for (var i = this.scanner.tokenStart + sign; i < this.scanner.tokenEnd; i++) {
|
||
|
if (!isDigit(this.scanner.source.charCodeAt(i))) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (i === this.scanner.tokenStart + sign) {
|
||
|
this.error('Integer is expected', this.scanner.tokenStart + sign);
|
||
|
}
|
||
|
|
||
|
expectCharCode.call(this, i - this.scanner.tokenStart, N);
|
||
|
a = this.scanner.source.substring(start, i);
|
||
|
|
||
|
// <n-dimension>
|
||
|
// <n-dimension> <signed-integer>
|
||
|
// <n-dimension> ['+' | '-'] <signless-integer>
|
||
|
if (i + 1 === this.scanner.tokenEnd) {
|
||
|
this.scanner.next();
|
||
|
b = consumeB.call(this);
|
||
|
} else {
|
||
|
expectCharCode.call(this, i - this.scanner.tokenStart + 1, HYPHENMINUS);
|
||
|
|
||
|
// <ndash-dimension> <signless-integer>
|
||
|
if (i + 2 === this.scanner.tokenEnd) {
|
||
|
this.scanner.next();
|
||
|
this.scanner.skipSC();
|
||
|
checkTokenIsInteger.call(this, DISALLOW_SIGN);
|
||
|
b = '-' + this.consume(NUMBER);
|
||
|
}
|
||
|
// <ndashdigit-dimension>
|
||
|
else {
|
||
|
checkInteger.call(this, i - this.scanner.tokenStart + 2, DISALLOW_SIGN);
|
||
|
this.scanner.next();
|
||
|
b = this.scanner.substrToCursor(i + 1);
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
this.error();
|
||
|
}
|
||
|
|
||
|
if (a !== null && a.charCodeAt(0) === PLUSSIGN) {
|
||
|
a = a.substr(1);
|
||
|
}
|
||
|
|
||
|
if (b !== null && b.charCodeAt(0) === PLUSSIGN) {
|
||
|
b = b.substr(1);
|
||
|
}
|
||
|
|
||
|
return {
|
||
|
type: 'AnPlusB',
|
||
|
loc: this.getLocation(start, this.scanner.tokenStart),
|
||
|
a: a,
|
||
|
b: b
|
||
|
};
|
||
|
},
|
||
|
generate: function(node) {
|
||
|
var a = node.a !== null && node.a !== undefined;
|
||
|
var b = node.b !== null && node.b !== undefined;
|
||
|
|
||
|
if (a) {
|
||
|
this.chunk(
|
||
|
node.a === '+1' ? '+n' : // eslint-disable-line operator-linebreak, indent
|
||
|
node.a === '1' ? 'n' : // eslint-disable-line operator-linebreak, indent
|
||
|
node.a === '-1' ? '-n' : // eslint-disable-line operator-linebreak, indent
|
||
|
node.a + 'n' // eslint-disable-line operator-linebreak, indent
|
||
|
);
|
||
|
|
||
|
if (b) {
|
||
|
b = String(node.b);
|
||
|
if (b.charAt(0) === '-' || b.charAt(0) === '+') {
|
||
|
this.chunk(b.charAt(0));
|
||
|
this.chunk(b.substr(1));
|
||
|
} else {
|
||
|
this.chunk('+');
|
||
|
this.chunk(b);
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
this.chunk(String(node.b));
|
||
|
}
|
||
|
}
|
||
|
};
|