868 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			868 lines
		
	
	
		
			30 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| var Marker = require('./marker');
 | |
| var Token = require('./token');
 | |
| 
 | |
| var formatPosition = require('../utils/format-position');
 | |
| 
 | |
| var Level = {
 | |
|   BLOCK: 'block',
 | |
|   COMMENT: 'comment',
 | |
|   DOUBLE_QUOTE: 'double-quote',
 | |
|   RULE: 'rule',
 | |
|   SINGLE_QUOTE: 'single-quote'
 | |
| };
 | |
| 
 | |
| var AT_RULES = [
 | |
|   '@charset',
 | |
|   '@import'
 | |
| ];
 | |
| 
 | |
| var BLOCK_RULES = [
 | |
|   '@-moz-document',
 | |
|   '@document',
 | |
|   '@-moz-keyframes',
 | |
|   '@-ms-keyframes',
 | |
|   '@-o-keyframes',
 | |
|   '@-webkit-keyframes',
 | |
|   '@keyframes',
 | |
|   '@media',
 | |
|   '@supports',
 | |
|   '@container',
 | |
|   '@layer'
 | |
| ];
 | |
| 
 | |
| var IGNORE_END_COMMENT_PATTERN = /\/\* clean-css ignore:end \*\/$/;
 | |
| var IGNORE_START_COMMENT_PATTERN = /^\/\* clean-css ignore:start \*\//;
 | |
| 
 | |
| var PAGE_MARGIN_BOXES = [
 | |
|   '@bottom-center',
 | |
|   '@bottom-left',
 | |
|   '@bottom-left-corner',
 | |
|   '@bottom-right',
 | |
|   '@bottom-right-corner',
 | |
|   '@left-bottom',
 | |
|   '@left-middle',
 | |
|   '@left-top',
 | |
|   '@right-bottom',
 | |
|   '@right-middle',
 | |
|   '@right-top',
 | |
|   '@top-center',
 | |
|   '@top-left',
 | |
|   '@top-left-corner',
 | |
|   '@top-right',
 | |
|   '@top-right-corner'
 | |
| ];
 | |
| 
 | |
| var EXTRA_PAGE_BOXES = [
 | |
|   '@footnote',
 | |
|   '@footnotes',
 | |
|   '@left',
 | |
|   '@page-float-bottom',
 | |
|   '@page-float-top',
 | |
|   '@right'
 | |
| ];
 | |
| 
 | |
| var REPEAT_PATTERN = /^\[\s{0,31}\d+\s{0,31}\]$/;
 | |
| var TAIL_BROKEN_VALUE_PATTERN = /([^}])\}*$/;
 | |
| var RULE_WORD_SEPARATOR_PATTERN = /[\s(]/;
 | |
| 
 | |
| function tokenize(source, externalContext) {
 | |
|   var internalContext = {
 | |
|     level: Level.BLOCK,
 | |
|     position: {
 | |
|       source: externalContext.source || undefined,
 | |
|       line: 1,
 | |
|       column: 0,
 | |
|       index: 0
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   return intoTokens(source, externalContext, internalContext, false);
 | |
| }
 | |
| 
 | |
| function intoTokens(source, externalContext, internalContext, isNested) {
 | |
|   var allTokens = [];
 | |
|   var newTokens = allTokens;
 | |
|   var lastToken;
 | |
|   var ruleToken;
 | |
|   var ruleTokens = [];
 | |
|   var propertyToken;
 | |
|   var metadata;
 | |
|   var metadatas = [];
 | |
|   var level = internalContext.level;
 | |
|   var levels = [];
 | |
|   var buffer = [];
 | |
|   var buffers = [];
 | |
|   var isBufferEmpty = true;
 | |
|   var serializedBuffer;
 | |
|   var serializedBufferPart;
 | |
|   var roundBracketLevel = 0;
 | |
|   var isQuoted;
 | |
|   var isSpace;
 | |
|   var isNewLineNix;
 | |
|   var isNewLineWin;
 | |
|   var isCarriageReturn;
 | |
|   var isCommentStart;
 | |
|   var wasCommentStart = false;
 | |
|   var isCommentEnd;
 | |
|   var wasCommentEnd = false;
 | |
|   var isCommentEndMarker;
 | |
|   var isEscaped;
 | |
|   var wasEscaped = false;
 | |
|   var characterWithNoSpecialMeaning;
 | |
|   var isPreviousDash = false;
 | |
|   var isVariable = false;
 | |
|   var isRaw = false;
 | |
|   var seekingValue = false;
 | |
|   var seekingPropertyBlockClosing = false;
 | |
|   var position = internalContext.position;
 | |
|   var lastCommentStartAt;
 | |
| 
 | |
|   for (; position.index < source.length; position.index++) {
 | |
|     var character = source[position.index];
 | |
| 
 | |
|     isQuoted = level == Level.SINGLE_QUOTE || level == Level.DOUBLE_QUOTE;
 | |
|     isSpace = character == Marker.SPACE || character == Marker.TAB;
 | |
|     isNewLineNix = character == Marker.NEW_LINE_NIX;
 | |
|     isNewLineWin = character == Marker.NEW_LINE_NIX
 | |
|       && source[position.index - 1] == Marker.CARRIAGE_RETURN;
 | |
|     isCarriageReturn = character == Marker.CARRIAGE_RETURN
 | |
|       && source[position.index + 1] && source[position.index + 1] != Marker.NEW_LINE_NIX;
 | |
|     isCommentStart = !wasCommentEnd
 | |
|       && level != Level.COMMENT && !isQuoted
 | |
|       && character == Marker.ASTERISK && source[position.index - 1] == Marker.FORWARD_SLASH;
 | |
|     isCommentEndMarker = !wasCommentStart
 | |
|       && !isQuoted && character == Marker.FORWARD_SLASH
 | |
|       && source[position.index - 1] == Marker.ASTERISK;
 | |
|     isCommentEnd = level == Level.COMMENT && isCommentEndMarker;
 | |
|     characterWithNoSpecialMeaning = !isSpace && !isCarriageReturn && (character >= 'A' && character <= 'Z' || character >= 'a' && character <= 'z' || character >= '0' && character <= '9' || character == '-');
 | |
|     isVariable = isVariable || (level != Level.COMMENT && !seekingValue && isPreviousDash && character === '-' && buffer.length === 1);
 | |
|     isPreviousDash = character === '-';
 | |
|     roundBracketLevel = Math.max(roundBracketLevel, 0);
 | |
| 
 | |
|     metadata = isBufferEmpty
 | |
|       ? [position.line, position.column, position.source]
 | |
|       : metadata;
 | |
| 
 | |
|     if (isEscaped) {
 | |
|       // previous character was a backslash
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (characterWithNoSpecialMeaning) {
 | |
|       // it's just an alphanumeric character or a hyphen (part of any rule or property name) so let's end it quickly
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if ((isSpace || isNewLineNix && !isNewLineWin) && (isQuoted || level == Level.COMMENT)) {
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if ((isSpace || isNewLineNix && !isNewLineWin) && isBufferEmpty) {
 | |
|       // noop
 | |
|     } else if (!isCommentEnd && level == Level.COMMENT) {
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (!isCommentStart && !isCommentEnd && isRaw) {
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (isCommentStart
 | |
|         && isVariable
 | |
|         && (level == Level.BLOCK || level == Level.RULE) && buffer.length > 1) {
 | |
|       // comment start within a variable, e.g. var(/*<--
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
| 
 | |
|       levels.push(level);
 | |
|       level = Level.COMMENT;
 | |
|     } else if (isCommentStart && (level == Level.BLOCK || level == Level.RULE) && buffer.length > 1) {
 | |
|       // comment start within block preceded by some content, e.g. div/*<--
 | |
|       metadatas.push(metadata);
 | |
|       buffer.push(character);
 | |
|       buffers.push(buffer.slice(0, -2));
 | |
|       isBufferEmpty = false;
 | |
| 
 | |
|       buffer = buffer.slice(-2);
 | |
|       metadata = [position.line, position.column - 1, position.source];
 | |
| 
 | |
|       levels.push(level);
 | |
|       level = Level.COMMENT;
 | |
|     } else if (isCommentStart) {
 | |
|       // comment start, e.g. /*<--
 | |
|       levels.push(level);
 | |
|       level = Level.COMMENT;
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (isCommentEnd && isVariable) {
 | |
|       // comment end within a variable, e.g. var(/*!*/<--
 | |
|       buffer.push(character);
 | |
|       level = levels.pop();
 | |
|     } else if (isCommentEnd && isIgnoreStartComment(buffer)) {
 | |
|       // ignore:start comment end, e.g. /* clean-css ignore:start */<--
 | |
|       serializedBuffer = buffer.join('').trim() + character;
 | |
|       lastToken = [
 | |
|         Token.COMMENT,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ];
 | |
|       newTokens.push(lastToken);
 | |
| 
 | |
|       isRaw = true;
 | |
|       metadata = metadatas.pop() || null;
 | |
|       buffer = buffers.pop() || [];
 | |
|       isBufferEmpty = buffer.length === 0;
 | |
|     } else if (isCommentEnd && isIgnoreEndComment(buffer)) {
 | |
|       // ignore:start comment end, e.g. /* clean-css ignore:end */<--
 | |
|       serializedBuffer = buffer.join('') + character;
 | |
|       lastCommentStartAt = serializedBuffer.lastIndexOf(Marker.FORWARD_SLASH + Marker.ASTERISK);
 | |
| 
 | |
|       serializedBufferPart = serializedBuffer.substring(0, lastCommentStartAt);
 | |
|       lastToken = [
 | |
|         Token.RAW,
 | |
|         serializedBufferPart,
 | |
|         [originalMetadata(metadata, serializedBufferPart, externalContext)]
 | |
|       ];
 | |
|       newTokens.push(lastToken);
 | |
| 
 | |
|       serializedBufferPart = serializedBuffer.substring(lastCommentStartAt);
 | |
|       metadata = [position.line, position.column - serializedBufferPart.length + 1, position.source];
 | |
|       lastToken = [
 | |
|         Token.COMMENT,
 | |
|         serializedBufferPart,
 | |
|         [originalMetadata(metadata, serializedBufferPart, externalContext)]
 | |
|       ];
 | |
|       newTokens.push(lastToken);
 | |
| 
 | |
|       isRaw = false;
 | |
|       level = levels.pop();
 | |
|       metadata = metadatas.pop() || null;
 | |
|       buffer = buffers.pop() || [];
 | |
|       isBufferEmpty = buffer.length === 0;
 | |
|     } else if (isCommentEnd) {
 | |
|       // comment end, e.g. /* comment */<--
 | |
|       serializedBuffer = buffer.join('').trim() + character;
 | |
|       lastToken = [
 | |
|         Token.COMMENT,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ];
 | |
|       newTokens.push(lastToken);
 | |
| 
 | |
|       level = levels.pop();
 | |
|       metadata = metadatas.pop() || null;
 | |
|       buffer = buffers.pop() || [];
 | |
|       isBufferEmpty = buffer.length === 0;
 | |
|     } else if (isCommentEndMarker && source[position.index + 1] != Marker.ASTERISK) {
 | |
|       externalContext.warnings.push('Unexpected \'*/\' at ' + formatPosition([position.line, position.column, position.source]) + '.');
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.SINGLE_QUOTE && !isQuoted) {
 | |
|       // single quotation start, e.g. a[href^='https<--
 | |
|       levels.push(level);
 | |
|       level = Level.SINGLE_QUOTE;
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (character == Marker.SINGLE_QUOTE && level == Level.SINGLE_QUOTE) {
 | |
|       // single quotation end, e.g. a[href^='https'<--
 | |
|       level = levels.pop();
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (character == Marker.DOUBLE_QUOTE && !isQuoted) {
 | |
|       // double quotation start, e.g. a[href^="<--
 | |
|       levels.push(level);
 | |
|       level = Level.DOUBLE_QUOTE;
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (character == Marker.DOUBLE_QUOTE && level == Level.DOUBLE_QUOTE) {
 | |
|       // double quotation end, e.g. a[href^="https"<--
 | |
|       level = levels.pop();
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (character != Marker.CLOSE_ROUND_BRACKET
 | |
|       && character != Marker.OPEN_ROUND_BRACKET
 | |
|       && level != Level.COMMENT && !isQuoted && roundBracketLevel > 0) {
 | |
|       // character inside any function, e.g. hsla(.<--
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (character == Marker.OPEN_ROUND_BRACKET
 | |
|       && !isQuoted && level != Level.COMMENT
 | |
|       && !seekingValue) {
 | |
|       // round open bracket, e.g. @import url(<--
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
| 
 | |
|       roundBracketLevel++;
 | |
|     } else if (character == Marker.CLOSE_ROUND_BRACKET
 | |
|       && !isQuoted
 | |
|       && level != Level.COMMENT
 | |
|       && !seekingValue) {
 | |
|       // round open bracket, e.g. @import url(test.css)<--
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
| 
 | |
|       roundBracketLevel--;
 | |
|     } else if (character == Marker.SEMICOLON && level == Level.BLOCK && buffer[0] == Marker.AT) {
 | |
|       // semicolon ending rule at block level, e.g. @import '...';<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       allTokens.push([
 | |
|         Token.AT_RULE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.COMMA && level == Level.BLOCK && ruleToken) {
 | |
|       // comma separator at block level, e.g. a,div,<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       ruleToken[1].push([
 | |
|         tokenScopeFrom(ruleToken[0]),
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext, ruleToken[1].length)]
 | |
|       ]);
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.COMMA && level == Level.BLOCK && tokenTypeFrom(buffer) == Token.AT_RULE) {
 | |
|       // comma separator at block level, e.g. @import url(...) screen,<--
 | |
|       // keep iterating as end semicolon will create the token
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (character == Marker.COMMA && level == Level.BLOCK) {
 | |
|       // comma separator at block level, e.g. a,<--
 | |
|       ruleToken = [tokenTypeFrom(buffer), [], []];
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       ruleToken[1].push([
 | |
|         tokenScopeFrom(ruleToken[0]),
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext, 0)]
 | |
|       ]);
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.OPEN_CURLY_BRACKET
 | |
|       && level == Level.BLOCK
 | |
|       && ruleToken
 | |
|       && ruleToken[0] == Token.NESTED_BLOCK) {
 | |
|       // open brace opening at-rule at block level, e.g. @media{<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       ruleToken[1].push([
 | |
|         Token.NESTED_BLOCK_SCOPE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
|       allTokens.push(ruleToken);
 | |
| 
 | |
|       levels.push(level);
 | |
|       position.column++;
 | |
|       position.index++;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
| 
 | |
|       ruleToken[2] = intoTokens(source, externalContext, internalContext, true);
 | |
|       ruleToken = null;
 | |
|     } else if (character == Marker.OPEN_CURLY_BRACKET
 | |
|       && level == Level.BLOCK
 | |
|       && tokenTypeFrom(buffer) == Token.NESTED_BLOCK) {
 | |
|       // open brace opening at-rule at block level, e.g. @media{<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       ruleToken = ruleToken || [Token.NESTED_BLOCK, [], []];
 | |
|       ruleToken[1].push([
 | |
|         Token.NESTED_BLOCK_SCOPE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
|       allTokens.push(ruleToken);
 | |
| 
 | |
|       levels.push(level);
 | |
|       position.column++;
 | |
|       position.index++;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|       isVariable = false;
 | |
| 
 | |
|       ruleToken[2] = intoTokens(source, externalContext, internalContext, true);
 | |
|       ruleToken = null;
 | |
|     } else if (character == Marker.OPEN_CURLY_BRACKET && level == Level.BLOCK) {
 | |
|       // open brace opening rule at block level, e.g. div{<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       ruleToken = ruleToken || [tokenTypeFrom(buffer), [], []];
 | |
|       ruleToken[1].push([
 | |
|         tokenScopeFrom(ruleToken[0]),
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext, ruleToken[1].length)]
 | |
|       ]);
 | |
|       newTokens = ruleToken[2];
 | |
|       allTokens.push(ruleToken);
 | |
| 
 | |
|       levels.push(level);
 | |
|       level = Level.RULE;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.OPEN_CURLY_BRACKET && level == Level.RULE && seekingValue) {
 | |
|       // open brace opening rule at rule level, e.g. div{--variable:{<--
 | |
|       ruleTokens.push(ruleToken);
 | |
|       ruleToken = [Token.PROPERTY_BLOCK, []];
 | |
|       propertyToken.push(ruleToken);
 | |
|       newTokens = ruleToken[1];
 | |
| 
 | |
|       levels.push(level);
 | |
|       level = Level.RULE;
 | |
|       seekingValue = false;
 | |
|     } else if (character == Marker.OPEN_CURLY_BRACKET && level == Level.RULE && isPageMarginBox(buffer)) {
 | |
|       // open brace opening page-margin box at rule level, e.g. @page{@top-center{<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       ruleTokens.push(ruleToken);
 | |
|       ruleToken = [Token.AT_RULE_BLOCK, [], []];
 | |
|       ruleToken[1].push([
 | |
|         Token.AT_RULE_BLOCK_SCOPE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
|       newTokens.push(ruleToken);
 | |
|       newTokens = ruleToken[2];
 | |
| 
 | |
|       levels.push(level);
 | |
|       level = Level.RULE;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.COLON && level == Level.RULE && !seekingValue) {
 | |
|       // colon at rule level, e.g. a{color:<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       propertyToken = [
 | |
|         Token.PROPERTY,
 | |
|         [
 | |
|           Token.PROPERTY_NAME,
 | |
|           serializedBuffer,
 | |
|           [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|         ]
 | |
|       ];
 | |
|       newTokens.push(propertyToken);
 | |
| 
 | |
|       seekingValue = true;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.SEMICOLON
 | |
|       && level == Level.RULE
 | |
|       && propertyToken
 | |
|       && ruleTokens.length > 0
 | |
|       && !isBufferEmpty
 | |
|       && buffer[0] == Marker.AT) {
 | |
|       // semicolon at rule level for at-rule, e.g. a{--color:{@apply(--other-color);<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       ruleToken[1].push([
 | |
|         Token.AT_RULE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.SEMICOLON && level == Level.RULE && propertyToken && !isBufferEmpty) {
 | |
|       // semicolon at rule level, e.g. a{color:red;<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
| 
 | |
|       propertyToken = null;
 | |
|       seekingValue = false;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|       isVariable = false;
 | |
|     } else if (character == Marker.SEMICOLON
 | |
|       && level == Level.RULE
 | |
|       && propertyToken
 | |
|       && isBufferEmpty
 | |
|       && isVariable
 | |
|       && !propertyToken[2]) {
 | |
|       // semicolon after empty variable value at rule level, e.g. a{--color: ;<--
 | |
|       propertyToken.push([Token.PROPERTY_VALUE, ' ', [originalMetadata(metadata, ' ', externalContext)]]);
 | |
|       isVariable = false;
 | |
|       propertyToken = null;
 | |
|       seekingValue = false;
 | |
|     } else if (character == Marker.SEMICOLON && level == Level.RULE && propertyToken && isBufferEmpty) {
 | |
|       // semicolon after bracketed value at rule level, e.g. a{color:rgb(...);<--
 | |
|       propertyToken = null;
 | |
|       seekingValue = false;
 | |
|     } else if (character == Marker.SEMICOLON
 | |
|       && level == Level.RULE
 | |
|       && !isBufferEmpty
 | |
|       && buffer[0] == Marker.AT) {
 | |
|       // semicolon for at-rule at rule level, e.g. a{@apply(--variable);<--
 | |
|       serializedBuffer = buffer.join('');
 | |
|       newTokens.push([
 | |
|         Token.AT_RULE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
| 
 | |
|       seekingValue = false;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.SEMICOLON && level == Level.RULE && seekingPropertyBlockClosing) {
 | |
|       // close brace after a property block at rule level, e.g. a{--custom:{color:red;};<--
 | |
|       seekingPropertyBlockClosing = false;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.SEMICOLON && level == Level.RULE && isBufferEmpty) {
 | |
|       // stray semicolon at rule level, e.g. a{;<--
 | |
|       // noop
 | |
|     } else if (character == Marker.CLOSE_CURLY_BRACKET
 | |
|       && level == Level.RULE
 | |
|       && propertyToken
 | |
|       && seekingValue
 | |
|       && !isBufferEmpty && ruleTokens.length > 0) {
 | |
|       // close brace at rule level, e.g. a{--color:{color:red}<--
 | |
|       serializedBuffer = buffer.join('');
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
|       propertyToken = null;
 | |
|       ruleToken = ruleTokens.pop();
 | |
|       newTokens = ruleToken[2];
 | |
| 
 | |
|       level = levels.pop();
 | |
|       seekingValue = false;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.CLOSE_CURLY_BRACKET
 | |
|       && level == Level.RULE
 | |
|       && propertyToken
 | |
|       && !isBufferEmpty
 | |
|       && buffer[0] == Marker.AT
 | |
|       && ruleTokens.length > 0) {
 | |
|       // close brace at rule level for at-rule, e.g. a{--color:{@apply(--other-color)}<--
 | |
|       serializedBuffer = buffer.join('');
 | |
|       ruleToken[1].push([
 | |
|         Token.AT_RULE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
|       propertyToken = null;
 | |
|       ruleToken = ruleTokens.pop();
 | |
|       newTokens = ruleToken[2];
 | |
| 
 | |
|       level = levels.pop();
 | |
|       seekingValue = false;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.CLOSE_CURLY_BRACKET
 | |
|       && level == Level.RULE
 | |
|       && propertyToken
 | |
|       && ruleTokens.length > 0) {
 | |
|       // close brace at rule level after space, e.g. a{--color:{color:red }<--
 | |
|       propertyToken = null;
 | |
|       ruleToken = ruleTokens.pop();
 | |
|       newTokens = ruleToken[2];
 | |
| 
 | |
|       level = levels.pop();
 | |
|       seekingValue = false;
 | |
|     } else if (character == Marker.CLOSE_CURLY_BRACKET
 | |
|       && level == Level.RULE
 | |
|       && propertyToken
 | |
|       && !isBufferEmpty) {
 | |
|       // close brace at rule level, e.g. a{color:red}<--
 | |
|       serializedBuffer = buffer.join('');
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
|       propertyToken = null;
 | |
|       ruleToken = ruleTokens.pop();
 | |
|       newTokens = allTokens;
 | |
| 
 | |
|       level = levels.pop();
 | |
|       seekingValue = false;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.CLOSE_CURLY_BRACKET
 | |
|       && level == Level.RULE
 | |
|       && !isBufferEmpty
 | |
|       && buffer[0] == Marker.AT) {
 | |
|       // close brace after at-rule at rule level, e.g. a{@apply(--variable)}<--
 | |
|       propertyToken = null;
 | |
|       ruleToken = null;
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       newTokens.push([
 | |
|         Token.AT_RULE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
|       newTokens = allTokens;
 | |
| 
 | |
|       level = levels.pop();
 | |
|       seekingValue = false;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.CLOSE_CURLY_BRACKET
 | |
|       && level == Level.RULE
 | |
|       && levels[levels.length - 1] == Level.RULE) {
 | |
|       // close brace after a property block at rule level, e.g. a{--custom:{color:red;}<--
 | |
|       propertyToken = null;
 | |
|       ruleToken = ruleTokens.pop();
 | |
|       newTokens = ruleToken[2];
 | |
| 
 | |
|       level = levels.pop();
 | |
|       seekingValue = false;
 | |
|       seekingPropertyBlockClosing = true;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.CLOSE_CURLY_BRACKET
 | |
|       && level == Level.RULE
 | |
|       && isVariable
 | |
|       && propertyToken
 | |
|       && !propertyToken[2]) {
 | |
|       // close brace after an empty variable declaration inside a rule, e.g. a{--color: }<--
 | |
|       propertyToken.push([Token.PROPERTY_VALUE, ' ', [originalMetadata(metadata, ' ', externalContext)]]);
 | |
|       isVariable = false;
 | |
|       propertyToken = null;
 | |
|       ruleToken = null;
 | |
|       newTokens = allTokens;
 | |
| 
 | |
|       level = levels.pop();
 | |
|       seekingValue = false;
 | |
|       isVariable = false;
 | |
|     } else if (character == Marker.CLOSE_CURLY_BRACKET && level == Level.RULE) {
 | |
|       // close brace after a rule, e.g. a{color:red;}<--
 | |
|       propertyToken = null;
 | |
|       ruleToken = null;
 | |
|       newTokens = allTokens;
 | |
| 
 | |
|       level = levels.pop();
 | |
|       seekingValue = false;
 | |
|       isVariable = false;
 | |
|     } else if (character == Marker.CLOSE_CURLY_BRACKET
 | |
|       && level == Level.BLOCK
 | |
|       && !isNested
 | |
|       && position.index <= source.length - 1) {
 | |
|       // stray close brace at block level, e.g. a{color:red}color:blue}<--
 | |
|       externalContext.warnings.push('Unexpected \'}\' at ' + formatPosition([position.line, position.column, position.source]) + '.');
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     } else if (character == Marker.CLOSE_CURLY_BRACKET && level == Level.BLOCK) {
 | |
|       // close brace at block level, e.g. @media screen {...}<--
 | |
|       break;
 | |
|     } else if (character == Marker.OPEN_ROUND_BRACKET && level == Level.RULE && seekingValue) {
 | |
|       // round open bracket, e.g. a{color:hsla(<--
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|       roundBracketLevel++;
 | |
|     } else if (character == Marker.CLOSE_ROUND_BRACKET
 | |
|       && level == Level.RULE
 | |
|       && seekingValue
 | |
|       && roundBracketLevel == 1) {
 | |
|       // round close bracket, e.g. a{color:hsla(0,0%,0%)<--
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
| 
 | |
|       roundBracketLevel--;
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|       isVariable = false;
 | |
|     } else if (character == Marker.CLOSE_ROUND_BRACKET && level == Level.RULE && seekingValue) {
 | |
|       // round close bracket within other brackets, e.g. a{width:calc((10rem / 2)<--
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|       isVariable = false;
 | |
|       roundBracketLevel--;
 | |
|     } else if (character == Marker.FORWARD_SLASH
 | |
|       && source[position.index + 1] != Marker.ASTERISK
 | |
|       && level == Level.RULE
 | |
|       && seekingValue
 | |
|       && !isBufferEmpty) {
 | |
|       // forward slash within a property, e.g. a{background:url(image.png) 0 0/<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         character,
 | |
|         [[position.line, position.column, position.source]]
 | |
|       ]);
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.FORWARD_SLASH
 | |
|       && source[position.index + 1] != Marker.ASTERISK
 | |
|       && level == Level.RULE
 | |
|       && seekingValue) {
 | |
|       // forward slash within a property after space, e.g. a{background:url(image.png) 0 0 /<--
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         character,
 | |
|         [[position.line, position.column, position.source]]
 | |
|       ]);
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.COMMA && level == Level.RULE && seekingValue && !isBufferEmpty) {
 | |
|       // comma within a property, e.g. a{background:url(image.png),<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         character,
 | |
|         [[position.line, position.column, position.source]]
 | |
|       ]);
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.COMMA && level == Level.RULE && seekingValue) {
 | |
|       // comma within a property after space, e.g. a{background:url(image.png) ,<--
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         character,
 | |
|         [[position.line, position.column, position.source]]
 | |
|       ]);
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (character == Marker.CLOSE_SQUARE_BRACKET
 | |
|       && propertyToken
 | |
|       && propertyToken.length > 1
 | |
|       && !isBufferEmpty
 | |
|       && isRepeatToken(buffer)) {
 | |
|       buffer.push(character);
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       propertyToken[propertyToken.length - 1][1] += serializedBuffer;
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if ((isSpace || (isNewLineNix && !isNewLineWin))
 | |
|       && level == Level.RULE
 | |
|       && seekingValue
 | |
|       && propertyToken
 | |
|       && !isBufferEmpty) {
 | |
|       // space or *nix newline within property, e.g. a{margin:0 <--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (isNewLineWin && level == Level.RULE && seekingValue && propertyToken && buffer.length > 1) {
 | |
|       // win newline within property, e.g. a{margin:0\r\n<--
 | |
|       serializedBuffer = buffer.join('').trim();
 | |
|       propertyToken.push([
 | |
|         Token.PROPERTY_VALUE,
 | |
|         serializedBuffer,
 | |
|         [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|       ]);
 | |
| 
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (isNewLineWin && level == Level.RULE && seekingValue) {
 | |
|       // win newline
 | |
|       buffer = [];
 | |
|       isBufferEmpty = true;
 | |
|     } else if (isNewLineWin && buffer.length == 1) {
 | |
|       // ignore windows newline which is composed of two characters
 | |
|       buffer.pop();
 | |
|       isBufferEmpty = buffer.length === 0;
 | |
|     } else if (!isBufferEmpty || !isSpace && !isNewLineNix && !isNewLineWin && !isCarriageReturn) {
 | |
|       // any character
 | |
|       buffer.push(character);
 | |
|       isBufferEmpty = false;
 | |
|     }
 | |
| 
 | |
|     wasEscaped = isEscaped;
 | |
|     isEscaped = !wasEscaped && character == Marker.BACK_SLASH;
 | |
|     wasCommentStart = isCommentStart;
 | |
|     wasCommentEnd = isCommentEnd;
 | |
| 
 | |
|     position.line = (isNewLineWin || isNewLineNix || isCarriageReturn) ? position.line + 1 : position.line;
 | |
|     position.column = (isNewLineWin || isNewLineNix || isCarriageReturn) ? 0 : position.column + 1;
 | |
|   }
 | |
| 
 | |
|   if (seekingValue) {
 | |
|     externalContext.warnings.push('Missing \'}\' at ' + formatPosition([position.line, position.column, position.source]) + '.');
 | |
|   }
 | |
| 
 | |
|   if (seekingValue && buffer.length > 0) {
 | |
|     serializedBuffer = buffer.join('').trimRight().replace(TAIL_BROKEN_VALUE_PATTERN, '$1').trimRight();
 | |
|     propertyToken.push([
 | |
|       Token.PROPERTY_VALUE,
 | |
|       serializedBuffer,
 | |
|       [originalMetadata(metadata, serializedBuffer, externalContext)]
 | |
|     ]);
 | |
| 
 | |
|     buffer = [];
 | |
|   }
 | |
| 
 | |
|   if (buffer.length > 0) {
 | |
|     externalContext.warnings.push('Invalid character(s) \'' + buffer.join('') + '\' at ' + formatPosition(metadata) + '. Ignoring.');
 | |
|   }
 | |
| 
 | |
|   return allTokens;
 | |
| }
 | |
| 
 | |
| function isIgnoreStartComment(buffer) {
 | |
|   return IGNORE_START_COMMENT_PATTERN.test(buffer.join('') + Marker.FORWARD_SLASH);
 | |
| }
 | |
| 
 | |
| function isIgnoreEndComment(buffer) {
 | |
|   return IGNORE_END_COMMENT_PATTERN.test(buffer.join('') + Marker.FORWARD_SLASH);
 | |
| }
 | |
| 
 | |
| function originalMetadata(metadata, value, externalContext, selectorFallbacks) {
 | |
|   var source = metadata[2];
 | |
| 
 | |
|   return externalContext.inputSourceMapTracker.isTracking(source)
 | |
|     ? externalContext.inputSourceMapTracker.originalPositionFor(metadata, value.length, selectorFallbacks)
 | |
|     : metadata;
 | |
| }
 | |
| 
 | |
| function tokenTypeFrom(buffer) {
 | |
|   var isAtRule = buffer[0] == Marker.AT || buffer[0] == Marker.UNDERSCORE;
 | |
|   var ruleWord = buffer.join('').split(RULE_WORD_SEPARATOR_PATTERN)[0];
 | |
| 
 | |
|   if (isAtRule && BLOCK_RULES.indexOf(ruleWord) > -1) {
 | |
|     return Token.NESTED_BLOCK;
 | |
|   } if (isAtRule && AT_RULES.indexOf(ruleWord) > -1) {
 | |
|     return Token.AT_RULE;
 | |
|   } if (isAtRule) {
 | |
|     return Token.AT_RULE_BLOCK;
 | |
|   }
 | |
|   return Token.RULE;
 | |
| }
 | |
| 
 | |
| function tokenScopeFrom(tokenType) {
 | |
|   if (tokenType == Token.RULE) {
 | |
|     return Token.RULE_SCOPE;
 | |
|   } if (tokenType == Token.NESTED_BLOCK) {
 | |
|     return Token.NESTED_BLOCK_SCOPE;
 | |
|   } if (tokenType == Token.AT_RULE_BLOCK) {
 | |
|     return Token.AT_RULE_BLOCK_SCOPE;
 | |
|   }
 | |
| }
 | |
| 
 | |
| function isPageMarginBox(buffer) {
 | |
|   var serializedBuffer = buffer.join('').trim();
 | |
| 
 | |
|   return PAGE_MARGIN_BOXES.indexOf(serializedBuffer) > -1 || EXTRA_PAGE_BOXES.indexOf(serializedBuffer) > -1;
 | |
| }
 | |
| 
 | |
| function isRepeatToken(buffer) {
 | |
|   return REPEAT_PATTERN.test(buffer.join('') + Marker.CLOSE_SQUARE_BRACKET);
 | |
| }
 | |
| 
 | |
| module.exports = tokenize;
 |