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