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));
 | 
						|
        }
 | 
						|
    }
 | 
						|
};
 |