work.suroh.tk/node_modules/gonzales-pe/lib/gonzales.js

22435 lines
548 KiB
JavaScript

(function webpackUniversalModuleDefinition(root, factory) {
if(typeof exports === 'object' && typeof module === 'object')
module.exports = factory();
else if(typeof define === 'function' && define.amd)
define([], factory);
else if(typeof exports === 'object')
exports["gonzales"] = factory();
else
root["gonzales"] = factory();
})(this, function() {
return /******/ (function(modules) { // webpackBootstrap
/******/ // The module cache
/******/ var installedModules = {};
/******/ // The require function
/******/ function __webpack_require__(moduleId) {
/******/ // Check if module is in cache
/******/ if(installedModules[moduleId])
/******/ return installedModules[moduleId].exports;
/******/ // Create a new module (and put it into the cache)
/******/ var module = installedModules[moduleId] = {
/******/ exports: {},
/******/ id: moduleId,
/******/ loaded: false
/******/ };
/******/ // Execute the module function
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
/******/ // Flag the module as loaded
/******/ module.loaded = true;
/******/ // Return the exports of the module
/******/ return module.exports;
/******/ }
/******/ // expose the modules object (__webpack_modules__)
/******/ __webpack_require__.m = modules;
/******/ // expose the module cache
/******/ __webpack_require__.c = installedModules;
/******/ // __webpack_public_path__
/******/ __webpack_require__.p = "";
/******/ // Load entry module and return exports
/******/ return __webpack_require__(0);
/******/ })
/************************************************************************/
/******/ ([
/* 0 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var Node = __webpack_require__(1);
var parse = __webpack_require__(7);
module.exports = {
createNode: function createNode(options) {
return new Node(options);
},
parse: parse
};
/***/ }),
/* 1 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
/**
* @param {string} type
* @param {array|string} content
* @param {number} line
* @param {number} column
* @constructor
*/
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
var Node = function () {
function Node(options) {
_classCallCheck(this, Node);
this.type = options.type;
this.content = options.content;
this.syntax = options.syntax;
if (options.start) this.start = options.start;
if (options.end) this.end = options.end;
}
/**
* @param {String} type Node type
* @return {Boolean} Whether there is a child node of given type
*/
Node.prototype.contains = function contains(type) {
if (!Array.isArray(this.content)) {
return false;
}
return this.content.some(function (node) {
return node.type === type;
});
};
/**
* @param {String} type Node type
* @param {Function} callback Function to call for every found node
*/
Node.prototype.eachFor = function eachFor(type, callback) {
if (!Array.isArray(this.content)) return;
if (typeof type !== 'string') {
callback = type;
type = null;
}
var l = this.content.length;
var breakLoop;
for (var i = l; i--;) {
if (breakLoop === null) break;
if (!type || this.content[i] && this.content[i].type === type) breakLoop = callback(this.content[i], i, this);
}
if (breakLoop === null) return null;
};
/**
* @param {String} type
* @return {?Node} First child node or `null` if nothing's been found.
*/
Node.prototype.first = function first(type) {
if (!Array.isArray(this.content)) return null;
if (!type) return this.content[0];
var i = 0;
var l = this.content.length;
for (; i < l; i++) {
if (this.content[i].type === type) return this.content[i];
}
return null;
};
/**
* @param {String} type Node type
* @param {Function} callback Function to call for every found node
*/
Node.prototype.forEach = function forEach(type, callback) {
if (!Array.isArray(this.content)) return;
if (typeof type !== 'string') {
callback = type;
type = null;
}
var i = 0;
var l = this.content.length;
var breakLoop;
for (; i < l; i++) {
if (breakLoop === null) break;
if (!type || this.content[i] && this.content[i].type === type) breakLoop = callback(this.content[i], i, this);
}
if (breakLoop === null) return null;
};
/**
* @param {Number} index
* @return {?Node}
*/
Node.prototype.get = function get(index) {
if (!Array.isArray(this.content)) return null;
var node = this.content[index];
return node ? node : null;
};
/**
* @param {Number} index
* @param {Node} node
*/
Node.prototype.insert = function insert(index, node) {
if (!Array.isArray(this.content)) return;
this.content.splice(index, 0, node);
};
/**
* @param {String} type
* @return {Boolean} Whether the node is of given type
*/
Node.prototype.is = function is(type) {
return this.type === type;
};
/**
* @param {String} type
* @return {?Node} Last child node or `null` if nothing's been found.
*/
Node.prototype.last = function last(type) {
if (!Array.isArray(this.content)) return null;
var i = this.content.length;
if (!type) return this.content[i - 1];
for (; i--;) {
if (this.content[i].type === type) return this.content[i];
}
return null;
};
/**
* Number of child nodes.
* @type {number}
*/
/**
* @param {Number} index
* @return {Node}
*/
Node.prototype.removeChild = function removeChild(index) {
if (!Array.isArray(this.content)) return;
var removedChild = this.content.splice(index, 1);
return removedChild;
};
Node.prototype.toJson = function toJson() {
return JSON.stringify(this, false, 2);
};
Node.prototype.toString = function toString() {
var stringify = void 0;
try {
stringify = __webpack_require__(2)("./" + this.syntax + '/stringify');
} catch (e) {
var message = 'Syntax "' + this.syntax + '" is not supported yet, sorry';
return console.error(message);
}
return stringify(this);
};
/**
* @param {Function} callback
*/
Node.prototype.traverse = function traverse(callback, index) {
var level = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 0;
var parent = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : null;
var breakLoop;
var x;
level++;
callback(this, index, parent, level);
if (!Array.isArray(this.content)) return;
for (var i = 0, l = this.content.length; i < l; i++) {
breakLoop = this.content[i].traverse(callback, i, level, this);
if (breakLoop === null) break;
// If some nodes were removed or added:
if (x = this.content.length - l) {
l += x;
i += x;
}
}
if (breakLoop === null) return null;
};
Node.prototype.traverseByType = function traverseByType(type, callback) {
this.traverse(function (node) {
if (node.type === type) callback.apply(node, arguments);
});
};
Node.prototype.traverseByTypes = function traverseByTypes(types, callback) {
this.traverse(function (node) {
if (types.indexOf(node.type) !== -1) callback.apply(node, arguments);
});
};
_createClass(Node, [{
key: 'length',
get: function get() {
if (!Array.isArray(this.content)) return 0;
return this.content.length;
}
}]);
return Node;
}();
module.exports = Node;
/***/ }),
/* 2 */
/***/ (function(module, exports, __webpack_require__) {
var map = {
"./css/stringify": 3,
"./less/stringify": 4,
"./sass/stringify": 5,
"./scss/stringify": 6
};
function webpackContext(req) {
return __webpack_require__(webpackContextResolve(req));
};
function webpackContextResolve(req) {
return map[req] || (function() { throw new Error("Cannot find module '" + req + "'.") }());
};
webpackContext.keys = function webpackContextKeys() {
return Object.keys(map);
};
webpackContext.resolve = webpackContextResolve;
module.exports = webpackContext;
webpackContext.id = 2;
/***/ }),
/* 3 */
/***/ (function(module, exports) {
'use strict';
module.exports = function stringify(tree) {
// TODO: Better error message
if (!tree) throw new Error('We need tree to translate');
function _t(tree) {
var type = tree.type;
if (_unique[type]) return _unique[type](tree);
if (typeof tree.content === 'string') return tree.content;
if (Array.isArray(tree.content)) return _composite(tree.content);
return '';
}
function _composite(t, i) {
if (!t) return '';
var s = '';
i = i || 0;
for (; i < t.length; i++) {
s += _t(t[i]);
}return s;
}
var _unique = {
'arguments': function _arguments(t) {
return '(' + _composite(t.content) + ')';
},
'atkeyword': function atkeyword(t) {
return '@' + _composite(t.content);
},
'attributeSelector': function attributeSelector(t) {
return '[' + _composite(t.content) + ']';
},
'block': function block(t) {
return '{' + _composite(t.content) + '}';
},
'brackets': function brackets(t) {
return '[' + _composite(t.content) + ']';
},
'class': function _class(t) {
return '.' + _composite(t.content);
},
'color': function color(t) {
return '#' + t.content;
},
'customProperty': function customProperty(t) {
return '--' + t.content;
},
'expression': function expression(t) {
return 'expression(' + t.content + ')';
},
'id': function id(t) {
return '#' + _composite(t.content);
},
'multilineComment': function multilineComment(t) {
return '/*' + t.content + '*/';
},
'nthSelector': function nthSelector(t) {
return ':' + _t(t.content[0]) + '(' + _composite(t.content.slice(1)) + ')';
},
'parentheses': function parentheses(t) {
return '(' + _composite(t.content) + ')';
},
'percentage': function percentage(t) {
return _composite(t.content) + '%';
},
'pseudoClass': function pseudoClass(t) {
return ':' + _composite(t.content);
},
'pseudoElement': function pseudoElement(t) {
return '::' + _composite(t.content);
},
'universalSelector': function universalSelector(t) {
return _composite(t.content) + '*';
},
'uri': function uri(t) {
return 'url(' + _composite(t.content) + ')';
}
};
return _t(tree);
};
/***/ }),
/* 4 */
/***/ (function(module, exports) {
'use strict';
module.exports = function stringify(tree) {
// TODO: Better error message
if (!tree) throw new Error('We need tree to translate');
function _t(tree) {
var type = tree.type;
if (_unique[type]) return _unique[type](tree);
if (typeof tree.content === 'string') return tree.content;
if (Array.isArray(tree.content)) return _composite(tree.content);
return '';
}
function _composite(t, i) {
if (!t) return '';
var s = '';
i = i || 0;
for (; i < t.length; i++) {
s += _t(t[i]);
}return s;
}
var _unique = {
'arguments': function _arguments(t) {
return '(' + _composite(t.content) + ')';
},
'atkeyword': function atkeyword(t) {
return '@' + _composite(t.content);
},
'attributeSelector': function attributeSelector(t) {
return '[' + _composite(t.content) + ']';
},
'block': function block(t) {
return '{' + _composite(t.content) + '}';
},
'brackets': function brackets(t) {
return '[' + _composite(t.content) + ']';
},
'class': function _class(t) {
return '.' + _composite(t.content);
},
'color': function color(t) {
return '#' + t.content;
},
'escapedString': function escapedString(t) {
return '~' + t.content;
},
'expression': function expression(t) {
return 'expression(' + t.content + ')';
},
'id': function id(t) {
return '#' + _composite(t.content);
},
'interpolatedVariable': function interpolatedVariable(t) {
return '@{' + _composite(t.content) + '}';
},
'multilineComment': function multilineComment(t) {
return '/*' + t.content + '*/';
},
'nthSelector': function nthSelector(t) {
return ':' + _t(t.content[0]) + '(' + _composite(t.content.slice(1)) + ')';
},
'parentheses': function parentheses(t) {
return '(' + _composite(t.content) + ')';
},
'percentage': function percentage(t) {
return _composite(t.content) + '%';
},
'pseudoClass': function pseudoClass(t) {
return ':' + _composite(t.content);
},
'pseudoElement': function pseudoElement(t) {
return '::' + _composite(t.content);
},
'singlelineComment': function singlelineComment(t) {
return '/' + '/' + t.content;
},
'universalSelector': function universalSelector(t) {
return _composite(t.content) + '*';
},
'uri': function uri(t) {
return 'url(' + _composite(t.content) + ')';
},
'variable': function variable(t) {
return '@' + _composite(t.content);
},
'variablesList': function variablesList(t) {
return _composite(t.content) + '...';
}
};
return _t(tree);
};
/***/ }),
/* 5 */
/***/ (function(module, exports) {
'use strict';
module.exports = function stringify(tree) {
// TODO: Better error message
if (!tree) throw new Error('We need tree to translate');
function _t(tree) {
var type = tree.type;
if (_unique[type]) return _unique[type](tree);
if (typeof tree.content === 'string') return tree.content;
if (Array.isArray(tree.content)) return _composite(tree.content);
return '';
}
function _composite(t, i) {
if (!t) return '';
var s = '';
i = i || 0;
for (; i < t.length; i++) {
s += _t(t[i]);
}return s;
}
var _unique = {
'arguments': function _arguments(t) {
return '(' + _composite(t.content) + ')';
},
'atkeyword': function atkeyword(t) {
return '@' + _composite(t.content);
},
'attributeSelector': function attributeSelector(t) {
return '[' + _composite(t.content) + ']';
},
'block': function block(t) {
return _composite(t.content);
},
'brackets': function brackets(t) {
return '[' + _composite(t.content) + ']';
},
'class': function _class(t) {
return '.' + _composite(t.content);
},
'color': function color(t) {
return '#' + t.content;
},
'customProperty': function customProperty(t) {
return '--' + t.content;
},
'expression': function expression(t) {
return 'expression(' + t.content + ')';
},
'functionsList': function functionsList(t) {
return _composite(t.content) + '...';
},
'id': function id(t) {
return '#' + _composite(t.content);
},
'interpolation': function interpolation(t) {
return '#{' + _composite(t.content) + '}';
},
'multilineComment': function multilineComment(t) {
var lines = t.content.split('\n');
var close = '';
if (lines.length > 1) {
var lastLine = lines[lines.length - 1];
if (lastLine.length < t.end.column) {
close = '*/';
}
} else if (t.content.length + 4 === t.end.column - t.start.column + 1) {
close = '*/';
}
return '/*' + t.content + close;
},
'nthSelector': function nthSelector(t) {
return ':' + _t(t.content[0]) + '(' + _composite(t.content.slice(1)) + ')';
},
'parentheses': function parentheses(t) {
return '(' + _composite(t.content) + ')';
},
'percentage': function percentage(t) {
return _composite(t.content) + '%';
},
'placeholder': function placeholder(t) {
return '%' + _composite(t.content);
},
'pseudoClass': function pseudoClass(t) {
return ':' + _composite(t.content);
},
'pseudoElement': function pseudoElement(t) {
return '::' + _composite(t.content);
},
'singlelineComment': function singlelineComment(t) {
return '/' + '/' + t.content;
},
'universalSelector': function universalSelector(t) {
return _composite(t.content) + '*';
},
'uri': function uri(t) {
return 'url(' + _composite(t.content) + ')';
},
'variable': function variable(t) {
return '$' + _composite(t.content);
},
'variablesList': function variablesList(t) {
return _composite(t.content) + '...';
}
};
return _t(tree);
};
/***/ }),
/* 6 */
/***/ (function(module, exports) {
'use strict';
module.exports = function stringify(tree) {
// TODO: Better error message
if (!tree) throw new Error('We need tree to translate');
function _t(tree) {
var type = tree.type;
if (_unique[type]) return _unique[type](tree);
if (typeof tree.content === 'string') return tree.content;
if (Array.isArray(tree.content)) return _composite(tree.content);
return '';
}
function _composite(t, i) {
if (!t) return '';
var s = '';
i = i || 0;
for (; i < t.length; i++) {
s += _t(t[i]);
}return s;
}
var _unique = {
'arguments': function _arguments(t) {
return '(' + _composite(t.content) + ')';
},
'atkeyword': function atkeyword(t) {
return '@' + _composite(t.content);
},
'attributeSelector': function attributeSelector(t) {
return '[' + _composite(t.content) + ']';
},
'block': function block(t) {
return '{' + _composite(t.content) + '}';
},
'brackets': function brackets(t) {
return '[' + _composite(t.content) + ']';
},
'class': function _class(t) {
return '.' + _composite(t.content);
},
'color': function color(t) {
return '#' + t.content;
},
'customProperty': function customProperty(t) {
return '--' + t.content;
},
'expression': function expression(t) {
return 'expression(' + t.content + ')';
},
'functionsList': function functionsList(t) {
return _composite(t.content) + '...';
},
'id': function id(t) {
return '#' + _composite(t.content);
},
'interpolation': function interpolation(t) {
return '#{' + _composite(t.content) + '}';
},
'multilineComment': function multilineComment(t) {
return '/*' + t.content + '*/';
},
'nthSelector': function nthSelector(t) {
return ':' + _t(t.content[0]) + '(' + _composite(t.content.slice(1)) + ')';
},
'parentheses': function parentheses(t) {
return '(' + _composite(t.content) + ')';
},
'percentage': function percentage(t) {
return _composite(t.content) + '%';
},
'placeholder': function placeholder(t) {
return '%' + _composite(t.content);
},
'pseudoClass': function pseudoClass(t) {
return ':' + _composite(t.content);
},
'pseudoElement': function pseudoElement(t) {
return '::' + _composite(t.content);
},
'singlelineComment': function singlelineComment(t) {
return '/' + '/' + t.content;
},
'universalSelector': function universalSelector(t) {
return _composite(t.content) + '*';
},
'uri': function uri(t) {
return 'url(' + _composite(t.content) + ')';
},
'variable': function variable(t) {
return '$' + _composite(t.content);
},
'variablesList': function variablesList(t) {
return _composite(t.content) + '...';
}
};
return _t(tree);
};
/***/ }),
/* 7 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var ParsingError = __webpack_require__(8);
var syntaxes = __webpack_require__(10);
var isInteger = Number.isInteger || function (value) {
return typeof value === 'number' && Math.floor(value) === value;
};
/**
* @param {String} css
* @param {Object} options
* @return {Object} AST
*/
function parser(css, options) {
if (typeof css !== 'string') throw new Error('Please, pass a string to parse');else if (!css) return __webpack_require__(29)();
var syntax = options && options.syntax || 'css';
var context = options && options.context || 'stylesheet';
var tabSize = options && options.tabSize;
if (!isInteger(tabSize) || tabSize < 1) tabSize = 1;
var syntaxParser = syntaxes[syntax];
if (!syntaxParser) {
var message = 'Syntax "' + syntax + '" is not supported yet, sorry';
return console.error(message);
}
var getTokens = syntaxParser.tokenizer;
var mark = syntaxParser.mark;
var parse = syntaxParser.parse;
var tokens = getTokens(css, tabSize);
mark(tokens);
var ast;
try {
ast = parse(tokens, context);
} catch (e) {
if (!e.syntax) throw e;
throw new ParsingError(e, css);
}
return ast;
}
module.exports = parser;
/***/ }),
/* 8 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var parserPackage = __webpack_require__(9);
/**
* @param {Error} e
* @param {String} css
*/
function ParsingError(e, css) {
this.line = e.line;
this.syntax = e.syntax;
this.css_ = css;
}
ParsingError.prototype = {
/**
* @type {String}
* @private
*/
customMessage_: '',
/**
* @type {Number}
*/
line: null,
/**
* @type {String}
*/
name: 'Parsing error',
/**
* @type {String}
*/
syntax: null,
/**
* @type {String}
*/
version: parserPackage.version,
/**
* @type {String}
*/
get context() {
var LINES_AROUND = 2;
var result = [];
var currentLineNumber = this.line;
var start = currentLineNumber - 1 - LINES_AROUND;
var end = currentLineNumber + LINES_AROUND;
var lines = this.css_.split(/\r\n|\r|\n/);
for (var i = start; i < end; i++) {
var line = lines[i];
if (!line) continue;
var ln = i + 1;
var mark = ln === currentLineNumber ? '*' : ' ';
result.push(ln + mark + '| ' + line);
}
return result.join('\n');
},
/**
* @type {String}
*/
get message() {
if (this.customMessage_) {
return this.customMessage_;
} else {
var message = 'Please check validity of the block';
if (typeof this.line === 'number') message += ' starting from line #' + this.line;
return message;
}
},
set message(message) {
this.customMessage_ = message;
},
/**
* @return {String}
*/
toString: function toString() {
return [this.name + ': ' + this.message, '', this.context, '', 'Syntax: ' + this.syntax, 'Gonzales PE version: ' + this.version].join('\n');
}
};
module.exports = ParsingError;
/***/ }),
/* 9 */
/***/ (function(module, exports) {
module.exports = {"name":"gonzales-pe","description":"Gonzales Preprocessor Edition (fast CSS parser)","version":"4.2.4","homepage":"http://github.com/tonyganch/gonzales-pe","bugs":"http://github.com/tonyganch/gonzales-pe/issues","license":"MIT","author":{"name":"Tony Ganch","email":"tonyganch+github@gmail.com","url":"http://tonyganch.com"},"main":"./lib/gonzales","repository":{"type":"git","url":"http://github.com/tonyganch/gonzales-pe.git"},"scripts":{"autofix-tests":"bash ./scripts/build.sh && bash ./scripts/autofix-tests.sh","build":"bash ./scripts/build.sh","init":"bash ./scripts/init.sh","lint":"bash ./scripts/lint.sh","log":"bash ./scripts/log.sh","prepublishOnly":"bash ./scripts/build.sh","test":"bash ./scripts/test.sh","watch":"bash ./scripts/watch.sh"},"bin":{"gonzales":"./bin/gonzales.js"},"dependencies":{"minimist":"1.1.x"},"devDependencies":{"babel-core":"^6.18.2","babel-loader":"^6.2.7","babel-plugin-add-module-exports":"^0.2.1","babel-preset-es2015":"^6.18.0","coffee-script":"~1.7.1","eslint":"^3.0.0","jscs":"2.1.0","jshint":"2.10.2","json-loader":"^0.5.3","mocha":"2.2.x","webpack":"^1.12.2","webpack-closure-compiler":"^2.0.2"},"engines":{"node":">=0.6.0"},"files":["MIT-LICENSE.txt","bin/gonzales.js","lib/gonzales.js"]}
/***/ }),
/* 10 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
module.exports = {
css: __webpack_require__(11),
less: __webpack_require__(17),
sass: __webpack_require__(21),
scss: __webpack_require__(25)
};
/***/ }),
/* 11 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
exports.__esModule = true;
exports.default = {
mark: __webpack_require__(12),
parse: __webpack_require__(14),
stringify: __webpack_require__(3),
tokenizer: __webpack_require__(16)
};
module.exports = exports['default'];
/***/ }),
/* 12 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var TokenType = __webpack_require__(13);
/**
* Mark whitespaces and comments
* @param {Array} tokens
*/
function markSpacesAndComments(tokens) {
var tokensLength = tokens.length;
var spaces = [-1, -1];
var type; // Current token's type
// For every token in the token list, mark spaces and line breaks
// as spaces (set both `ws` and `sc` flags). Mark multiline comments
// with `sc` flag.
// If there are several spaces or tabs or line breaks or multiline
// comments in a row, group them: take the last one's index number
// and save it to the first token in the group as a reference:
// e.g., `ws_last = 7` for a group of whitespaces or `sc_last = 9`
// for a group of whitespaces and comments.
for (var i = 0; i < tokensLength; i++) {
type = tokens[i].type;
if (type === TokenType.Space || type === TokenType.Tab || type === TokenType.Newline) {
markSpace(tokens, i, spaces);
} else if (type === TokenType.CommentML) {
markComment(tokens, i, spaces);
} else {
markEndOfSpacesAndComments(tokens, i, spaces);
}
}
markEndOfSpacesAndComments(tokens, i, spaces);
}
function markSpace(tokens, i, spaces) {
var token = tokens[i];
token.ws = true;
token.sc = true;
if (spaces[0] === -1) spaces[0] = i;
if (spaces[1] === -1) spaces[1] = i;
}
function markComment(tokens, i, spaces) {
var ws = spaces[0];
tokens[i].sc = true;
if (ws !== -1) {
tokens[ws].ws_last = i - 1;
spaces[0] = -1;
}
}
function markEndOfSpacesAndComments(tokens, i, spaces) {
var ws = spaces[0];
var sc = spaces[1];
if (ws !== -1) {
tokens[ws].ws_last = i - 1;
spaces[0] = -1;
}
if (sc !== -1) {
tokens[sc].sc_last = i - 1;
spaces[1] = -1;
}
}
/**
* Pair brackets
* @param {Array} tokens
*/
function markBrackets(tokens) {
var tokensLength = tokens.length;
var ps = []; // Parentheses
var sbs = []; // Square brackets
var cbs = []; // Curly brackets
var t = void 0; // Current token
// For every token in the token list, if we meet an opening (left)
// bracket, push its index number to a corresponding array.
// If we then meet a closing (right) bracket, look at the corresponding
// array. If there are any elements (records about previously met
// left brackets), take a token of the last left bracket (take
// the last index number from the array and find a token with
// this index number) and save right bracket's index as a reference:
for (var i = 0; i < tokensLength; i++) {
t = tokens[i];
var type = t.type;
if (type === TokenType.LeftParenthesis) {
ps.push(i);
} else if (type === TokenType.RightParenthesis) {
if (ps.length) {
t.left = ps.pop();
tokens[t.left].right = i;
}
} else if (type === TokenType.LeftSquareBracket) {
sbs.push(i);
} else if (type === TokenType.RightSquareBracket) {
if (sbs.length) {
t.left = sbs.pop();
tokens[t.left].right = i;
}
} else if (type === TokenType.LeftCurlyBracket) {
cbs.push(i);
} else if (type === TokenType.RightCurlyBracket) {
if (cbs.length) {
t.left = cbs.pop();
tokens[t.left].right = i;
}
}
}
}
/**
* @param {Array} tokens
*/
function markTokens(tokens) {
// Mark paired brackets:
markBrackets(tokens);
// Mark whitespaces and comments:
markSpacesAndComments(tokens);
}
module.exports = markTokens;
/***/ }),
/* 13 */
/***/ (function(module, exports) {
// jscs:disable
'use strict';
module.exports = {
StringSQ: 'StringSQ',
StringDQ: 'StringDQ',
CommentML: 'CommentML',
CommentSL: 'CommentSL',
Newline: 'Newline',
Space: 'Space',
Tab: 'Tab',
ExclamationMark: 'ExclamationMark', // !
QuotationMark: 'QuotationMark', // "
NumberSign: 'NumberSign', // #
DollarSign: 'DollarSign', // $
PercentSign: 'PercentSign', // %
Ampersand: 'Ampersand', // &
Apostrophe: 'Apostrophe', // '
LeftParenthesis: 'LeftParenthesis', // (
RightParenthesis: 'RightParenthesis', // )
Asterisk: 'Asterisk', // *
PlusSign: 'PlusSign', // +
Comma: 'Comma', // ,
HyphenMinus: 'HyphenMinus', // -
FullStop: 'FullStop', // .
Solidus: 'Solidus', // /
Colon: 'Colon', // :
Semicolon: 'Semicolon', // ;
LessThanSign: 'LessThanSign', // <
EqualsSign: 'EqualsSign', // =
EqualitySign: 'EqualitySign', // ==
InequalitySign: 'InequalitySign', // !=
GreaterThanSign: 'GreaterThanSign', // >
QuestionMark: 'QuestionMark', // ?
CommercialAt: 'CommercialAt', // @
LeftSquareBracket: 'LeftSquareBracket', // [
ReverseSolidus: 'ReverseSolidus', // \
RightSquareBracket: 'RightSquareBracket', // ]
CircumflexAccent: 'CircumflexAccent', // ^
LowLine: 'LowLine', // _
LeftCurlyBracket: 'LeftCurlyBracket', // {
VerticalLine: 'VerticalLine', // |
RightCurlyBracket: 'RightCurlyBracket', // }
Tilde: 'Tilde', // ~
Identifier: 'Identifier',
DecimalNumber: 'DecimalNumber'
};
/***/ }),
/* 14 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var Node = __webpack_require__(1);
var NodeType = __webpack_require__(15);
var TokenType = __webpack_require__(13);
/**
* @type {Array}
*/
var tokens = void 0;
/**
* @type {Number}
*/
var tokensLength = void 0;
/**
* @type {Number}
*/
var pos = void 0;
var contexts = {
'atkeyword': function atkeyword() {
return checkAtkeyword(pos) && getAtkeyword();
},
'atrule': function atrule() {
return checkAtrule(pos) && getAtrule();
},
'attributeSelector': function attributeSelector() {
return checkAttributeSelector(pos) && getAttributeSelector();
},
'block': function block() {
return checkBlock(pos) && getBlock();
},
'brackets': function brackets() {
return checkBrackets(pos) && getBrackets();
},
'class': function _class() {
return checkClass(pos) && getClass();
},
'combinator': function combinator() {
return checkCombinator(pos) && getCombinator();
},
'commentML': function commentML() {
return checkCommentML(pos) && getCommentML();
},
'declaration': function declaration() {
return checkDeclaration(pos) && getDeclaration();
},
'declDelim': function declDelim() {
return checkDeclDelim(pos) && getDeclDelim();
},
'delim': function delim() {
return checkDelim(pos) && getDelim();
},
'dimension': function dimension() {
return checkDimension(pos) && getDimension();
},
'expression': function expression() {
return checkExpression(pos) && getExpression();
},
'function': function _function() {
return checkFunction(pos) && getFunction();
},
'ident': function ident() {
return checkIdent(pos) && getIdent();
},
'important': function important() {
return checkImportant(pos) && getImportant();
},
'namespace': function namespace() {
return checkNamespace(pos) && getNamespace();
},
'number': function number() {
return checkNumber(pos) && getNumber();
},
'operator': function operator() {
return checkOperator(pos) && getOperator();
},
'parentheses': function parentheses() {
return checkParentheses(pos) && getParentheses();
},
'percentage': function percentage() {
return checkPercentage(pos) && getPercentage();
},
'progid': function progid() {
return checkProgid(pos) && getProgid();
},
'property': function property() {
return checkProperty(pos) && getProperty();
},
'propertyDelim': function propertyDelim() {
return checkPropertyDelim(pos) && getPropertyDelim();
},
'pseudoc': function pseudoc() {
return checkPseudoc(pos) && getPseudoc();
},
'pseudoe': function pseudoe() {
return checkPseudoe(pos) && getPseudoe();
},
'ruleset': function ruleset() {
return checkRuleset(pos) && getRuleset();
},
's': function s() {
return checkS(pos) && getS();
},
'selector': function selector() {
return checkSelector(pos) && getSelector();
},
'shash': function shash() {
return checkShash(pos) && getShash();
},
'string': function string() {
return checkString(pos) && getString();
},
'stylesheet': function stylesheet() {
return checkStylesheet(pos) && getStylesheet();
},
'unary': function unary() {
return checkUnary(pos) && getUnary();
},
'unicodeRange': function unicodeRange() {
return checkUnicodeRange(pos) && getUnicodeRange();
},
'universalSelector': function universalSelector() {
return checkUniversalSelector(pos) && getUniversalSelector();
},
'urange': function urange() {
return checkUrange(pos) && getUrange();
},
'uri': function uri() {
return checkUri(pos) && getUri();
},
'value': function value() {
return checkValue(pos) && getValue();
},
'vhash': function vhash() {
return checkVhash(pos) && getVhash();
}
};
/**
* Stop parsing and display error
* @param {Number=} i Token's index number
*/
function throwError(i) {
var ln = tokens[i].ln;
throw { line: ln, syntax: 'css' };
}
/**
* @param {Object} exclude
* @param {Number} i Token's index number
* @return {Number}
*/
function checkExcluding(exclude, i) {
var start = i;
while (i < tokensLength) {
if (exclude[tokens[i++].type]) break;
}
return i - start - 2;
}
/**
* @param {Number} start
* @param {Number} finish
* @return {String}
*/
function joinValues(start, finish) {
var s = '';
for (var i = start; i < finish + 1; i++) {
s += tokens[i].value;
}
return s;
}
/**
* @param {Number} start
* @param {Number} num
* @return {String}
*/
function joinValues2(start, num) {
if (start + num - 1 >= tokensLength) return;
var s = '';
for (var i = 0; i < num; i++) {
s += tokens[start + i].value;
}
return s;
}
function getLastPosition(content, line, column, colOffset) {
return typeof content === 'string' ? getLastPositionForString(content, line, column, colOffset) : getLastPositionForArray(content, line, column, colOffset);
}
function getLastPositionForString(content, line, column, colOffset) {
var position = [];
if (!content) {
position = [line, column];
if (colOffset) position[1] += colOffset - 1;
return position;
}
var lastLinebreak = content.lastIndexOf('\n');
var endsWithLinebreak = lastLinebreak === content.length - 1;
var splitContent = content.split('\n');
var linebreaksCount = splitContent.length - 1;
var prevLinebreak = linebreaksCount === 0 || linebreaksCount === 1 ? -1 : content.length - splitContent[linebreaksCount - 1].length - 2;
// Line:
var offset = endsWithLinebreak ? linebreaksCount - 1 : linebreaksCount;
position[0] = line + offset;
// Column:
if (endsWithLinebreak) {
offset = prevLinebreak !== -1 ? content.length - prevLinebreak : content.length - 1;
} else {
offset = linebreaksCount !== 0 ? content.length - lastLinebreak - column - 1 : content.length - 1;
}
position[1] = column + offset;
if (!colOffset) return position;
if (endsWithLinebreak) {
position[0]++;
position[1] = colOffset;
} else {
position[1] += colOffset;
}
return position;
}
function getLastPositionForArray(content, line, column, colOffset) {
var position = void 0;
if (content.length === 0) {
position = [line, column];
} else {
var c = content[content.length - 1];
if (c.hasOwnProperty('end')) {
position = [c.end.line, c.end.column];
} else {
position = getLastPosition(c.content, line, column);
}
}
if (!colOffset) return position;
if (tokens[pos - 1] && tokens[pos - 1].type !== 'Newline') {
position[1] += colOffset;
} else {
position[0]++;
position[1] = 1;
}
return position;
}
function newNode(type, content, line, column, end) {
if (!end) end = getLastPosition(content, line, column);
return new Node({
type: type,
content: content,
start: {
line: line,
column: column
},
end: {
line: end[0],
column: end[1]
},
syntax: 'css'
});
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkAny(i) {
var l = void 0;
if (l = checkBrackets(i)) tokens[i].any_child = 1;else if (l = checkParentheses(i)) tokens[i].any_child = 2;else if (l = checkString(i)) tokens[i].any_child = 3;else if (l = checkPercentage(i)) tokens[i].any_child = 4;else if (l = checkDimension(i)) tokens[i].any_child = 5;else if (l = checkUnicodeRange(i)) tokens[i].any_child = 13;else if (l = checkNumber(i)) tokens[i].any_child = 6;else if (l = checkUri(i)) tokens[i].any_child = 7;else if (l = checkExpression(i)) tokens[i].any_child = 8;else if (l = checkFunction(i)) tokens[i].any_child = 9;else if (l = checkIdent(i)) tokens[i].any_child = 10;else if (l = checkClass(i)) tokens[i].any_child = 11;else if (l = checkUnary(i)) tokens[i].any_child = 12;
return l;
}
/**
* @return {Node}
*/
function getAny() {
var childType = tokens[pos].any_child;
if (childType === 1) return getBrackets();
if (childType === 2) return getParentheses();
if (childType === 3) return getString();
if (childType === 4) return getPercentage();
if (childType === 5) return getDimension();
if (childType === 13) return getUnicodeRange();
if (childType === 6) return getNumber();
if (childType === 7) return getUri();
if (childType === 8) return getExpression();
if (childType === 9) return getFunction();
if (childType === 10) return getIdent();
if (childType === 11) return getClass();
if (childType === 12) return getUnary();
}
/**
* @return {Node}
*/
function getArguments() {
var type = NodeType.ArgumentsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var body = void 0;
// Skip `(`.
pos++;
while (pos < tokensLength && tokens[pos].type !== TokenType.RightParenthesis) {
if (checkDeclaration(pos)) content.push(getDeclaration());else if (checkArgument(pos)) {
body = getArgument();
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
} else if (checkClass(pos)) content.push(getClass());else throwError(pos);
}
var end = getLastPosition(content, line, column, 1);
// Skip `)`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkArgument(i) {
var l = void 0;
if (l = checkVhash(i)) tokens[i].argument_child = 1;else if (l = checkCustomProperty(i)) tokens[i].argument_child = 2;else if (l = checkAny(i)) tokens[i].argument_child = 3;else if (l = checkSC(i)) tokens[i].argument_child = 4;else if (l = checkOperator(i)) tokens[i].argument_child = 5;
return l;
}
/**
* @return {Node}
*/
function getArgument() {
var childType = tokens[pos].argument_child;
if (childType === 1) return getVhash();
if (childType === 2) return getCustomProperty();
if (childType === 3) return getAny();
if (childType === 4) return getSC();
if (childType === 5) return getOperator();
}
/**
* Check if token is part of an @-word (e.g. `@import`, `@include`)
* @param {Number} i Token's index number
* @return {Number}
*/
function checkAtkeyword(i) {
var l = void 0;
// Check that token is `@`:
if (i >= tokensLength || tokens[i++].type !== TokenType.CommercialAt) return 0;
return (l = checkIdent(i)) ? l + 1 : 0;
}
/**
* Get node with @-word
* @return {Node}
*/
function getAtkeyword() {
var type = NodeType.AtkeywordType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `@`.
pos++;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* Check if token is a part of an @-rule
* @param {Number} i Token's index number
* @return {Number} Length of @-rule
*/
function checkAtrule(i) {
var l = void 0;
if (i >= tokensLength) return 0;
// If token already has a record of being part of an @-rule,
// return the @-rule's length:
if (tokens[i].atrule_l !== undefined) return tokens[i].atrule_l;
// If token is part of an @-rule, save the rule's type to token.
// @keyframes:
if (l = checkKeyframesRule(i)) tokens[i].atrule_type = 4;
// @-rule with ruleset:
else if (l = checkAtruler(i)) tokens[i].atrule_type = 1;
// Block @-rule:
else if (l = checkAtruleb(i)) tokens[i].atrule_type = 2;
// Single-line @-rule:
else if (l = checkAtrules(i)) tokens[i].atrule_type = 3;else return 0;
// If token is part of an @-rule, save the rule's length to token:
tokens[i].atrule_l = l;
return l;
}
/**
* Get node with @-rule
* @return {Node}
*/
function getAtrule() {
var childType = tokens[pos].atrule_type;
if (childType === 1) return getAtruler(); // @-rule with ruleset
if (childType === 2) return getAtruleb(); // Block @-rule
if (childType === 3) return getAtrules(); // Single-line @-rule
if (childType === 4) return getKeyframesRule();
}
/**
* Check if token is part of a block @-rule
* @param {Number} i Token's index number
* @return {Number} Length of the @-rule
*/
function checkAtruleb(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a block @-rule
* @return {Node}
*/
function getAtruleb() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an @-rule with ruleset
* @param {Number} i Token's index number
* @return {Number} Length of the @-rule
*/
function checkAtruler(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
if (l = checkAtrulers(i)) i += l;
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
return i - start;
}
/**
* Get node with an @-rule with ruleset
* @return {Node}
*/
function getAtruler() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets(), getAtrulers());
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkAtrulers(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkSC(i)) i += l;
while (i < tokensLength) {
if (l = checkSC(i)) tokens[i].atrulers_child = 1;else if (l = checkAtrule(i)) tokens[i].atrulers_child = 2;else if (l = checkRuleset(i)) tokens[i].atrulers_child = 3;else break;
i += l;
}
if (i < tokensLength) tokens[i].atrulers_end = 1;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* @return {Node}
*/
function getAtrulers() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `{`.
pos++;
content = content.concat(getSC());
while (pos < tokensLength && !tokens[pos].atrulers_end) {
var childType = tokens[pos].atrulers_child;
if (childType === 1) content = content.concat(getSC());else if (childType === 2) content.push(getAtrule());else if (childType === 3) content.push(getRuleset());else break;
}
content = content.concat(getSC());
var end = getLastPosition(content, line, column, 1);
// Skip `}`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkAtrules(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
return i - start;
}
/**
* @return {Node}
*/
function getAtrules() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a block (e.g. `{...}`).
* @param {Number} i Token's index number
* @return {Number} Length of the block
*/
function checkBlock(i) {
return i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket ? tokens[i].right - i + 1 : 0;
}
/**
* Get node with a block
* @return {Node}
*/
function getBlock() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = tokens[pos].right;
var content = [];
// Skip `{`.
pos++;
while (pos < end) {
if (checkBlockdecl(pos)) content = content.concat(getBlockdecl());else throwError(pos);
}
var end_ = getLastPosition(content, line, column, 1);
pos = end + 1;
return newNode(type, content, line, column, end_);
}
/**
* Check if token is part of a declaration (property-value pair)
* @param {Number} i Token's index number
* @return {Number} Length of the declaration
*/
function checkBlockdecl(i) {
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkBlockdecl1(i)) tokens[i].bd_type = 1;else if (l = checkBlockdecl2(i)) tokens[i].bd_type = 2;else if (l = checkBlockdecl3(i)) tokens[i].bd_type = 3;else if (l = checkBlockdecl4(i)) tokens[i].bd_type = 4;else return 0;
return l;
}
/**
* @return {Array}
*/
function getBlockdecl() {
var childType = tokens[pos].bd_type;
if (childType === 1) return getBlockdecl1();
if (childType === 2) return getBlockdecl2();
if (childType === 3) return getBlockdecl3();
if (childType === 4) return getBlockdecl4();
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkBlockdecl1(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkDeclaration(i)) tokens[i].bd_kind = 1;else if (l = checkAtrule(i)) tokens[i].bd_kind = 2;else return 0;
i += l;
if (l = checkSC(i)) i += l;
if (i < tokensLength && (l = checkDeclDelim(i))) i += l;else return 0;
if (l = checkSC(i)) i += l;else return 0;
return i - start;
}
/**
* @return {Array}
*/
function getBlockdecl1() {
var sc = getSC();
var content = void 0;
switch (tokens[pos].bd_kind) {
case 1:
content = getDeclaration();
break;
case 2:
content = getAtrule();
break;
}
return sc.concat(content, getSC(), getDeclDelim(), getSC());
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkBlockdecl2(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkDeclaration(i)) tokens[i].bd_kind = 1;else if (l = checkAtrule(i)) tokens[i].bd_kind = 2;else return 0;
i += l;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* @return {Array}
*/
function getBlockdecl2() {
var sc = getSC();
var content = void 0;
switch (tokens[pos].bd_kind) {
case 1:
content = getDeclaration();
break;
case 2:
content = getAtrule();
break;
}
return sc.concat(content, getSC());
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkBlockdecl3(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkDeclDelim(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* @return {Array}
*/
function getBlockdecl3() {
return [].concat(getSC(), getDeclDelim(), getSC());
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkBlockdecl4(i) {
return checkSC(i);
}
/**
* @return {Array}
*/
function getBlockdecl4() {
return getSC();
}
/**
* Check if token is part of text inside square brackets, e.g. `[1]`
* @param {Number} i Token's index number
* @return {Number}
*/
function checkBrackets(i) {
if (i >= tokensLength) return 0;
var start = i;
// Skip `[`.
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
if (i < tokens[start].right) {
var l = checkTsets(i);
if (l) i += l;else return 0;
}
// Skip `]`.
i++;
return i - start;
}
/**
* Get node with text inside square brackets, e.g. `[1]`
* @return {Node}
*/
function getBrackets() {
var type = NodeType.BracketsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var right = token.right;
var content = [];
// Skip `[`.
pos++;
if (pos < right) {
content = getTsets();
}
var end = getLastPosition(content, line, column, 1);
// Skip `]`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a class selector (e.g. `.abc`)
* @param {Number} i Token's index number
* @return {Number} Length of the class selector
*/
function checkClass(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].class_l) return tokens[i].class_l;
// Skip `.`.
if (tokens[i].type === TokenType.FullStop) i++;else return 0;
if (l = checkIdent(i)) {
tokens[start].class_l = l + 1;
i += l;
} else return 0;
tokens[start].classEnd = i;
return i - start;
}
/**
* Get node with a class selector
* @return {Node}
*/
function getClass() {
var type = NodeType.ClassType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `.`
pos++;
var content = [getIdent()];
return newNode(type, content, line, column);
}
function checkCombinator(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkCombinator1(i)) tokens[i].combinatorType = 1;else if (l = checkCombinator2(i)) tokens[i].combinatorType = 2;else if (l = checkCombinator3(i)) tokens[i].combinatorType = 3;else if (l = checkCombinator4(i)) tokens[i].combinatorType = 4;
return l;
}
function getCombinator() {
var type = tokens[pos].combinatorType;
if (type === 1) return getCombinator1();
if (type === 2) return getCombinator2();
if (type === 3) return getCombinator3();
if (type === 4) return getCombinator4();
}
/**
* (1) `>>>`
*
* @param {Number} i
* @return {Number}
*/
function checkCombinator1(i) {
if (i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign) return 3;
return 0;
}
/**
* @return {Node}
*/
function getCombinator1() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '>>>';
// Skip combinator
pos += 3;
return newNode(type, content, line, column);
}
/**
* (1) `||`
* (2) `>>`
*
* @param {Number} i
* @return {Number}
*/
function checkCombinator2(i) {
if (i + 1 >= tokensLength) return 0;
if (tokens[i].type === TokenType.VerticalLine && tokens[i + 1].type === TokenType.VerticalLine) return 2;
if (tokens[i].type === TokenType.GreaterThanSign && tokens[i + 1].type === TokenType.GreaterThanSign) return 2;
return 0;
}
/**
* @return {Node}
*/
function getCombinator2() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '' + token.value + tokens[pos + 1].value;
// Skip combinator
pos += 2;
return newNode(type, content, line, column);
}
/**
* (1) `>`
* (2) `+`
* (3) `~`
*
* @param {Number} i
* @return {Number}
*/
function checkCombinator3(i) {
var type = tokens[i].type;
if (type === TokenType.PlusSign || type === TokenType.GreaterThanSign || type === TokenType.Tilde) return 1;else return 0;
}
/**
* @return {Node}
*/
function getCombinator3() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
// Skip combinator
pos++;
return newNode(type, content, line, column);
}
/**
* (1) `/panda/`
*/
function checkCombinator4(i) {
var start = i;
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
var l = void 0;
if (l = checkIdent(i)) i += l;else return 0;
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getCombinator4() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `/`.
pos++;
var ident = getIdent();
// Skip `/`.
pos++;
var content = '/' + ident.content + '/';
return newNode(type, content, line, column);
}
/**
* Check if token is a multiline comment.
* @param {Number} i Token's index number
* @return {Number} `1` if token is a multiline comment, otherwise `0`
*/
function checkCommentML(i) {
return i < tokensLength && tokens[i].type === TokenType.CommentML ? 1 : 0;
}
/**
* Get node with a multiline comment
* @return {Node}
*/
function getCommentML() {
var type = NodeType.CommentMLType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos].value.substring(2);
var l = content.length;
if (content.charAt(l - 2) === '*' && content.charAt(l - 1) === '/') content = content.substring(0, l - 2);
var end = getLastPosition(content, line, column, 2);
if (end[0] === line) end[1] += 2;
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a declaration (property-value pair)
* @param {Number} i Token's index number
* @return {Number} Length of the declaration
*/
function checkDeclaration(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkProperty(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkPropertyDelim(i)) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkValue(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a declaration
* @return {Node}
*/
function getDeclaration() {
var type = NodeType.DeclarationType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getProperty(), getSC(), getPropertyDelim(), getSC(), getValue());
return newNode(type, content, line, column);
}
/**
* Check if token is a semicolon
* @param {Number} i Token's index number
* @return {Number} `1` if token is a semicolon, otherwise `0`
*/
function checkDeclDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Semicolon ? 1 : 0;
}
/**
* Get node with a semicolon
* @return {Node}
*/
function getDeclDelim() {
var type = NodeType.DeclDelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ';';
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a comma
* @param {Number} i Token's index number
* @return {Number} `1` if token is a comma, otherwise `0`
*/
function checkDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Comma ? 1 : 0;
}
/**
* Get node with a comma
* @return {Node}
*/
function getDelim() {
var type = NodeType.DelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ',';
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is part of a number with dimension unit (e.g. `10px`)
* @param {Number} i Token's index number
* @return {Number}
*/
function checkDimension(i) {
var ln = checkNumber(i);
var li = void 0;
if (i >= tokensLength || !ln || i + ln >= tokensLength) return 0;
return (li = checkUnit(i + ln)) ? ln + li : 0;
}
/**
* Get node of a number with dimension unit
* @return {Node}
*/
function getDimension() {
var type = NodeType.DimensionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getNumber(), getUnit()];
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkExpression(i) {
var start = i;
if (i >= tokensLength || tokens[i++].value !== 'expression' || i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) {
return 0;
}
return tokens[i].right - start + 1;
}
/**
* @return {Node}
*/
function getExpression() {
var type = NodeType.ExpressionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
pos++;
var content = joinValues(pos + 1, tokens[pos].right - 1);
var end = getLastPosition(content, line, column, 1);
if (end[0] === line) end[1] += 11;
pos = tokens[pos].right + 1;
return newNode(type, content, line, column, end);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkFunction(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
return i < tokensLength && tokens[i].type === TokenType.LeftParenthesis ? tokens[i].right - start + 1 : 0;
}
/**
* @return {Node}
*/
function getFunction() {
var type = NodeType.FunctionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getIdent(), getArguments());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an identifierю
* Grammar from CSS spec:
* h [0-9a-f]
* nonascii [\240-\377]
* unicode \\{h}{1,6}(\r\n|[ \t\r\n\f])?
* escape {unicode}|\\[^\r\n\f0-9a-f]
* nmstart [_a-z]|{nonascii}|{escape}
* nmchar [_a-z0-9-]|{nonascii}|{escape}
* ident -?{nmstart}{nmchar}*
*
* @param {number} i Token's index number
* @return {number} Length of the identifier
*/
function checkIdent(i) {
var start = i;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.HyphenMinus) i++;
if (tokens[i].type === TokenType.LowLine || tokens[i].type === TokenType.Identifier) i++;else return 0;
for (; i < tokensLength; i++) {
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
}
tokens[start].ident_last = i - 1;
return i - start;
}
/**
* Get node with an identifier
* @return {Node}
*/
function getIdent() {
var type = NodeType.IdentType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, tokens[pos].ident_last);
pos = tokens[pos].ident_last + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of `!important` word
* @param {Number} i Token's index number
* @return {Number}
*/
function checkImportant(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].value === 'important') {
tokens[start].importantEnd = i;
return i - start + 1;
} else {
return 0;
}
}
/**
* Get node with `!important` word
* @return {Node}
*/
function getImportant() {
var type = NodeType.ImportantType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, token.importantEnd);
pos = token.importantEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check a single keyframe block - `5% {}`
* @param {Number} i
* @returns {Number}
*/
function checkKeyframesBlock(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkKeyframesSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get a single keyframe block - `5% {}`
* @returns {Node}
*/
function getKeyframesBlock() {
var type = NodeType.RulesetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getKeyframesSelectorsGroup(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check all keyframe blocks - `5% {} 100% {}`
* @param {Number} i
* @returns {Number}
*/
function checkKeyframesBlocks(i) {
var start = i;
var l = void 0;
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlock(i)) i += l;else return 0;
while (tokens[i].type !== TokenType.RightCurlyBracket) {
if (l = checkSC(i)) i += l;else if (l = checkKeyframesBlock(i)) i += l;else break;
}
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
return i - start;
}
/**
* Get all keyframe blocks - `5% {} 100% {}`
* @returns {Node}
*/
function getKeyframesBlocks() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var keyframesBlocksEnd = token.right;
var content = [];
// Skip `{`.
pos++;
while (pos < keyframesBlocksEnd) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkKeyframesBlock(pos)) content.push(getKeyframesBlock());
}
var end = getLastPosition(content, line, column, 1);
// Skip `}`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a @keyframes rule.
* @param {Number} i Token's index number
* @return {Number} Length of the @keyframes rule
*/
function checkKeyframesRule(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
var atruleName = joinValues2(i - l, l);
if (atruleName.toLowerCase().indexOf('keyframes') === -1) return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getKeyframesRule() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdent(), getSC(), getKeyframesBlocks());
return newNode(type, content, line, column);
}
/**
* Check a single keyframe selector - `5%`, `from` etc
* @param {Number} i
* @returns {Number}
*/
function checkKeyframesSelector(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) {
// Valid selectors are only `from` and `to`.
var selector = joinValues2(i, l);
if (selector !== 'from' && selector !== 'to') return 0;
i += l;
tokens[start].keyframesSelectorType = 1;
} else if (l = checkPercentage(i)) {
i += l;
tokens[start].keyframesSelectorType = 2;
} else {
return 0;
}
return i - start;
}
/**
* Get a single keyframe selector
* @returns {Node}
*/
function getKeyframesSelector() {
var keyframesSelectorType = NodeType.KeyframesSelectorType;
var selectorType = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (token.keyframesSelectorType === 1) {
content.push(getIdent());
} else {
content.push(getPercentage());
}
var keyframesSelector = newNode(keyframesSelectorType, content, line, column);
return newNode(selectorType, [keyframesSelector], line, column);
}
/**
* Check the keyframe's selector groups
* @param {Number} i
* @returns {Number}
*/
function checkKeyframesSelectorsGroup(i) {
var start = i;
var l = void 0;
if (l = checkKeyframesSelector(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
var comma = checkDelim(i + spaceBefore);
if (!comma) break;
var spaceAfter = checkSC(i + spaceBefore + comma);
if (l = checkKeyframesSelector(i + spaceBefore + comma + spaceAfter)) {
i += spaceBefore + comma + spaceAfter + l;
} else break;
}
tokens[start].selectorsGroupEnd = i;
return i - start;
}
/**
* Get the keyframe's selector groups
* @returns {Array} An array of keyframe selectors
*/
function getKeyframesSelectorsGroup() {
var selectorsGroup = [];
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
selectorsGroup.push(getKeyframesSelector());
while (pos < selectorsGroupEnd) {
selectorsGroup = selectorsGroup.concat(getSC(), getDelim(), getSC(), getKeyframesSelector());
}
return selectorsGroup;
}
/**
* Check if token is a namespace sign (`|`)
* @param {Number} i Token's index number
* @return {Number} `1` if token is `|`, `0` if not
*/
function checkNamespace(i) {
return i < tokensLength && tokens[i].type === TokenType.VerticalLine ? 1 : 0;
}
/**
* Get node with a namespace sign
* @return {Node}
*/
function getNamespace() {
var type = NodeType.NamespaceType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '|';
pos++;
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkNmName2(i) {
if (tokens[i].type === TokenType.Identifier) return 1;else if (tokens[i].type !== TokenType.DecimalNumber) return 0;
i++;
return i < tokensLength && tokens[i].type === TokenType.Identifier ? 2 : 1;
}
/**
* @return {String}
*/
function getNmName2() {
var s = tokens[pos].value;
if (tokens[pos++].type === TokenType.DecimalNumber && pos < tokensLength && tokens[pos].type === TokenType.Identifier) s += tokens[pos++].value;
return s;
}
/**
* Check if token is part of a number
* @param {Number} i Token's index number
* @return {Number} Length of number
*/
function checkNumber(i) {
if (i >= tokensLength) return 0;
if (tokens[i].number_l) return tokens[i].number_l;
// `10`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && (!tokens[i + 1] || tokens[i + 1] && tokens[i + 1].type !== TokenType.FullStop)) {
tokens[i].number_l = 1;
return 1;
}
// `10.`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && (!tokens[i + 2] || tokens[i + 2].type !== TokenType.DecimalNumber)) {
tokens[i].number_l = 2;
return 2;
}
// `.10`:
if (i < tokensLength && tokens[i].type === TokenType.FullStop && tokens[i + 1].type === TokenType.DecimalNumber) {
tokens[i].number_l = 2;
return 2;
}
// `10.10`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && tokens[i + 2] && tokens[i + 2].type === TokenType.DecimalNumber) {
tokens[i].number_l = 3;
return 3;
}
return 0;
}
/**
* Get node with number
* @return {Node}
*/
function getNumber() {
var type = NodeType.NumberType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var l = tokens[pos].number_l;
var content = '';
for (var j = 0; j < l; j++) {
content += tokens[pos + j].value;
}
pos += l;
return newNode(type, content, line, column);
}
/**
* Check if token is an operator (`/`, `,`, `:`, `=`, `*`).
* @param {Number} i Token's index number
* @return {Number} `1` if token is an operator, otherwise `0`
*/
function checkOperator(i) {
if (i >= tokensLength) return 0;
switch (tokens[i].type) {
case TokenType.Solidus:
case TokenType.Comma:
case TokenType.Colon:
case TokenType.EqualsSign:
case TokenType.Asterisk:
return 1;
}
return 0;
}
/**
* Get node with an operator
* @return {Node}
*/
function getOperator() {
var type = NodeType.OperatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is part of text inside parentheses, e.g. `(1)`
* @param {Number} i Token's index number
* @return {Number}
*/
function checkParentheses(i) {
if (i >= tokensLength) return 0;
var start = i;
var right = tokens[i].right;
// Skip `(`.
if (tokens[i].type === TokenType.LeftParenthesis) i++;else return 0;
if (i < right) {
var l = checkTsets(i);
if (l) i += l;else return 0;
}
// Skip `)`.
i++;
return i - start;
}
/**
* Get node with text inside parentheses, e.g. `(1)`
* @return {Node}
*/
function getParentheses() {
var type = NodeType.ParenthesesType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var right = token.right;
var content = [];
// Skip `(`.
pos++;
if (pos < right) {
content = getTsets();
}
var end = getLastPosition(content, line, column, 1);
// Skip `)`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a number with percent sign (e.g. `10%`)
* @param {Number} i Token's index number
* @return {Number}
*/
function checkPercentage(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkNumber(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
// Skip `%`.
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
return i - start;
}
/**
* Get node of number with percent sign
* @return {Node}
*/
function getPercentage() {
var type = NodeType.PercentageType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getNumber()];
var end = getLastPosition(content, line, column, 1);
// Skip `%`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkProgid(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (joinValues2(i, 6) === 'progid:DXImageTransform.Microsoft.') i += 6;else return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type === TokenType.LeftParenthesis) {
tokens[start].progid_end = tokens[i].right;
i = tokens[i].right + 1;
} else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getProgid() {
var type = NodeType.ProgidType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var progid_end = token.progid_end;
var content = joinValues(pos, progid_end);
pos = progid_end + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of a property
* @param {Number} i Token's index number
* @return {Number} Length of the property
*/
function checkProperty(i) {
var start = i;
var l = void 0;
if (l = checkProperty1(i)) tokens[start].propertyType = 1;else if (l = checkProperty2(i)) tokens[start].propertyType = 2;
return l;
}
/**
* Get node with a property
* @return {Node}
*/
function getProperty() {
var type = tokens[pos].propertyType;
if (type === 1) return getProperty1();
if (type === 2) return getProperty2();
}
/**
* Check if token is part of a property
* @param {Number} i Token's index number
* @return {Number} Length of the property
*/
function checkProperty1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a property
* @return {Node}
*/
function getProperty1() {
var type = NodeType.PropertyType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* Check if token is part of a custom property
* @param {Number} i Token's index number
* @return {Number} Length of the property
*/
function checkProperty2(i) {
return checkCustomProperty(i);
}
/**
* Get node with a custom property
* @return {Node}
*/
function getProperty2() {
return getCustomProperty();
}
/**
* Check if token is part of a custom property
* @param {Number} i Token's index number
* @return {Number} Length of the property
*/
function checkCustomProperty(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.HyphenMinus || tokens[i + 1] && tokens[i + 1].type !== TokenType.HyphenMinus) return 0;
// Skip `--`
i += 2;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a custom property
* @return {Node}
*/
function getCustomProperty() {
var type = NodeType.CustomPropertyType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `--`
pos += 2;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* Check if token is a colon
* @param {Number} i Token's index number
* @return {Number} `1` if token is a colon, otherwise `0`
*/
function checkPropertyDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Colon ? 1 : 0;
}
/**
* Get node with a colon
* @return {Node}
*/
function getPropertyDelim() {
var type = NodeType.PropertyDelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ':';
// Skip `:`.
pos++;
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkPseudo(i) {
return checkPseudoe(i) || checkPseudoc(i);
}
/**
* @return {Node}
*/
function getPseudo() {
if (checkPseudoe(pos)) return getPseudoe();
if (checkPseudoc(pos)) return getPseudoc();
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkPseudoe(i) {
var l = void 0;
// Check `::`
if (i >= tokensLength || tokens[i].type !== TokenType.Colon || i >= tokensLength || tokens[i + 1].type !== TokenType.Colon) return 0;
if (l = checkPseudoElement1(i)) tokens[i].pseudoElementType = 1;else if (l = checkPseudoElement2(i)) tokens[i].pseudoElementType = 2;else return 0;
return l;
}
/**
* @return {Node}
*/
function getPseudoe() {
var childType = tokens[pos].pseudoElementType;
if (childType === 1) return getPseudoElement1();
if (childType === 2) return getPseudoElement2();
}
/**
* (1) `::slotted(selector)`
* (2) `::slotted(selector, selector)`
*/
function checkPseudoElement1(i) {
var start = i;
var l = void 0;
// Skip `::`.
i += 2;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
/**
* (1) `::slotted(selector)`
* (2) `::slotted(selector, selector)`
*/
function getPseudoElement1() {
var type = NodeType.PseudoeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `::`.
pos += 2;
content.push(getIdent());
{
var _type = NodeType.ArgumentsType;
var _token = tokens[pos];
var _line = _token.ln;
var _column = _token.col;
// Skip `(`.
pos++;
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
var end = getLastPosition(selectorContent, _line, _column, 1);
var args = newNode(_type, selectorContent, _line, _column, end);
content.push(args);
// Skip `)`.
pos++;
}
return newNode(type, content, line, column);
}
function checkPseudoElement2(i) {
var start = i;
var l = void 0;
// Skip `::`.
i += 2;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getPseudoElement2() {
var type = NodeType.PseudoeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `::`.
pos += 2;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkPseudoc(i) {
var l = void 0;
if (i >= tokensLength || tokens[i].type !== TokenType.Colon) return 0;
if (l = checkPseudoClass1(i)) tokens[i].pseudoClassType = 1;else if (l = checkPseudoClass2(i)) tokens[i].pseudoClassType = 2;else if (l = checkPseudoClass3(i)) tokens[i].pseudoClassType = 3;else if (l = checkPseudoClass4(i)) tokens[i].pseudoClassType = 4;else if (l = checkPseudoClass5(i)) tokens[i].pseudoClassType = 5;else if (l = checkPseudoClass6(i)) tokens[i].pseudoClassType = 6;else return 0;
return l;
}
/**
* @return {Node}
*/
function getPseudoc() {
var childType = tokens[pos].pseudoClassType;
if (childType === 1) return getPseudoClass1();
if (childType === 2) return getPseudoClass2();
if (childType === 3) return getPseudoClass3();
if (childType === 4) return getPseudoClass4();
if (childType === 5) return getPseudoClass5();
if (childType === 6) return getPseudoClass6();
}
/**
* (1) `:panda(selector)`
* (2) `:panda(selector, selector)`
*/
function checkPseudoClass1(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
/**
* (-) `:not(panda)`
*/
function getPseudoClass1() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content.push(getIdent());
{
var _type2 = NodeType.ArgumentsType;
var _token2 = tokens[pos];
var _line2 = _token2.ln;
var _column2 = _token2.col;
// Skip `(`.
pos++;
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
var end = getLastPosition(selectorContent, _line2, _column2, 1);
var args = newNode(_type2, selectorContent, _line2, _column2, end);
content.push(args);
// Skip `)`.
pos++;
}
return newNode(type, content, line, column);
}
/**
* (1) `:nth-child(odd)`
* (2) `:nth-child(even)`
* (3) `:lang(de-DE)`
*/
function checkPseudoClass2(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkIdent(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass2() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content.push(getIdent());
// Skip `(`.
pos++;
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [].concat(getSC(), getIdent(), getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(-3n + 2)`
*/
function checkPseudoClass3(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.DecimalNumber) i++;
if (i >= tokensLength) return 0;
if (tokens[i].value === 'n') i++;else return 0;
if (l = checkSC(i)) i += l;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.PlusSign || tokens[i].type === TokenType.HyphenMinus) i++;else return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass3() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content.push(getIdent());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkNumber(pos)) value.push(getNumber());
{
var _l = tokens[pos].ln;
var _c = tokens[pos].col;
var _content = tokens[pos].value;
var ident = newNode(NodeType.IdentType, _content, _l, _c);
value.push(ident);
pos++;
}
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
value = value.concat(getSC());
if (checkNumber(pos)) value.push(getNumber());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(-3n)`
*/
function checkPseudoClass4(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (tokens[i].type === TokenType.DecimalNumber) i++;
if (tokens[i].value === 'n') i++;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass4() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content.push(getIdent());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkNumber(pos)) value.push(getNumber());
if (checkIdent(pos)) value.push(getIdent());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(+8)`
*/
function checkPseudoClass5(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass5() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content.push(getIdent());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkNumber(pos)) value.push(getNumber());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:checked`
*/
function checkPseudoClass6(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
function getPseudoClass6() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `:`.
pos++;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkRuleset(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getRuleset() {
var type = NodeType.RulesetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getSelectorsGroup(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is marked as a space (if it's a space or a tab
* or a line break).
* @param {Number} i
* @return {Number} Number of spaces in a row starting with the given token.
*/
function checkS(i) {
return i < tokensLength && tokens[i].ws ? tokens[i].ws_last - i + 1 : 0;
}
/**
* Get node with spaces
* @return {Node}
*/
function getS() {
var type = NodeType.SType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, tokens[pos].ws_last);
pos = tokens[pos].ws_last + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is a space or a comment.
* @param {Number} i Token's index number
* @return {Number} Number of similar (space or comment) tokens
* in a row starting with the given token.
*/
function checkSC(i) {
if (i >= tokensLength) return 0;
var l = void 0;
var lsc = 0;
while (i < tokensLength) {
if (l = checkS(i)) tokens[i].sc_child = 1;else if (l = checkCommentML(i)) tokens[i].sc_child = 2;else break;
i += l;
lsc += l;
}
return lsc || 0;
}
/**
* Get node with spaces and comments
* @return {Array}
*/
function getSC() {
var sc = [];
if (pos >= tokensLength) return sc;
while (pos < tokensLength) {
var childType = tokens[pos].sc_child;
if (childType === 1) sc.push(getS());else if (childType === 2) sc.push(getCommentML());else break;
}
return sc;
}
/**
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
* a simple selector
* @param {Number} i Token's index number
* @return {Number}
*/
function checkShash(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a hexadecimal number (e.g. `#fff`) inside a simple
* selector
* @return {Node}
*/
function getShash() {
var type = NodeType.ShashType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `#`.
pos++;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* Check if token is part of a string (text wrapped in quotes)
* @param {Number} i Token's index number
* @return {Number} `1` if token is part of a string, `0` if not
*/
function checkString(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.StringSQ || tokens[i].type === TokenType.StringDQ) {
return 1;
}
return 0;
}
/**
* Get string's node
* @return {Array} `['string', x]` where `x` is a string (including
* quotes).
*/
function getString() {
var type = NodeType.StringType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Validate stylesheet: it should consist of any number (0 or more) of
* rulesets (sets of rules with selectors), @-rules, whitespaces or
* comments.
* @param {Number} i Token's index number
* @return {Number}
*/
function checkStylesheet(i) {
var start = i;
var l = void 0;
while (i < tokensLength) {
if (l = checkSC(i)) tokens[i].stylesheet_child = 1;else if (l = checkRuleset(i)) tokens[i].stylesheet_child = 2;else if (l = checkAtrule(i)) tokens[i].stylesheet_child = 3;else if (l = checkDeclDelim(i)) tokens[i].stylesheet_child = 4;else throwError(i);
i += l;
}
return i - start;
}
/**
* @return {Array} `['stylesheet', x]` where `x` is all stylesheet's
* nodes.
*/
function getStylesheet() {
var type = NodeType.StylesheetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
while (pos < tokensLength) {
var childType = tokens[pos].stylesheet_child;
if (childType === 1) content = content.concat(getSC());
if (childType === 2) content.push(getRuleset());
if (childType === 3) content.push(getAtrule());
if (childType === 4) content.push(getDeclDelim());
}
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkTset(i) {
var l = void 0;
if (l = checkVhash(i)) tokens[i].tset_child = 1;else if (l = checkAny(i)) tokens[i].tset_child = 2;else if (l = checkSC(i)) tokens[i].tset_child = 3;else if (l = checkOperator(i)) tokens[i].tset_child = 4;
return l;
}
/**
* @return {Array}
*/
function getTset() {
var childType = tokens[pos].tset_child;
if (childType === 1) return getVhash();
if (childType === 2) return getAny();
if (childType === 3) return getSC();
if (childType === 4) return getOperator();
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkTsets(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
while (l = checkTset(i)) {
i += l;
}
tokens[start].tsets_end = i;
return i - start;
}
/**
* @return {Array}
*/
function getTsets() {
var content = [];
var t = void 0;
if (pos >= tokensLength) return content;
var end = tokens[pos].tsets_end;
while (pos < end) {
t = getTset();
if (typeof t.content === 'string') content.push(t);else content = content.concat(t);
}
return content;
}
/**
* Check if token is an unary (arithmetical) sign (`+` or `-`)
* @param {Number} i Token's index number
* @return {Number} `1` if token is an unary sign, `0` if not
*/
function checkUnary(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.HyphenMinus || tokens[i].type === TokenType.PlusSign) {
return 1;
}
return 0;
}
/**
* Get node with an unary (arithmetical) sign (`+` or `-`)
* @return {Array} `['unary', x]` where `x` is an unary sign
* converted to string.
*/
function getUnary() {
var type = NodeType.OperatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a unicode range (single or multiple <urange> nodes)
* @param {number} i Token's index
* @return {number} Unicode range node's length
*/
function checkUnicodeRange(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkUrange(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
var comma = checkDelim(i + spaceBefore);
if (!comma) break;
var spaceAfter = checkSC(i + spaceBefore + comma);
if (l = checkUrange(i + spaceBefore + comma + spaceAfter)) {
i += spaceBefore + comma + spaceAfter + l;
} else break;
}
return i - start;
}
/**
* Get a unicode range node
* @return {Node}
*/
function getUnicodeRange() {
var type = NodeType.UnicodeRangeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
while (pos < tokensLength) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkDelim(pos)) content.push(getDelim());else if (checkUrange(pos)) content.push(getUrange());else break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is unit
* @param {Number} i Token's index number
* @return {Number}
*/
function checkUnit(i) {
var units = ['em', 'ex', 'ch', 'rem', 'vh', 'vw', 'vmin', 'vmax', 'px', 'mm', 'q', 'cm', 'in', 'pt', 'pc', 'deg', 'grad', 'rad', 'turn', 's', 'ms', 'Hz', 'kHz', 'dpi', 'dpcm', 'dppx'];
return units.indexOf(tokens[i].value) !== -1 ? 1 : 0;
}
/**
* Get unit node of type ident
* @return {Node} An ident node containing the unit value
*/
function getUnit() {
var type = NodeType.IdentType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a u-range (part of a unicode-range)
* (1) `U+416`
* (2) `U+400-4ff`
* (3) `U+4??`
* @param {number} i Token's index
* @return {number} Urange node's length
*/
function checkUrange(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
// Check for unicode prefix (u+ or U+)
if (tokens[i].value === 'U' || tokens[i].value === 'u') i += 1;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].value === '+') i += 1;else return 0;
while (i < tokensLength) {
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = _checkUnicodeWildcard(i)) i += l;else break;
}
tokens[start].urangeEnd = i - 1;
return i - start;
}
/**
* Get a u-range node (part of a unicode-range)
* @return {Node}
*/
function getUrange() {
var startPos = pos;
var type = NodeType.UrangeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content = joinValues(startPos, tokens[startPos].urangeEnd);
pos = tokens[startPos].urangeEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check for unicode wildcard characters `?`
* @param {number} i Token's index
* @return {number} Wildcard length
*/
function _checkUnicodeWildcard(i) {
var start = i;
if (i >= tokensLength) return 0;
while (i < tokensLength) {
if (tokens[i].type === TokenType.QuestionMark) i += 1;else break;
}
return i - start;
}
/**
* Check if token is part of URI (e.g. `url('/css/styles.css')`)
* @param {Number} i Token's index number
* @return {Number} Length of URI
*/
function checkUri(i) {
var start = i;
if (i >= tokensLength || tokens[i].value !== 'url') return 0;
// Skip `url`.
i++;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
return tokens[i].right - start + 1;
}
/**
* Get node with URI
* @return {Array} `['uri', x]` where `x` is URI's nodes (without `url`
* and braces, e.g. `['string', ''/css/styles.css'']`).
*/
function getUri() {
var startPos = pos;
var uriExcluding = {};
var uri = void 0;
var l = void 0;
var raw = void 0;
var rawContent = void 0;
var t = void 0;
pos += 2;
uriExcluding[TokenType.Space] = 1;
uriExcluding[TokenType.Tab] = 1;
uriExcluding[TokenType.Newline] = 1;
uriExcluding[TokenType.LeftParenthesis] = 1;
uriExcluding[TokenType.RightParenthesis] = 1;
if (checkUri1(pos)) {
uri = [].concat(getSC(), getString(), getSC());
} else {
uri = checkSC(pos) ? getSC() : [];
l = checkExcluding(uriExcluding, pos);
rawContent = joinValues(pos, pos + l);
t = tokens[pos];
raw = newNode(NodeType.RawType, rawContent, t.ln, t.col);
uri.push(raw);
pos += l + 1;
if (checkSC(pos)) uri = uri.concat(getSC());
}
t = tokens[startPos];
var line = t.ln;
var column = t.col;
var end = getLastPosition(uri, line, column, 1);
pos++;
return newNode(NodeType.UriType, uri, line, column, end);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkUri1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type !== TokenType.StringDQ && tokens[i].type !== TokenType.StringSQ) return 0;
i++;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* Check if token is part of a value
* @param {Number} i Token's index number
* @return {Number} Length of the value
*/
function checkValue(i) {
var start = i;
var l = void 0;
var s = void 0;
var _i = void 0;
while (i < tokensLength) {
s = checkSC(i);
_i = i + s;
if (l = _checkValue(_i)) i += l + s;else break;
}
tokens[start].value_end = i;
return i - start;
}
/**
* @return {Array}
*/
function getValue() {
var type = NodeType.ValueType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = tokens[pos].value_end;
var content = [];
while (pos < end) {
if (tokens[pos].value_child) content.push(_getValue());else content = content.concat(getSC());
}
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function _checkValue(i) {
var l = void 0;
if (l = checkProgid(i)) tokens[i].value_child = 1;else if (l = checkVhash(i)) tokens[i].value_child = 2;else if (l = checkAny(i)) tokens[i].value_child = 3;else if (l = checkOperator(i)) tokens[i].value_child = 4;else if (l = checkImportant(i)) tokens[i].value_child = 5;
return l;
}
/**
* @return {Array}
*/
function _getValue() {
var childType = tokens[pos].value_child;
if (childType === 1) return getProgid();
if (childType === 2) return getVhash();
if (childType === 3) return getAny();
if (childType === 4) return getOperator();
if (childType === 5) return getImportant();
}
/**
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
* some value
* @param {Number} i Token's index number
* @return {Number}
*/
function checkVhash(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
// Skip `#`.
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
if (l = checkNmName2(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a hexadecimal number (e.g. `#fff`) inside some value
* @return {Array} `['vhash', x]` where `x` is a hexadecimal number
* converted to string (without `#`, e.g. `'fff'`).
*/
function getVhash() {
var type = NodeType.VhashType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `#`.
pos++;
var content = getNmName2();
var end = getLastPosition(content, line, column + 1);
return newNode(type, content, line, column, end);
}
function checkSelectorsGroup(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
var selectorCounter = 0;
var delimCounter = 0;
if (l = checkSelector(i)) {
i += l;
selectorCounter++;
} else return 0;
while (i < tokensLength) {
var tempStart = i;
var tempIndex = i;
var tempLength = void 0;
var spaceBefore = checkSC(tempIndex);
if (tempLength = checkDelim(tempIndex + spaceBefore)) {
tempIndex += spaceBefore + tempLength;
delimCounter++;
if (tempLength = checkSC(tempIndex)) tempIndex += tempLength;
if (tempLength = checkSelector(tempIndex)) {
tempIndex += tempLength;
selectorCounter++;
}
} else break;
i += tempIndex - tempStart;
}
tokens[start].selectorsGroupEnd = i;
tokens[start].selectorsGroupSelectorCount = selectorCounter;
tokens[start].selectorsGroupDelimCount = delimCounter;
return i - start;
}
function getSelectorsGroup() {
var selectorsGroup = [];
var selectorCounter = 0;
var delimCounter = 0;
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
var selectorCount = tokens[pos].selectorsGroupSelectorCount;
var delimCount = tokens[pos].selectorsGroupDelimCount;
selectorsGroup.push(getSelector());
selectorCounter++;
while (pos < selectorsGroupEnd) {
if (delimCounter < delimCount) {
selectorsGroup = selectorsGroup.concat(getSC());
selectorsGroup = selectorsGroup.concat(getDelim());
delimCounter++;
selectorsGroup = selectorsGroup.concat(getSC());
if (selectorCounter < selectorCount) {
selectorsGroup = selectorsGroup.concat(getSelector());
selectorCounter++;
}
}
}
return selectorsGroup;
}
function checkSelector(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkCompoundSelector(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
var comma = checkCombinator(i + spaceBefore);
if (!spaceBefore && !comma) break;
var spaceAfter = checkSC(i + spaceBefore + comma);
if (l = checkCompoundSelector(i + spaceBefore + comma + spaceAfter)) {
i += spaceBefore + comma + spaceAfter + l;
} else break;
}
tokens[start].selectorEnd = i;
return i - start;
}
function getSelector() {
var type = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var selectorEnd = token.selectorEnd;
var content = getCompoundSelector();
while (pos < selectorEnd) {
content = content.concat(getSC());
if (checkCombinator(pos)) content.push(getCombinator());
content = content.concat(getSC(), getCompoundSelector());
}
return newNode(type, content, line, column);
}
function checkCompoundSelector(i) {
var l = void 0;
if (l = checkCompoundSelector1(i)) {
tokens[i].compoundSelectorType = 1;
} else if (l = checkCompoundSelector2(i)) {
tokens[i].compoundSelectorType = 2;
}
return l;
}
function getCompoundSelector() {
var type = tokens[pos].compoundSelectorType;
if (type === 1) return getCompoundSelector1();
if (type === 2) return getCompoundSelector2();
}
function checkCompoundSelector1(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkUniversalSelector(i) || checkTypeSelector(i)) i += l;else return 0;
while (i < tokensLength) {
var _l2 = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i);
if (_l2) i += _l2;else break;
}
tokens[start].compoundSelectorEnd = i;
return i - start;
}
function getCompoundSelector1() {
var sequence = [];
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
if (checkUniversalSelector(pos)) sequence.push(getUniversalSelector());else sequence.push(getTypeSelector());
while (pos < compoundSelectorEnd) {
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());
}
return sequence;
}
function checkCompoundSelector2(i) {
if (i >= tokensLength) return 0;
var start = i;
while (i < tokensLength) {
var l = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i);
if (l) i += l;else break;
}
tokens[start].compoundSelectorEnd = i;
return i - start;
}
function getCompoundSelector2() {
var sequence = [];
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
while (pos < compoundSelectorEnd) {
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());else break;
}
return sequence;
}
function checkUniversalSelector(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (tokens[i].type === TokenType.Asterisk) i++;else return 0;
return i - start;
}
function getUniversalSelector() {
var type = NodeType.UniversalSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var end = void 0;
if (checkNamePrefix(pos)) {
content.push(getNamePrefix());
end = getLastPosition(content, line, column, 1);
}
pos++;
return newNode(type, content, line, column, end);
}
function checkTypeSelector(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
function getTypeSelector() {
var type = NodeType.TypeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkNamePrefix(pos)) content.push(getNamePrefix());
content.push(getIdent());
return newNode(type, content, line, column);
}
function checkAttributeSelector(i) {
var l = void 0;
if (l = checkAttributeSelector1(i)) tokens[i].attributeSelectorType = 1;else if (l = checkAttributeSelector2(i)) tokens[i].attributeSelectorType = 2;
return l;
}
function getAttributeSelector() {
var type = tokens[pos].attributeSelectorType;
if (type === 1) return getAttributeSelector1();else return getAttributeSelector2();
}
/**
* (1) `[panda=nani]`
* (2) `[panda='nani']`
* (3) `[panda='nani' i]`
*
*/
function checkAttributeSelector1(i) {
var start = i;
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeName(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeMatch(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeValue(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeFlags(i)) {
i += l;
if (l = checkSC(i)) i += l;
}
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
return i - start;
}
function getAttributeSelector1() {
var type = NodeType.AttributeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `[`.
pos++;
content = content.concat(getSC(), getAttributeName(), getSC(), getAttributeMatch(), getSC(), getAttributeValue(), getSC());
if (checkAttributeFlags(pos)) {
content.push(getAttributeFlags());
content = content.concat(getSC());
}
// Skip `]`.
pos++;
var end = getLastPosition(content, line, column, 1);
return newNode(type, content, line, column, end);
}
/**
* (1) `[panda]`
*/
function checkAttributeSelector2(i) {
var start = i;
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeName(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
return i - start;
}
function getAttributeSelector2() {
var type = NodeType.AttributeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `[`.
pos++;
content = content.concat(getSC(), getAttributeName(), getSC());
// Skip `]`.
pos++;
var end = getLastPosition(content, line, column, 1);
return newNode(type, content, line, column, end);
}
function checkAttributeName(i) {
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
function getAttributeName() {
var type = NodeType.AttributeNameType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkNamePrefix(pos)) content.push(getNamePrefix());
content.push(getIdent());
return newNode(type, content, line, column);
}
function checkAttributeMatch(i) {
var l = void 0;
if (l = checkAttributeMatch1(i)) tokens[i].attributeMatchType = 1;else if (l = checkAttributeMatch2(i)) tokens[i].attributeMatchType = 2;
return l;
}
function getAttributeMatch() {
var type = tokens[pos].attributeMatchType;
if (type === 1) return getAttributeMatch1();else return getAttributeMatch2();
}
function checkAttributeMatch1(i) {
var start = i;
var type = tokens[i].type;
if (type === TokenType.Tilde || type === TokenType.VerticalLine || type === TokenType.CircumflexAccent || type === TokenType.DollarSign || type === TokenType.Asterisk) i++;else return 0;
if (tokens[i].type === TokenType.EqualsSign) i++;else return 0;
return i - start;
}
function getAttributeMatch1() {
var type = NodeType.AttributeMatchType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos].value + tokens[pos + 1].value;
pos += 2;
return newNode(type, content, line, column);
}
function checkAttributeMatch2(i) {
if (tokens[i].type === TokenType.EqualsSign) return 1;else return 0;
}
function getAttributeMatch2() {
var type = NodeType.AttributeMatchType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '=';
pos++;
return newNode(type, content, line, column);
}
function checkAttributeValue(i) {
return checkString(i) || checkIdent(i);
}
function getAttributeValue() {
var type = NodeType.AttributeValueType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkString(pos)) content.push(getString());else content.push(getIdent());
return newNode(type, content, line, column);
}
function checkAttributeFlags(i) {
return checkIdent(i);
}
function getAttributeFlags() {
var type = NodeType.AttributeFlagsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getIdent()];
return newNode(type, content, line, column);
}
function checkNamePrefix(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkNamePrefix1(i)) tokens[i].namePrefixType = 1;else if (l = checkNamePrefix2(i)) tokens[i].namePrefixType = 2;
return l;
}
function getNamePrefix() {
var type = tokens[pos].namePrefixType;
if (type === 1) return getNamePrefix1();else return getNamePrefix2();
}
/**
* (1) `panda|`
* (2) `panda<comment>|`
*/
function checkNamePrefix1(i) {
var start = i;
var l = void 0;
if (l = checkNamespacePrefix(i)) i += l;else return 0;
if (l = checkCommentML(i)) i += l;
if (l = checkNamespaceSeparator(i)) i += l;else return 0;
return i - start;
}
function getNamePrefix1() {
var type = NodeType.NamePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(getNamespacePrefix());
if (checkCommentML(pos)) content.push(getCommentML());
content.push(getNamespaceSeparator());
return newNode(type, content, line, column);
}
/**
* (1) `|`
*/
function checkNamePrefix2(i) {
return checkNamespaceSeparator(i);
}
function getNamePrefix2() {
var type = NodeType.NamePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getNamespaceSeparator()];
return newNode(type, content, line, column);
}
/**
* (1) `*`
* (2) `panda`
*/
function checkNamespacePrefix(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (tokens[i].type === TokenType.Asterisk) return 1;else if (l = checkIdent(i)) return l;else return 0;
}
function getNamespacePrefix() {
var type = NodeType.NamespacePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (token.type === TokenType.Asterisk) {
var asteriskNode = newNode(NodeType.IdentType, '*', line, column);
content.push(asteriskNode);
pos++;
} else if (checkIdent(pos)) content.push(getIdent());
return newNode(type, content, line, column);
}
/**
* (1) `|`
*/
function checkNamespaceSeparator(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.VerticalLine) return 0;
// Return false if `|=` - [attr|=value]
if (tokens[i + 1] && tokens[i + 1].type === TokenType.EqualsSign) return 0;
return 1;
}
function getNamespaceSeparator() {
var type = NodeType.NamespaceSeparatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '|';
pos++;
return newNode(type, content, line, column);
}
module.exports = function (_tokens, context) {
tokens = _tokens;
tokensLength = tokens.length;
pos = 0;
return contexts[context]();
};
/***/ }),
/* 15 */
/***/ (function(module, exports) {
'use strict';
module.exports = {
ArgumentsType: 'arguments',
AtkeywordType: 'atkeyword',
AtruleType: 'atrule',
AttributeSelectorType: 'attributeSelector',
AttributeNameType: 'attributeName',
AttributeFlagsType: 'attributeFlags',
AttributeMatchType: 'attributeMatch',
AttributeValueType: 'attributeValue',
BlockType: 'block',
BracketsType: 'brackets',
ClassType: 'class',
CombinatorType: 'combinator',
CommentMLType: 'multilineComment',
CommentSLType: 'singlelineComment',
ConditionType: 'condition',
ConditionalStatementType: 'conditionalStatement',
CustomPropertyType: 'customProperty',
DeclarationType: 'declaration',
DeclDelimType: 'declarationDelimiter',
DefaultType: 'default',
DelimType: 'delimiter',
DimensionType: 'dimension',
EscapedStringType: 'escapedString',
ExtendType: 'extend',
ExpressionType: 'expression',
FunctionType: 'function',
FunctionsListType: 'functionsList',
GlobalType: 'global',
IdentType: 'ident',
ImportantType: 'important',
IncludeType: 'include',
InterpolationType: 'interpolation',
InterpolatedVariableType: 'interpolatedVariable',
KeyframesSelectorType: 'keyframesSelector',
LoopType: 'loop',
MixinType: 'mixin',
NamePrefixType: 'namePrefix',
NamespacePrefixType: 'namespacePrefix',
NamespaceSeparatorType: 'namespaceSeparator',
NumberType: 'number',
OperatorType: 'operator',
OptionalType: 'optional',
ParenthesesType: 'parentheses',
ParentSelectorType: 'parentSelector',
ParentSelectorExtensionType: 'parentSelectorExtension',
PercentageType: 'percentage',
PlaceholderType: 'placeholder',
ProgidType: 'progid',
PropertyType: 'property',
PropertyDelimType: 'propertyDelimiter',
PseudocType: 'pseudoClass',
PseudoeType: 'pseudoElement',
RawType: 'raw',
RulesetType: 'ruleset',
SType: 'space',
SelectorType: 'selector',
ShashType: 'id',
StringType: 'string',
StylesheetType: 'stylesheet',
TypeSelectorType: 'typeSelector',
UnicodeRangeType: 'unicodeRange',
UniversalSelectorType: 'universalSelector',
UriType: 'uri',
UrangeType: 'urange',
ValueType: 'value',
VariableType: 'variable',
VariablesListType: 'variablesList',
VhashType: 'color'
};
/***/ }),
/* 16 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
module.exports = function (css, tabSize) {
var TokenType = __webpack_require__(13);
var tokens = [];
var urlMode = false;
var blockMode = 0;
var pos = 0;
var tn = 0;
var ln = 1;
var col = 1;
var cssLength = 0;
var Punctuation = {
' ': TokenType.Space,
'\n': TokenType.Newline,
'\r': TokenType.Newline,
'\t': TokenType.Tab,
'!': TokenType.ExclamationMark,
'"': TokenType.QuotationMark,
'#': TokenType.NumberSign,
'$': TokenType.DollarSign,
'%': TokenType.PercentSign,
'&': TokenType.Ampersand,
'\'': TokenType.Apostrophe,
'(': TokenType.LeftParenthesis,
')': TokenType.RightParenthesis,
'*': TokenType.Asterisk,
'+': TokenType.PlusSign,
',': TokenType.Comma,
'-': TokenType.HyphenMinus,
'.': TokenType.FullStop,
'/': TokenType.Solidus,
':': TokenType.Colon,
';': TokenType.Semicolon,
'<': TokenType.LessThanSign,
'=': TokenType.EqualsSign,
'>': TokenType.GreaterThanSign,
'?': TokenType.QuestionMark,
'@': TokenType.CommercialAt,
'[': TokenType.LeftSquareBracket,
']': TokenType.RightSquareBracket,
'^': TokenType.CircumflexAccent,
'_': TokenType.LowLine,
'{': TokenType.LeftCurlyBracket,
'|': TokenType.VerticalLine,
'}': TokenType.RightCurlyBracket,
'~': TokenType.Tilde
};
/**
* Add a token to the token list
* @param {string} type
* @param {string} value
*/
function pushToken(type, value, column) {
tokens.push({
tn: tn++,
ln: ln,
col: column,
type: type,
value: value
});
}
/**
* Check if a character is a decimal digit
* @param {string} c Character
* @returns {boolean}
*/
function isDecimalDigit(c) {
return '0123456789'.indexOf(c) >= 0;
}
/**
* Parse spaces
* @param {string} css Unparsed part of CSS string
*/
function parseSpaces(css) {
var start = pos;
// Read the string until we meet a non-space character:
for (; pos < cssLength; pos++) {
if (css.charAt(pos) !== ' ') break;
}
// Add a substring containing only spaces to tokens:
pushToken(TokenType.Space, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Parse a string within quotes
* @param {string} css Unparsed part of CSS string
* @param {string} q Quote (either `'` or `"`)
*/
function parseString(css, q) {
var start = pos;
// Read the string until we meet a matching quote:
for (pos++; pos < cssLength; pos++) {
// Skip escaped quotes:
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) === q) break;
}
// Add the string (including quotes) to tokens:
pushToken(q === '"' ? TokenType.StringDQ : TokenType.StringSQ, css.substring(start, pos + 1), col);
col += pos - start;
}
/**
* Parse numbers
* @param {string} css Unparsed part of CSS string
*/
function parseDecimalNumber(css) {
var start = pos;
// Read the string until we meet a character that's not a digit:
for (; pos < cssLength; pos++) {
if (!isDecimalDigit(css.charAt(pos))) break;
}
// Add the number to tokens:
pushToken(TokenType.DecimalNumber, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Parse identifier
* @param {string} css Unparsed part of CSS string
*/
function parseIdentifier(css) {
var start = pos;
// Skip all opening slashes:
while (css.charAt(pos) === '/') {
pos++;
} // Read the string until we meet a punctuation mark:
for (; pos < cssLength; pos++) {
// Skip all '\':
if (css.charAt(pos) === '\\') pos++;else if (Punctuation[css.charAt(pos)]) break;
}
var ident = css.substring(start, pos--);
// Enter url mode if parsed substring is `url`:
urlMode = urlMode || ident === 'url';
// Add identifier to tokens:
pushToken(TokenType.Identifier, ident, col);
col += pos - start;
}
/**
* Parse a multiline comment
* @param {string} css Unparsed part of CSS string
*/
function parseMLComment(css) {
var start = pos;
// Read the string until we meet `*/`.
// Since we already know first 2 characters (`/*`), start reading
// from `pos + 2`:
for (pos = pos + 2; pos < cssLength; pos++) {
if (css.charAt(pos) === '*' && css.charAt(pos + 1) === '/') {
pos++;
break;
}
}
// Add full comment (including `/*` and `*/`) to the list of tokens:
var comment = css.substring(start, pos + 1);
pushToken(TokenType.CommentML, comment, col);
var newlines = comment.split('\n');
if (newlines.length > 1) {
ln += newlines.length - 1;
col = newlines[newlines.length - 1].length;
} else {
col += pos - start;
}
}
function parseSLComment(css) {
var start = pos;
// Read the string until we meet line break.
// Since we already know first 2 characters (`//`), start reading
// from `pos + 2`:
for (pos += 2; pos < cssLength; pos++) {
if (css.charAt(pos) === '\n' || css.charAt(pos) === '\r') {
break;
}
}
// Add comment (including `//` and line break) to the list of tokens:
pushToken(TokenType.CommentSL, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Convert a CSS string to a list of tokens
* @param {string} css CSS string
* @returns {Array} List of tokens
* @private
*/
function getTokens(css) {
var c; // Current character
var cn; // Next character
cssLength = css.length;
// Parse string, character by character:
for (pos = 0; pos < cssLength; col++, pos++) {
c = css.charAt(pos);
cn = css.charAt(pos + 1);
// If we meet `/*`, it's a start of a multiline comment.
// Parse following characters as a multiline comment:
if (c === '/' && cn === '*') {
parseMLComment(css);
}
// If we meet `//` and it is not a part of url:
else if (!urlMode && c === '/' && cn === '/') {
// If we're currently inside a block, treat `//` as a start
// of identifier. Else treat `//` as a start of a single-line
// comment:
if (blockMode > 0) parseIdentifier(css);else parseSLComment(css);
}
// If current character is a double or single quote, it's a start
// of a string:
else if (c === '"' || c === "'") {
parseString(css, c);
}
// If current character is a space:
else if (c === ' ') {
parseSpaces(css);
}
// If current character is a punctuation mark:
else if (Punctuation[c]) {
// Add it to the list of tokens:
pushToken(Punctuation[c], c, col);
if (c === '\n' || c === '\r') {
ln++;
col = 0;
} // Go to next line
else if (c === ')') urlMode = false; // Exit url mode
else if (c === '{') blockMode++; // Enter a block
else if (c === '}') blockMode--; // Exit a block
else if (c === '\t' && tabSize > 1) col += tabSize - 1;
}
// If current character is a decimal digit:
else if (isDecimalDigit(c)) {
parseDecimalNumber(css);
}
// If current character is anything else:
else {
parseIdentifier(css);
}
}
return tokens;
}
return getTokens(css);
};
/***/ }),
/* 17 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
exports.__esModule = true;
exports.default = {
mark: __webpack_require__(18),
parse: __webpack_require__(19),
stringify: __webpack_require__(4),
tokenizer: __webpack_require__(20)
};
module.exports = exports['default'];
/***/ }),
/* 18 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var TokenType = __webpack_require__(13);
module.exports = function () {
/**
* Mark whitespaces and comments
*/
function markSC(tokens) {
var tokensLength = tokens.length;
var ws = -1; // Flag for whitespaces
var sc = -1; // Flag for whitespaces and comments
var t = void 0; // Current token
// For every token in the token list, mark spaces and line breaks
// as spaces (set both `ws` and `sc` flags). Mark multiline comments
// with `sc` flag.
// If there are several spaces or tabs or line breaks or multiline
// comments in a row, group them: take the last one's index number
// and save it to the first token in the group as a reference:
// e.g., `ws_last = 7` for a group of whitespaces or `sc_last = 9`
// for a group of whitespaces and comments.
for (var i = 0; i < tokensLength; i++) {
t = tokens[i];
switch (t.type) {
case TokenType.Space:
case TokenType.Tab:
case TokenType.Newline:
t.ws = true;
t.sc = true;
if (ws === -1) ws = i;
if (sc === -1) sc = i;
break;
case TokenType.CommentML:
case TokenType.CommentSL:
if (ws !== -1) {
tokens[ws].ws_last = i - 1;
ws = -1;
}
t.sc = true;
break;
default:
if (ws !== -1) {
tokens[ws].ws_last = i - 1;
ws = -1;
}
if (sc !== -1) {
tokens[sc].sc_last = i - 1;
sc = -1;
}
}
}
if (ws !== -1) tokens[ws].ws_last = i - 1;
if (sc !== -1) tokens[sc].sc_last = i - 1;
}
/**
* Pair brackets
*/
function markBrackets(tokens) {
var tokensLength = tokens.length;
var ps = []; // Parentheses
var sbs = []; // Square brackets
var cbs = []; // Curly brackets
var t = void 0; // Current token
// For every token in the token list, if we meet an opening (left)
// bracket, push its index number to a corresponding array.
// If we then meet a closing (right) bracket, look at the corresponding
// array. If there are any elements (records about previously met
// left brackets), take a token of the last left bracket (take
// the last index number from the array and find a token with
// this index number) and save right bracket's index as a reference:
for (var i = 0; i < tokensLength; i++) {
t = tokens[i];
switch (t.type) {
case TokenType.LeftParenthesis:
ps.push(i);
break;
case TokenType.RightParenthesis:
if (ps.length) {
t.left = ps.pop();
tokens[t.left].right = i;
}
break;
case TokenType.LeftSquareBracket:
sbs.push(i);
break;
case TokenType.RightSquareBracket:
if (sbs.length) {
t.left = sbs.pop();
tokens[t.left].right = i;
}
break;
case TokenType.LeftCurlyBracket:
cbs.push(i);
break;
case TokenType.RightCurlyBracket:
if (cbs.length) {
t.left = cbs.pop();
tokens[t.left].right = i;
}
break;
}
}
}
return function (tokens) {
markBrackets(tokens);
markSC(tokens);
};
}();
/***/ }),
/* 19 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var Node = __webpack_require__(1);
var NodeType = __webpack_require__(15);
var TokenType = __webpack_require__(13);
var tokens = void 0;
var tokensLength = void 0;
var pos = void 0;
var contexts = {
'arguments': function _arguments() {
return checkArguments(pos) && getArguments();
},
'atkeyword': function atkeyword() {
return checkAtkeyword(pos) && getAtkeyword();
},
'atrule': function atrule() {
return checkAtrule(pos) && getAtrule();
},
'attributeSelector': function attributeSelector() {
return checkAttributeSelector(pos) && getAttributeSelector();
},
'block': function block() {
return checkBlock(pos) && getBlock();
},
'brackets': function brackets() {
return checkBrackets(pos) && getBrackets();
},
'class': function _class() {
return checkClass(pos) && getClass();
},
'combinator': function combinator() {
return checkCombinator(pos) && getCombinator();
},
'commentML': function commentML() {
return checkCommentML(pos) && getCommentML();
},
'commentSL': function commentSL() {
return checkCommentSL(pos) && getCommentSL();
},
'condition': function condition() {
return checkCondition(pos) && getCondition();
},
'declaration': function declaration() {
return checkDeclaration(pos) && getDeclaration();
},
'declDelim': function declDelim() {
return checkDeclDelim(pos) && getDeclDelim();
},
'delim': function delim() {
return checkDelim(pos) && getDelim();
},
'dimension': function dimension() {
return checkDimension(pos) && getDimension();
},
'escapedString': function escapedString() {
return checkEscapedString(pos) && getEscapedString();
},
'expression': function expression() {
return checkExpression(pos) && getExpression();
},
'extend': function extend() {
return checkExtend(pos) && getExtend();
},
'function': function _function() {
return checkFunction(pos) && getFunction();
},
'ident': function ident() {
return checkIdent(pos) && getIdent();
},
'important': function important() {
return checkImportant(pos) && getImportant();
},
'include': function include() {
return checkInclude(pos) && getInclude();
},
'interpolatedVariable': function interpolatedVariable() {
return checkInterpolatedVariable(pos) && getInterpolatedVariable();
},
'mixin': function mixin() {
return checkMixin(pos) && getMixin();
},
'namespace': function namespace() {
return checkNamespace(pos) && getNamespace();
},
'number': function number() {
return checkNumber(pos) && getNumber();
},
'operator': function operator() {
return checkOperator(pos) && getOperator();
},
'parentheses': function parentheses() {
return checkParentheses(pos) && getParentheses();
},
'parentselector': function parentselector() {
return checkParentSelector(pos) && getParentSelector();
},
'percentage': function percentage() {
return checkPercentage(pos) && getPercentage();
},
'progid': function progid() {
return checkProgid(pos) && getProgid();
},
'property': function property() {
return checkProperty(pos) && getProperty();
},
'propertyDelim': function propertyDelim() {
return checkPropertyDelim(pos) && getPropertyDelim();
},
'pseudoc': function pseudoc() {
return checkPseudoc(pos) && getPseudoc();
},
'pseudoe': function pseudoe() {
return checkPseudoe(pos) && getPseudoe();
},
'ruleset': function ruleset() {
return checkRuleset(pos) && getRuleset();
},
's': function s() {
return checkS(pos) && getS();
},
'selector': function selector() {
return checkSelector(pos) && getSelector();
},
'shash': function shash() {
return checkShash(pos) && getShash();
},
'string': function string() {
return checkString(pos) && getString();
},
'stylesheet': function stylesheet() {
return checkStylesheet(pos) && getStylesheet();
},
'unary': function unary() {
return checkUnary(pos) && getUnary();
},
'unicodeRange': function unicodeRange() {
return checkUnicodeRange(pos) && getUnicodeRange();
},
'universalSelector': function universalSelector() {
return checkUniversalSelector(pos) && getUniversalSelector();
},
'urange': function urange() {
return checkUrange(pos) && getUrange();
},
'uri': function uri() {
return checkUri(pos) && getUri();
},
'value': function value() {
return checkValue(pos) && getValue();
},
'variable': function variable() {
return checkVariable(pos) && getVariable();
},
'variableslist': function variableslist() {
return checkVariablesList(pos) && getVariablesList();
},
'vhash': function vhash() {
return checkVhash(pos) && getVhash();
}
};
/**
* Stop parsing and display error
* @param {Number=} i Token's index number
*/
function throwError(i) {
var ln = tokens[i].ln;
throw { line: ln, syntax: 'less' };
}
/**
* @param {Object} exclude
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkExcluding(exclude, i) {
var start = i;
while (i < tokensLength) {
if (exclude[tokens[i++].type]) break;
}
return i - start - 2;
}
/**
* @param {Number} start
* @param {Number} finish
* @returns {String}
*/
function joinValues(start, finish) {
var s = '';
for (var i = start; i < finish + 1; i++) {
s += tokens[i].value;
}
return s;
}
/**
* @param {Number} start
* @param {Number} num
* @returns {String}
*/
function joinValues2(start, num) {
if (start + num - 1 >= tokensLength) return;
var s = '';
for (var i = 0; i < num; i++) {
s += tokens[start + i].value;
}
return s;
}
function getLastPosition(content, line, column, colOffset) {
return typeof content === 'string' ? getLastPositionForString(content, line, column, colOffset) : getLastPositionForArray(content, line, column, colOffset);
}
function getLastPositionForString(content, line, column, colOffset) {
var position = [];
if (!content) {
position = [line, column];
if (colOffset) position[1] += colOffset - 1;
return position;
}
var lastLinebreak = content.lastIndexOf('\n');
var endsWithLinebreak = lastLinebreak === content.length - 1;
var splitContent = content.split('\n');
var linebreaksCount = splitContent.length - 1;
var prevLinebreak = linebreaksCount === 0 || linebreaksCount === 1 ? -1 : content.length - splitContent[linebreaksCount - 1].length - 2;
// Line:
var offset = endsWithLinebreak ? linebreaksCount - 1 : linebreaksCount;
position[0] = line + offset;
// Column:
if (endsWithLinebreak) {
offset = prevLinebreak !== -1 ? content.length - prevLinebreak : content.length - 1;
} else {
offset = linebreaksCount !== 0 ? content.length - lastLinebreak - column - 1 : content.length - 1;
}
position[1] = column + offset;
if (!colOffset) return position;
if (endsWithLinebreak) {
position[0]++;
position[1] = colOffset;
} else {
position[1] += colOffset;
}
return position;
}
function getLastPositionForArray(content, line, column, colOffset) {
var position = void 0;
if (content.length === 0) {
position = [line, column];
} else {
var c = content[content.length - 1];
if (c.hasOwnProperty('end')) {
position = [c.end.line, c.end.column];
} else {
position = getLastPosition(c.content, line, column);
}
}
if (!colOffset) return position;
if (tokens[pos - 1] && tokens[pos - 1].type !== 'Newline') {
position[1] += colOffset;
} else {
position[0]++;
position[1] = 1;
}
return position;
}
function newNode(type, content, line, column, end) {
if (!end) end = getLastPosition(content, line, column);
return new Node({
type: type,
content: content,
start: {
line: line,
column: column
},
end: {
line: end[0],
column: end[1]
},
syntax: 'less'
});
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkAny(i) {
var l = void 0;
if (l = checkBrackets(i)) tokens[i].any_child = 1;else if (l = checkParentheses(i)) tokens[i].any_child = 2;else if (l = checkString(i)) tokens[i].any_child = 3;else if (l = checkVariablesList(i)) tokens[i].any_child = 4;else if (l = checkVariable(i)) tokens[i].any_child = 5;else if (l = checkPercentage(i)) tokens[i].any_child = 6;else if (l = checkDimension(i)) tokens[i].any_child = 7;else if (l = checkUnicodeRange(i)) tokens[i].any_child = 15;else if (l = checkNumber(i)) tokens[i].any_child = 8;else if (l = checkUri(i)) tokens[i].any_child = 9;else if (l = checkExpression(i)) tokens[i].any_child = 10;else if (l = checkFunction(i)) tokens[i].any_child = 11;else if (l = checkIdent(i)) tokens[i].any_child = 12;else if (l = checkClass(i)) tokens[i].any_child = 13;else if (l = checkUnary(i)) tokens[i].any_child = 14;
return l;
}
/**
* @returns {Array}
*/
function getAny() {
var childType = tokens[pos].any_child;
if (childType === 1) return getBrackets();
if (childType === 2) return getParentheses();
if (childType === 3) return getString();
if (childType === 4) return getVariablesList();
if (childType === 5) return getVariable();
if (childType === 6) return getPercentage();
if (childType === 7) return getDimension();
if (childType === 15) return getUnicodeRange();
if (childType === 8) return getNumber();
if (childType === 9) return getUri();
if (childType === 10) return getExpression();
if (childType === 11) return getFunction();
if (childType === 12) return getIdent();
if (childType === 13) return getClass();
if (childType === 14) return getUnary();
}
/**
* Check if token is part of mixin's arguments.
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkArguments(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
// Skip `(`.
i++;
while (i < tokens[start].right) {
if (l = checkArgument(i)) i += l;else return 0;
}
return tokens[start].right - start + 1;
}
/**
* @returns {Array}
*/
function getArguments() {
var type = NodeType.ArgumentsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var body = void 0;
// Skip `(`.
pos++;
while (pos < tokensLength && tokens[pos].type !== TokenType.RightParenthesis) {
if (checkDeclaration(pos)) content.push(getDeclaration());else if (checkArgument(pos)) {
body = getArgument();
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
} else if (checkClass(pos)) content.push(getClass());else throwError(pos);
}
var end = getLastPosition(content, line, column, 1);
// Skip `)`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is valid to be part of arguments list.
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkArgument(i) {
var l = void 0;
if (l = checkEscapedString(i)) tokens[i].argument_child = 1;else if (l = checkDeclaration(i)) tokens[i].argument_child = 2;else if (l = checkVariablesList(i)) tokens[i].argument_child = 3;else if (l = checkVariable(i)) tokens[i].argument_child = 4;else if (l = checkSC(i)) tokens[i].argument_child = 5;else if (l = checkUnary(i)) tokens[i].argument_child = 6;else if (l = checkOperator(i)) tokens[i].argument_child = 7;else if (l = checkDelim(i)) tokens[i].argument_child = 8;else if (l = checkDeclDelim(i)) tokens[i].argument_child = 9;else if (l = checkString(i)) tokens[i].argument_child = 10;else if (l = checkPercentage(i)) tokens[i].argument_child = 11;else if (l = checkDimension(i)) tokens[i].argument_child = 12;else if (l = checkNumber(i)) tokens[i].argument_child = 13;else if (l = checkUri(i)) tokens[i].argument_child = 14;else if (l = checkFunction(i)) tokens[i].argument_child = 15;else if (l = checkIdent(i)) tokens[i].argument_child = 16;else if (l = checkVhash(i)) tokens[i].argument_child = 17;else if (l = checkBlock(i)) tokens[i].argument_child = 18;else if (l = checkParentheses(i)) tokens[i].argument_child = 19;
return l;
}
/**
* @returns {Array} Node that is part of arguments list.
*/
function getArgument() {
var childType = tokens[pos].argument_child;
if (childType === 1) return getEscapedString();
if (childType === 2) return getDeclaration();
if (childType === 3) return getVariablesList();
if (childType === 4) return getVariable();
if (childType === 5) return getSC();
if (childType === 6) return getUnary();
if (childType === 7) return getOperator();
if (childType === 8) return getDelim();
if (childType === 9) return getDeclDelim();
if (childType === 10) return getString();
if (childType === 11) return getPercentage();
if (childType === 12) return getDimension();
if (childType === 13) return getNumber();
if (childType === 14) return getUri();
if (childType === 15) return getFunction();
if (childType === 16) return getIdent();
if (childType === 17) return getVhash();
if (childType === 18) return getBlock();
if (childType === 19) return getParentheses();
}
/**
* Check if token is part of an @-word (e.g. `@import`, `@include`)
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkAtkeyword(i) {
var l = void 0;
// Check that token is `@`:
if (i >= tokensLength || tokens[i++].type !== TokenType.CommercialAt) return 0;
return (l = checkIdent(i)) ? l + 1 : 0;
}
/**
* Get node with @-word
* @returns {Array} `['atkeyword', ['ident', x]]` where `x` is
* an identifier without
* `@` (e.g. `import`, `include`)
*/
function getAtkeyword() {
var type = NodeType.AtkeywordType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `@`.
pos++;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* Check if token is a part of an @-rule
* @param {Number} i Token's index number
* @returns {Number} Length of @-rule
*/
function checkAtrule(i) {
var l = void 0;
if (i >= tokensLength) return 0;
// If token already has a record of being part of an @-rule,
// return the @-rule's length:
if (tokens[i].atrule_l !== undefined) return tokens[i].atrule_l;
// If token is part of an @-rule, save the rule's type to token.
if (l = checkKeyframesRule(i)) tokens[i].atrule_type = 4;
// @-rule with ruleset:
else if (l = checkAtruler(i)) tokens[i].atrule_type = 1;
// Block @-rule:
else if (l = checkAtruleb(i)) tokens[i].atrule_type = 2;
// Single-line @-rule:
else if (l = checkAtrules(i)) tokens[i].atrule_type = 3;else return 0;
// If token is part of an @-rule, save the rule's length to token:
tokens[i].atrule_l = l;
return l;
}
/**
* Get node with @-rule
* @returns {Array}
*/
function getAtrule() {
var childType = tokens[pos].atrule_type;
if (childType === 1) return getAtruler(); // @-rule with ruleset
if (childType === 2) return getAtruleb(); // Block @-rule
if (childType === 3) return getAtrules(); // Single-line @-rule
if (childType === 4) return getKeyframesRule();
}
/**
* Check if token is part of a block @-rule
* @param {Number} i Token's index number
* @returns {Number} Length of the @-rule
*/
function checkAtruleb(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a block @-rule
* @returns {Array} `['atruleb', ['atkeyword', x], y, ['block', z]]`
*/
function getAtruleb() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an @-rule with ruleset
* @param {Number} i Token's index number
* @returns {Number} Length of the @-rule
*/
function checkAtruler(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
if (l = checkAtrulers(i)) i += l;
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
return i - start;
}
/**
* Get node with an @-rule with ruleset
* @returns {Array} ['atruler', ['atkeyword', x], y, z]
*/
function getAtruler() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets(), getAtrulers());
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkAtrulers(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkSC(i)) i += l;
while (i < tokensLength) {
if (l = checkSC(i)) tokens[i].atrulers_child = 1;else if (l = checkAtrule(i)) tokens[i].atrulers_child = 2;else if (l = checkRuleset(i)) tokens[i].atrulers_child = 3;else break;
i += l;
}
if (i < tokensLength) tokens[i].atrulers_end = 1;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* @returns {Array} `['atrulers', x]`
*/
function getAtrulers() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `{`.
pos++;
content = content.concat(getSC());
while (pos < tokensLength && !tokens[pos].atrulers_end) {
var childType = tokens[pos].atrulers_child;
if (childType === 1) content = content.concat(getSC());else if (childType === 2) content.push(getAtrule());else if (childType === 3) content.push(getRuleset());else break;
}
content = content.concat(getSC());
var end = getLastPosition(content, line, column, 1);
// Skip `}`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkAtrules(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
return i - start;
}
/**
* @returns {Array} `['atrules', ['atkeyword', x], y]`
*/
function getAtrules() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a block (e.g. `{...}`).
* @param {Number} i Token's index number
* @returns {Number} Length of the block
*/
function checkBlock(i) {
return i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket ? tokens[i].right - i + 1 : 0;
}
/**
* Get node with a block
* @returns {Array} `['block', x]`
*/
function getBlock() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = tokens[pos].right;
var content = [];
// Skip `{`.
pos++;
while (pos < end) {
if (checkBlockdecl(pos)) content = content.concat(getBlockdecl());else throwError(pos);
}
var end_ = getLastPosition(content, line, column, 1);
pos = end + 1;
return newNode(type, content, line, column, end_);
}
/**
* Check if token is part of a declaration (property-value pair)
* @param {Number} i Token's index number
* @returns {Number} Length of the declaration
*/
function checkBlockdecl(i) {
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkBlockdecl1(i)) tokens[i].bd_type = 1;else if (l = checkBlockdecl2(i)) tokens[i].bd_type = 2;else if (l = checkBlockdecl3(i)) tokens[i].bd_type = 3;else if (l = checkBlockdecl4(i)) tokens[i].bd_type = 4;else return 0;
return l;
}
/**
* @returns {Array}
*/
function getBlockdecl() {
var childType = tokens[pos].bd_type;
if (childType === 1) return getBlockdecl1();
if (childType === 2) return getBlockdecl2();
if (childType === 3) return getBlockdecl3();
if (childType === 4) return getBlockdecl4();
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkBlockdecl1(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkCondition(i)) tokens[i].bd_kind = 1;else if (l = checkExtend(i)) tokens[i].bd_kind = 6;else if (l = checkRuleset(i)) tokens[i].bd_kind = 2;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 3;else if (l = checkAtrule(i)) tokens[i].bd_kind = 4;else if (l = checkInclude(i)) tokens[i].bd_kind = 5;else return 0;
i += l;
if (i < tokensLength && (l = checkDeclDelim(i))) i += l;else return 0;
if (l = checkSC(i)) i += l;else return 0;
return i - start;
}
/**
* @returns {Array}
*/
function getBlockdecl1() {
var sc = getSC();
var content = void 0;
switch (tokens[pos].bd_kind) {
case 1:
content = getCondition();
break;
case 2:
content = getRuleset();
break;
case 3:
content = getDeclaration();
break;
case 4:
content = getAtrule();
break;
case 5:
content = getInclude();
break;
case 6:
content = getExtend();
break;
}
return sc.concat(content, getDeclDelim(), getSC());
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkBlockdecl2(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkCondition(i)) tokens[i].bd_kind = 1;else if (l = checkExtend(i)) tokens[i].bd_kind = 3;else if (l = checkRuleset(i)) tokens[i].bd_kind = 6;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 4;else if (l = checkAtrule(i)) tokens[i].bd_kind = 5;else if (l = checkInclude(i)) tokens[i].bd_kind = 2;else return 0;
i += l;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* @returns {Array}
*/
function getBlockdecl2() {
var sc = getSC();
var content = void 0;
switch (tokens[pos].bd_kind) {
case 1:
content = getCondition();
break;
case 2:
content = getInclude();
break;
case 3:
content = getExtend();
break;
case 4:
content = getDeclaration();
break;
case 5:
content = getAtrule();
break;
case 6:
content = getRuleset();
break;
}
return sc.concat(content, getSC());
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkBlockdecl3(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkDeclDelim(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* @returns {Array} `[s0, ['declDelim'], s1]` where `s0` and `s1` are
* are optional whitespaces.
*/
function getBlockdecl3() {
return [].concat(getSC(), getDeclDelim(), getSC());
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkBlockdecl4(i) {
return checkSC(i);
}
/**
* @returns {Array}
*/
function getBlockdecl4() {
return getSC();
}
/**
* Check if token is part of text inside square brackets, e.g. `[1]`
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkBrackets(i) {
if (i >= tokensLength) return 0;
var start = i;
// Skip `[`.
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
if (i < tokens[start].right) {
var l = checkTsets(i);
if (l) i += l;else return 0;
}
// Skip `]`.
i++;
return i - start;
}
/**
* Get node with text inside square brackets, e.g. `[1]`
* @returns {Node}
*/
function getBrackets() {
var type = NodeType.BracketsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var right = token.right;
var content = [];
// Skip `[`.
pos++;
if (pos < right) {
content = getTsets();
}
var end = getLastPosition(content, line, column, 1);
// Skip `]`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a class selector (e.g. `.abc`)
* @param {Number} i Token's index number
* @returns {Number} Length of the class selector
*/
function checkClass(i) {
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].class_l) return tokens[i].class_l;
if (tokens[i++].type === TokenType.FullStop) {
if (l = checkInterpolatedVariable(i)) tokens[i].class_child = 1;else if (l = checkIdent(i)) tokens[i].class_child = 2;else return 0;
tokens[i].class_l = l + 1;
return l + 1;
}
return 0;
}
/**
* Get node with a class selector
* @returns {Array} `['class', ['ident', x]]` where x is a class's
* identifier (without `.`, e.g. `abc`).
*/
function getClass() {
var type = NodeType.ClassType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `.`
pos++;
var childType = tokens[pos].class_child;
if (childType === 1) content.push(getInterpolatedVariable());else content.push(getIdent());
return newNode(type, content, line, column);
}
function checkCombinator(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkCombinator1(i)) tokens[i].combinatorType = 1;else if (l = checkCombinator2(i)) tokens[i].combinatorType = 2;else if (l = checkCombinator3(i)) tokens[i].combinatorType = 3;else if (l = checkCombinator4(i)) tokens[i].combinatorType = 4;
return l;
}
function getCombinator() {
var type = tokens[pos].combinatorType;
if (type === 1) return getCombinator1();
if (type === 2) return getCombinator2();
if (type === 3) return getCombinator3();
if (type === 4) return getCombinator4();
}
/**
* (1) `>>>`
*
* @param {Number} i
* @return {Number}
*/
function checkCombinator1(i) {
if (i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign) return 3;
return 0;
}
/**
* @return {Node}
*/
function getCombinator1() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '>>>';
// Skip combinator
pos += 3;
return newNode(type, content, line, column);
}
/**
* (1) `||`
* (2) `>>`
*
* @param {Number} i
* @return {Number}
*/
function checkCombinator2(i) {
if (i + 1 >= tokensLength) return 0;
if (tokens[i].type === TokenType.VerticalLine && tokens[i + 1].type === TokenType.VerticalLine) return 2;
if (tokens[i].type === TokenType.GreaterThanSign && tokens[i + 1].type === TokenType.GreaterThanSign) return 2;
return 0;
}
/**
* @return {Node}
*/
function getCombinator2() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '' + token.value + tokens[pos + 1].value;
// Skip combinator
pos += 2;
return newNode(type, content, line, column);
}
/**
* (1) `>`
* (2) `+`
* (3) `~`
*
* @param {Number} i
* @return {Number}
*/
function checkCombinator3(i) {
var type = tokens[i].type;
if (type === TokenType.PlusSign || type === TokenType.GreaterThanSign || type === TokenType.Tilde) return 1;else return 0;
}
/**
* @return {Node}
*/
function getCombinator3() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
// Skip combinator
pos++;
return newNode(type, content, line, column);
}
/**
* (1) `/panda/`
*/
function checkCombinator4(i) {
var start = i;
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
var l = void 0;
if (l = checkIdent(i)) i += l;else return 0;
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getCombinator4() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `/`.
pos++;
var ident = getIdent();
// Skip `/`.
pos++;
var content = '/' + ident.content + '/';
return newNode(type, content, line, column);
}
/**
* Check if token is a multiline comment.
* @param {Number} i Token's index number
* @returns {Number} `1` if token is a multiline comment, otherwise `0`
*/
function checkCommentML(i) {
return i < tokensLength && tokens[i].type === TokenType.CommentML ? 1 : 0;
}
/**
* Get node with a multiline comment
* @returns {Array} `['commentML', x]` where `x`
* is the comment's text (without `/*` and `* /`).
*/
function getCommentML() {
var type = NodeType.CommentMLType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos].value.substring(2);
var l = content.length;
if (content.charAt(l - 2) === '*' && content.charAt(l - 1) === '/') content = content.substring(0, l - 2);
var end = getLastPosition(content, line, column, 2);
if (end[0] === line) end[1] += 2;
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a single-line comment.
* @param {Number} i Token's index number
* @returns {Number} `1` if token is a single-line comment, otherwise `0`
*/
function checkCommentSL(i) {
return i < tokensLength && tokens[i].type === TokenType.CommentSL ? 1 : 0;
}
/**
* Get node with a single-line comment.
* @returns {Array}
*/
function getCommentSL() {
var type = NodeType.CommentSLType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos++].value.substring(2);
var end = getLastPosition(content, line, column + 2);
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a condition.
* @param {Number} i Token's index number
* @return {Number} Length of the condition
*/
function checkCondition(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if ((l = checkIdent(i)) && tokens[i].value === 'when') i += l;else return 0;
while (i < tokensLength) {
if (l = checkBlock(i)) {
tokens[i].condition_child = 0;
break;
} else if (l = checkFunction(i)) tokens[i].condition_child = 1;else if (l = checkBrackets(i)) tokens[i].condition_child = 2;else if (l = checkParentheses(i)) tokens[i].condition_child = 3;else if (l = checkVariable(i)) tokens[i].condition_child = 4;else if (l = checkIdent(i)) tokens[i].condition_child = 5;else if (l = checkNumber(i)) tokens[i].condition_child = 6;else if (l = checkDelim(i)) tokens[i].condition_child = 7;else if (l = checkOperator(i)) tokens[i].condition_child = 8;else if (l = checkCombinator(i)) tokens[i].condition_child = 9;else if (l = checkSC(i)) tokens[i].condition_child = 10;else if (l = checkString(i)) tokens[i].condition_child = 11;else return 0;
i += l;
}
return i - start;
}
/**
* Get node with a condition.
* @returns {Array} `['condition', x]`
*/
function getCondition() {
var type = NodeType.ConditionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(getIdent());
while (pos < tokensLength) {
var childType = tokens[pos].condition_child;
if (childType === 0) break;else if (childType === 1) content.push(getFunction());else if (childType === 2) content.push(getBrackets());else if (childType === 3) content.push(getParentheses());else if (childType === 4) content.push(getVariable());else if (childType === 5) content.push(getIdent());else if (childType === 6) content.push(getNumber());else if (childType === 7) content.push(getDelim());else if (childType === 8) content.push(getOperator());else if (childType === 9) content.push(getCombinator());else if (childType === 10) content = content.concat(getSC());else if (childType === 11) content.push(getString());
}
return newNode(type, content, line, column);
}
/**
* Check if token is part of a declaration (property-value pair)
* @param {Number} i Token's index number
* @returns {Number} Length of the declaration
*/
function checkDeclaration(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkProperty(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkPropertyDelim(i)) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkValue(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a declaration
* @returns {Array} `['declaration', ['property', x], ['propertyDelim'],
* ['value', y]]`
*/
function getDeclaration() {
var type = NodeType.DeclarationType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getProperty(), getSC(), getPropertyDelim(), getSC(), getValue());
return newNode(type, content, line, column);
}
/**
* Check if token is a semicolon
* @param {Number} i Token's index number
* @returns {Number} `1` if token is a semicolon, otherwise `0`
*/
function checkDeclDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Semicolon ? 1 : 0;
}
/**
* Get node with a semicolon
* @returns {Array} `['declDelim']`
*/
function getDeclDelim() {
var type = NodeType.DeclDelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ';';
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a comma
* @param {Number} i Token's index number
* @returns {Number} `1` if token is a comma, otherwise `0`
*/
function checkDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Comma ? 1 : 0;
}
/**
* Get node with a comma
* @returns {Array} `['delim']`
*/
function getDelim() {
var type = NodeType.DelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ',';
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is part of a number with dimension unit (e.g. `10px`)
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkDimension(i) {
var ln = checkNumber(i);
var li = void 0;
if (i >= tokensLength || !ln || i + ln >= tokensLength) return 0;
return (li = checkUnit(i + ln)) ? ln + li : 0;
}
/**
* Get node of a number with dimension unit
* @return {Node}
*/
function getDimension() {
var type = NodeType.DimensionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getNumber(), getUnit()];
return newNode(type, content, line, column);
}
/**
* Check if token is part of an escaped string (e.g. `~"ms:something"`).
* @param {Number} i Token's index number
* @returns {Number} Length of the string (including `~` and quotes)
*/
function checkEscapedString(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.Tilde && (l = checkString(i + 1))) return i + l - start;else return 0;
}
/**
* Get node with an escaped string
* @returns {Array} `['escapedString', ['string', x]]` where `x` is a string
* without `~` but with quotes
*/
function getEscapedString() {
var type = NodeType.EscapedStringType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
pos++;
var content = tokens[pos].value;
var end = getLastPosition(content, line, column + 1);
pos++;
return newNode(type, content, line, column, end);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkExpression(i) {
var start = i;
if (i >= tokensLength || tokens[i++].value !== 'expression' || i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) {
return 0;
}
return tokens[i].right - start + 1;
}
/**
* @returns {Array}
*/
function getExpression() {
var type = NodeType.ExpressionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
pos++;
var content = joinValues(pos + 1, tokens[pos].right - 1);
var end = getLastPosition(content, line, column, 1);
if (end[0] === line) end[1] += 11;
pos = tokens[pos].right + 1;
return newNode(type, content, line, column, end);
}
function checkExtend(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkExtend1(i)) tokens[i].extendType = 1;else if (l = checkExtend2(i)) tokens[i].extendType = 2;else return 0;
return l;
}
function getExtend() {
var childType = tokens[pos].extendType;
if (childType === 1) return getExtend1();
if (childType === 2) return getExtend2();
}
/**
* (1) `selector:extend(selector) {...}`
*/
function checkExtend1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkExtendSelector(i)) i += l;else return 0;
if (tokens[i + 1] && tokens[i + 1].value === 'extend' && (l = checkPseudoc(i))) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
function getExtend1() {
var type = NodeType.ExtendType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getExtendSelector(), getPseudoc(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* (1) `selector:extend(selector)`
*/
function checkExtend2(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkExtendSelector(i)) i += l;else return 0;
if (tokens[i + 1] && tokens[i + 1].value === 'extend' && (l = checkPseudoc(i))) i += l;else return 0;
return i - start;
}
function getExtend2() {
var type = NodeType.ExtendType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getExtendSelector(), getPseudoc());
return newNode(type, content, line, column);
}
function checkExtendSelector(i) {
var l = void 0;
if (l = checkParentSelectorWithExtension(i)) tokens[i].extend_type = 1;else if (l = checkIdent(i)) tokens[i].extend_type = 2;else if (l = checkClass(i)) tokens[i].extend_type = 3;else if (l = checkShash(i)) tokens[i].extend_type = 4;
return l;
}
function getExtendSelector() {
var childType = tokens[pos].extend_type;
if (childType === 1) return getParentSelectorWithExtension();
if (childType === 2) return [getIdent()];
if (childType === 3) return [getClass()];
if (childType === 4) return [getShash()];
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkFunction(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
return i < tokensLength && tokens[i].type === TokenType.LeftParenthesis ? tokens[i].right - start + 1 : 0;
}
/**
* @returns {Array}
*/
function getFunction() {
var type = NodeType.FunctionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getIdent(), getArguments());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an identifier
* @param {Number} i Token's index number
* @returns {Number} Length of the identifier
*/
function checkIdent(i) {
var start = i;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.HyphenMinus) i++;
if (tokens[i].type === TokenType.LowLine || tokens[i].type === TokenType.Identifier) i++;else return 0;
for (; i < tokensLength; i++) {
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
}
tokens[start].ident_last = i - 1;
return i - start;
}
/**
* Get node with an identifier
* @returns {Array} `['ident', x]` where `x` is identifier's name
*/
function getIdent() {
var type = NodeType.IdentType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, tokens[pos].ident_last);
pos = tokens[pos].ident_last + 1;
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @returns {number} Length of the identifier
*/
function checkPartialIdent(i) {
var start = i;
if (i >= tokensLength) return 0;
for (; i < tokensLength; i++) {
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
}
tokens[start].ident_last = i - 1;
return i - start;
}
/**
* Check if token is part of `!important` word
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkImportant(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].value === 'important') {
tokens[start].importantEnd = i;
return i - start + 1;
} else {
return 0;
}
}
/**
* Get node with `!important` word
* @returns {Array} `['important', sc]` where `sc` is optional whitespace
*/
function getImportant() {
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, token.importantEnd);
pos = token.importantEnd + 1;
return newNode(NodeType.ImportantType, content, line, column);
}
/**
* Check if token is part of an include (`@include` or `@extend` directive).
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkInclude(i) {
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkInclude1(i)) tokens[i].include_type = 1;else if (l = checkInclude2(i)) tokens[i].include_type = 2;
return l;
}
/**
* Get node with included mixin
* @returns {Array} `['include', x]`
*/
function getInclude() {
var type = tokens[pos].include_type;
if (type === 1) return getInclude1();
if (type === 2) return getInclude2();
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkInclude1(i) {
var start = i;
var l = void 0;
if (l = checkClass(i) || checkShash(i)) i += l;else return 0;
while (i < tokensLength) {
if (l = checkClass(i) || checkShash(i) || checkSC(i)) i += l;else if (tokens[i].type === TokenType.GreaterThanSign) i++;else break;
}
if (l = checkArguments(i)) i += l;else return 0;
if (i < tokensLength && (l = checkSC(i))) i += l;
if (i < tokensLength && (l = checkImportant(i))) i += l;
return i - start;
}
/**
* @returns {Array} `['include', x]`
*/
function getInclude1() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(checkClass(pos) ? getClass() : getShash());
while (pos < tokensLength) {
if (checkClass(pos)) content.push(getClass());else if (checkShash(pos)) content.push(getShash());else if (checkSC(pos)) content = content.concat(getSC());else if (checkOperator(pos)) content.push(getOperator());else break;
}
content.push(getArguments());
content = content.concat(getSC());
if (checkImportant(pos)) content.push(getImportant());
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkInclude2(i) {
var start = i;
var l = void 0;
if (l = checkClass(i) || checkShash(i)) i += l;else return 0;
while (i < tokensLength) {
if (l = checkClass(i) || checkShash(i) || checkSC(i)) i += l;else if (tokens[i].type === TokenType.GreaterThanSign) i++;else break;
}
return i - start;
}
/**
* @returns {Array} `['include', x]`
*/
function getInclude2() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(checkClass(pos) ? getClass() : getShash());
while (pos < tokensLength) {
if (checkClass(pos)) content.push(getClass());else if (checkShash(pos)) content.push(getShash());else if (checkSC(pos)) content = content.concat(getSC());else if (checkOperator(pos)) content.push(getOperator());else break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is part of LESS interpolated variable
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkInterpolatedVariable(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.CommercialAt || !tokens[i + 1] || tokens[i + 1].type !== TokenType.LeftCurlyBracket) {
return 0;
}
i += 2;
if (l = checkIdent(i)) i += l;else return 0;
return tokens[i].type === TokenType.RightCurlyBracket ? i - start + 1 : 0;
}
/**
* Get node with LESS interpolated variable
* @returns {Array} `['interpolatedVariable', x]`
*/
function getInterpolatedVariable() {
var type = NodeType.InterpolatedVariableType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `@{`:
pos += 2;
content.push(getIdent());
var end = getLastPosition(content, line, column, 1);
// Skip `}`:
pos++;
return newNode(type, content, line, column, end);
}
function checkKeyframesBlock(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkKeyframesSelector(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
function getKeyframesBlock() {
var type = NodeType.RulesetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getKeyframesSelector(), getSC(), getBlock());
return newNode(type, content, line, column);
}
function checkKeyframesBlocks(i) {
var start = i;
var l = void 0;
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlock(i)) i += l;else return 0;
while (tokens[i].type !== TokenType.RightCurlyBracket) {
if (l = checkSC(i)) i += l;else if (l = checkKeyframesBlock(i)) i += l;else break;
}
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
return i - start;
}
function getKeyframesBlocks() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var keyframesBlocksEnd = token.right;
var content = [];
// Skip `{`.
pos++;
while (pos < keyframesBlocksEnd) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkKeyframesBlock(pos)) content.push(getKeyframesBlock());
}
var end = getLastPosition(content, line, column, 1);
// Skip `}`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a @keyframes rule.
* @param {Number} i Token's index number
* @return {Number} Length of the @keyframes rule
*/
function checkKeyframesRule(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
var atruleName = joinValues2(i - l, l);
if (atruleName.toLowerCase().indexOf('keyframes') === -1) return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getKeyframesRule() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdent(), getSC(), getKeyframesBlocks());
return newNode(type, content, line, column);
}
function checkKeyframesSelector(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) {
// Valid selectors are only `from` and `to`.
var selector = joinValues2(i, l);
if (selector !== 'from' && selector !== 'to') return 0;
i += l;
tokens[start].keyframesSelectorType = 1;
} else if (l = checkPercentage(i)) {
i += l;
tokens[start].keyframesSelectorType = 2;
} else {
return 0;
}
return i - start;
}
function getKeyframesSelector() {
var keyframesSelectorType = NodeType.KeyframesSelectorType;
var selectorType = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (token.keyframesSelectorType === 1) {
content.push(getIdent());
} else {
content.push(getPercentage());
}
var keyframesSelector = newNode(keyframesSelectorType, content, line, column);
return newNode(selectorType, [keyframesSelector], line, column);
}
/**
* Check if token is part of a LESS mixin
* @param {Number} i Token's index number
* @returns {Number} Length of the mixin
*/
function checkMixin(i) {
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkMixin1(i)) tokens[i].mixin_type = 1;else if (l = checkMixin2(i)) tokens[i].mixin_type = 2;else return 0;
return l;
}
/**
* @returns {Array}
*/
function getMixin() {
var type = tokens[pos].mixin_type;
if (type === 1) return getMixin1();
if (type === 2) return getMixin2();
}
function checkMixin1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkClass(i) || checkShash(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a mixin
* @returns {Array} `['mixin', x]`
*/
function getMixin1() {
var type = NodeType.MixinType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(checkClass(pos) ? getClass() : getShash());
content = content.concat(getSC());
if (checkArguments(pos)) content.push(getArguments());
content = content.concat(getSC());
if (checkBlock(pos)) content.push(getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a LESS mixin
* @param {Number} i Token's index number
* @returns {Number} Length of the mixin
*/
function checkMixin2(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkClass(i) || checkShash(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;
return i - start;
}
/**
* Get node with a mixin
* @returns {Array} `['mixin', x]`
*/
function getMixin2() {
var type = NodeType.MixinType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(checkClass(pos) ? getClass() : getShash());
content = content.concat(getSC());
if (checkArguments(pos)) content.push(getArguments());
return newNode(type, content, line, column);
}
/**
* Check if token is a namespace sign (`|`)
* @param {Number} i Token's index number
* @returns {Number} `1` if token is `|`, `0` if not
*/
function checkNamespace(i) {
return i < tokensLength && tokens[i].type === TokenType.VerticalLine ? 1 : 0;
}
/**
* Get node with a namespace sign
* @returns {Array} `['namespace']`
*/
function getNamespace() {
var type = NodeType.NamespaceType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '|';
pos++;
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkNmName2(i) {
if (tokens[i].type === TokenType.Identifier) return 1;else if (tokens[i].type !== TokenType.DecimalNumber) return 0;
i++;
return i < tokensLength && tokens[i].type === TokenType.Identifier ? 2 : 1;
}
/**
* @returns {String}
*/
function getNmName2() {
var s = tokens[pos].value;
if (tokens[pos++].type === TokenType.DecimalNumber && pos < tokensLength && tokens[pos].type === TokenType.Identifier) s += tokens[pos++].value;
return s;
}
/**
* Check if token is part of a number
* @param {Number} i Token's index number
* @returns {Number} Length of number
*/
function checkNumber(i) {
if (i >= tokensLength) return 0;
if (tokens[i].number_l) return tokens[i].number_l;
// `10`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && (!tokens[i + 1] || tokens[i + 1] && tokens[i + 1].type !== TokenType.FullStop)) {
tokens[i].number_l = 1;
return 1;
}
// `10.`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && (!tokens[i + 2] || tokens[i + 2].type !== TokenType.DecimalNumber)) {
tokens[i].number_l = 2;
return 2;
}
// `.10`:
if (i < tokensLength && tokens[i].type === TokenType.FullStop && tokens[i + 1].type === TokenType.DecimalNumber) {
tokens[i].number_l = 2;
return 2;
}
// `10.10`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && tokens[i + 2] && tokens[i + 2].type === TokenType.DecimalNumber) {
tokens[i].number_l = 3;
return 3;
}
return 0;
}
/**
* Get node with number
* @returns {Array} `['number', x]` where `x` is a number converted
* to string.
*/
function getNumber() {
var type = NodeType.NumberType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var l = tokens[pos].number_l;
var content = '';
for (var j = 0; j < l; j++) {
content += tokens[pos + j].value;
}
pos += l;
return newNode(type, content, line, column);
}
/**
* Check if token is an operator (`/`, `,`, `:`, `=`, `>`, `<` or `*`)
* @param {Number} i Token's index number
* @returns {Number} `1` if token is an operator, otherwise `0`
*/
function checkOperator(i) {
if (i >= tokensLength) return 0;
switch (tokens[i].type) {
case TokenType.Solidus:
case TokenType.Comma:
case TokenType.Colon:
case TokenType.EqualsSign:
case TokenType.LessThanSign:
case TokenType.GreaterThanSign:
case TokenType.Asterisk:
return 1;
}
return 0;
}
/**
* Get node with an operator
* @returns {Array} `['operator', x]` where `x` is an operator converted
* to string.
*/
function getOperator() {
var type = NodeType.OperatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is part of text inside parentheses, e.g. `(1)`
* @param {Number} i Token's index number
* @return {Number}
*/
function checkParentheses(i) {
if (i >= tokensLength) return 0;
var start = i;
var right = tokens[i].right;
// Skip `(`.
if (tokens[i].type === TokenType.LeftParenthesis) i++;else return 0;
if (i < right) {
var l = checkTsets(i);
if (l) i += l;else return 0;
}
// Skip `)`.
i++;
return i - start;
}
/**
* Get node with text inside parentheses, e.g. `(1)`
* @return {Node}
*/
function getParentheses() {
var type = NodeType.ParenthesesType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var right = token.right;
var content = [];
// Skip `(`.
pos++;
if (pos < right) {
content = getTsets();
}
var end = getLastPosition(content, line, column, 1);
// Skip `)`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is a parent selector (`&`).
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkParentSelector(i) {
return i < tokensLength && tokens[i].type === TokenType.Ampersand ? 1 : 0;
}
/**
* Get node with a parent selector
* @returns {Array} `['parentSelector']`
*/
function getParentSelector() {
var type = NodeType.ParentSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '&';
pos++;
return newNode(type, content, line, column);
}
function checkParentSelectorExtension(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
while (i < tokensLength) {
if (l = checkNumber(i) || checkPartialIdent(i)) i += l;else break;
}
return i - start;
}
function getParentSelectorExtension() {
var type = NodeType.ParentSelectorExtensionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
while (pos < tokensLength) {
if (checkNumber(pos)) {
content.push(getNumber());
} else if (checkPartialIdent(pos)) {
content.push(getIdent());
} else break;
}
return newNode(type, content, line, column);
}
function checkParentSelectorWithExtension(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkParentSelector(i)) i += l;else return 0;
if (l = checkParentSelectorExtension(i)) i += l;
return i - start;
}
function getParentSelectorWithExtension() {
var content = [getParentSelector()];
if (checkParentSelectorExtension(pos)) content.push(getParentSelectorExtension());
return content;
}
/**
* Check if token is part of a number with percent sign (e.g. `10%`)
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkPercentage(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkNumber(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
// Skip `%`.
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
return i - start;
}
/**
* Get node of number with percent sign
* @returns {Array} `['percentage', ['number', x]]` where `x` is a number
* (without percent sign) converted to string.
*/
function getPercentage() {
var type = NodeType.PercentageType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getNumber()];
var end = getLastPosition(content, line, column, 1);
// Skip `%`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkProgid(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (joinValues2(i, 6) === 'progid:DXImageTransform.Microsoft.') i += 6;else return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type === TokenType.LeftParenthesis) {
tokens[start].progid_end = tokens[i].right;
i = tokens[i].right + 1;
} else return 0;
return i - start;
}
/**
* @returns {Array}
*/
function getProgid() {
var type = NodeType.ProgidType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var progid_end = token.progid_end;
var content = joinValues(pos, progid_end);
pos = progid_end + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of a property
* @param {Number} i Token's index number
* @returns {Number} Length of the property
*/
function checkProperty(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkVariable(i) || checkIdent(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a property
* @returns {Array} `['property', x]`
*/
function getProperty() {
var type = NodeType.PropertyType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkVariable(pos)) {
content.push(getVariable());
} else {
content.push(getIdent());
}
return newNode(type, content, line, column);
}
/**
* Check if token is a colon
* @param {Number} i Token's index number
* @returns {Number} `1` if token is a colon, otherwise `0`
*/
function checkPropertyDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Colon ? 1 : 0;
}
/**
* Get node with a colon
* @returns {Array} `['propertyDelim']`
*/
function getPropertyDelim() {
var type = NodeType.PropertyDelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ':';
// Skip `:`.
pos++;
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkPseudo(i) {
return checkPseudoe(i) || checkPseudoc(i);
}
/**
* @returns {Array}
*/
function getPseudo() {
if (checkPseudoe(pos)) return getPseudoe();
if (checkPseudoc(pos)) return getPseudoc();
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkPseudoe(i) {
var l = void 0;
// Check `::`
if (i >= tokensLength || tokens[i].type !== TokenType.Colon || i + 1 >= tokensLength || tokens[i + 1].type !== TokenType.Colon) return 0;
if (l = checkPseudoElement1(i)) tokens[i].pseudoElementType = 1;else if (l = checkPseudoElement2(i)) tokens[i].pseudoElementType = 2;else return 0;
return l;
}
/**
* @returns {Node}
*/
function getPseudoe() {
var childType = tokens[pos].pseudoElementType;
if (childType === 1) return getPseudoElement1();
if (childType === 2) return getPseudoElement2();
}
/**
* (1) `::slotted(selector)`
* (2) `::slotted(selector, selector)`
*/
function checkPseudoElement1(i) {
var start = i;
var l = void 0;
// Skip `::`.
i += 2;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
/**
* (1) `::slotted(selector)`
* (2) `::slotted(selector, selector)`
*/
function getPseudoElement1() {
var type = NodeType.PseudoeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `::`.
pos += 2;
content.push(getIdent());
{
var _type = NodeType.ArgumentsType;
var _token = tokens[pos];
var _line = _token.ln;
var _column = _token.col;
// Skip `(`.
pos++;
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
var end = getLastPosition(selectorContent, _line, _column, 1);
var args = newNode(_type, selectorContent, _line, _column, end);
content.push(args);
// Skip `)`.
pos++;
}
return newNode(type, content, line, column);
}
function checkPseudoElement2(i) {
var start = i;
var l = void 0;
// Skip `::`.
i += 2;
if (l = checkInterpolatedVariable(i) || checkIdent(i)) i += l;else return 0;
return i - start;
}
/**
* @returns {Array}
*/
function getPseudoElement2() {
var type = NodeType.PseudoeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `::`.
pos += 2;
var content = [];
if (checkInterpolatedVariable(pos)) {
content.push(getInterpolatedVariable());
} else {
content.push(getIdent());
}
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkPseudoc(i) {
var l = void 0;
if (i >= tokensLength || tokens[i].type !== TokenType.Colon) return 0;
if (l = checkPseudoClass3(i)) tokens[i].pseudoClassType = 3;else if (l = checkPseudoClass4(i)) tokens[i].pseudoClassType = 4;else if (l = checkPseudoClass5(i)) tokens[i].pseudoClassType = 5;else if (l = checkPseudoClass1(i)) tokens[i].pseudoClassType = 1;else if (l = checkPseudoClass2(i)) tokens[i].pseudoClassType = 2;else if (l = checkPseudoClass6(i)) tokens[i].pseudoClassType = 6;else return 0;
return l;
}
function getPseudoc() {
var childType = tokens[pos].pseudoClassType;
if (childType === 1) return getPseudoClass1();
if (childType === 2) return getPseudoClass2();
if (childType === 3) return getPseudoClass3();
if (childType === 4) return getPseudoClass4();
if (childType === 5) return getPseudoClass5();
if (childType === 6) return getPseudoClass6();
}
/**
* (1) `:not(selector)`
* (2) `:extend(selector, selector)`
*/
function checkPseudoClass1(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
/**
* (-) `:not(panda)`
*/
function getPseudoClass1() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content.push(getIdent());
{
var _type2 = NodeType.ArgumentsType;
var _token2 = tokens[pos];
var _line2 = _token2.ln;
var _column2 = _token2.col;
// Skip `(`.
pos++;
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
var end = getLastPosition(selectorContent, _line2, _column2, 1);
var args = newNode(_type2, selectorContent, _line2, _column2, end);
content.push(args);
// Skip `)`.
pos++;
}
return newNode(type, content, line, column);
}
/**
* (1) `:nth-child(odd)`
* (2) `:nth-child(even)`
* (3) `:lang(de-DE)`
*/
function checkPseudoClass2(i) {
var start = i;
var l = 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkIdent(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass2() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content.push(getIdent());
{
// Skip `(`.
pos++;
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [].concat(getSC(), getIdent(), getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
}
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(-3n + 2)`
*/
function checkPseudoClass3(i) {
var start = i;
var l = 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.DecimalNumber) i++;
if (i >= tokensLength) return 0;
if (tokens[i].value === 'n') i++;else return 0;
if (l = checkSC(i)) i += l;
if (i >= tokensLength) return 0;
if (tokens[i].value === '+' || tokens[i].value === '-') i++;else return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass3() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content.push(getIdent());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkNumber(pos)) value.push(getNumber());
{
var _l = tokens[pos].ln;
var _c = tokens[pos].col;
var _content = tokens[pos].value;
var ident = newNode(NodeType.IdentType, _content, _l, _c);
value.push(ident);
pos++;
}
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
value = value.concat(getSC());
if (checkNumber(pos)) value.push(getNumber());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(-3n)`
*/
function checkPseudoClass4(i) {
var start = i;
var l = 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (tokens[i].type === TokenType.DecimalNumber) i++;
if (tokens[i].value === 'n') i++;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass4() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content.push(getIdent());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkNumber(pos)) value.push(getNumber());
if (checkIdent(pos)) value.push(getIdent());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(+8)`
*/
function checkPseudoClass5(i) {
var start = i;
var l = 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass5() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content.push(getIdent());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkNumber(pos)) value.push(getNumber());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:checked`
*/
function checkPseudoClass6(i) {
var start = i;
var l = 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkInterpolatedVariable(i)) i += l;else if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
function getPseudoClass6() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
var ident = checkInterpolatedVariable(pos) ? getInterpolatedVariable() : getIdent();
content.push(ident);
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkRuleset(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
function getRuleset() {
var type = NodeType.RulesetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getSelectorsGroup(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is marked as a space (if it's a space or a tab
* or a line break).
* @param {Number} i
* @returns {Number} Number of spaces in a row starting with the given token.
*/
function checkS(i) {
return i < tokensLength && tokens[i].ws ? tokens[i].ws_last - i + 1 : 0;
}
/**
* Get node with spaces
* @returns {Array} `['s', x]` where `x` is a string containing spaces
*/
function getS() {
var type = NodeType.SType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, tokens[pos].ws_last);
pos = tokens[pos].ws_last + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is a space or a comment.
* @param {Number} i Token's index number
* @returns {Number} Number of similar (space or comment) tokens
* in a row starting with the given token.
*/
function checkSC(i) {
if (i >= tokensLength) return 0;
var l = void 0;
var lsc = 0;
while (i < tokensLength) {
if (!(l = checkS(i)) && !(l = checkCommentML(i)) && !(l = checkCommentSL(i))) break;
i += l;
lsc += l;
}
return lsc || 0;
}
/**
* Get node with spaces and comments
* @returns {Array} Array containing nodes with spaces (if there are any)
* and nodes with comments (if there are any):
* `[['s', x]*, ['comment', y]*]` where `x` is a string of spaces
* and `y` is a comment's text (without `/*` and `* /`).
*/
function getSC() {
var sc = [];
if (pos >= tokensLength) return sc;
while (pos < tokensLength) {
if (checkS(pos)) sc.push(getS());else if (checkCommentML(pos)) sc.push(getCommentML());else if (checkCommentSL(pos)) sc.push(getCommentSL());else break;
}
return sc;
}
/**
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
* a simple selector
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkShash(i) {
var l = void 0;
if (i >= tokensLength || tokens[i].type !== TokenType.NumberSign) return 0;
if (l = checkInterpolatedVariable(i + 1) || checkIdent(i + 1)) return l + 1;else return 0;
}
/**
* Get node with a hexadecimal number (e.g. `#fff`) inside a simple
* selector
* @returns {Array} `['shash', x]` where `x` is a hexadecimal number
* converted to string (without `#`, e.g. `fff`)
*/
function getShash() {
var type = NodeType.ShashType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `#`.
pos++;
if (checkInterpolatedVariable(pos)) content.push(getInterpolatedVariable());else content.push(getIdent());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a string (text wrapped in quotes)
* @param {Number} i Token's index number
* @returns {Number} `1` if token is part of a string, `0` if not
*/
function checkString(i) {
if (i >= tokensLength) {
return 0;
}
if (tokens[i].type === TokenType.StringSQ || tokens[i].type === TokenType.StringDQ) {
return 1;
}
return 0;
}
/**
* Get string's node
* @returns {Array} `['string', x]` where `x` is a string (including
* quotes).
*/
function getString() {
var type = NodeType.StringType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Validate stylesheet: it should consist of any number (0 or more) of
* rulesets (sets of rules with selectors), @-rules, whitespaces or
* comments.
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkStylesheet(i) {
var start = i;
var l = void 0;
// Check every token:
while (i < tokensLength) {
if (l = checkSC(i) || checkRuleset(i) || checkDeclaration(i) || checkDeclDelim(i) || checkAtrule(i) || checkMixin(i)) i += l;else throwError(i);
}
return i - start;
}
/**
* @returns {Array} `['stylesheet', x]` where `x` is all stylesheet's
* nodes.
*/
function getStylesheet() {
var type = NodeType.StylesheetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
while (pos < tokensLength) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkRuleset(pos)) content.push(getRuleset());else if (checkDeclaration(pos)) content.push(getDeclaration());else if (checkDeclDelim(pos)) content.push(getDeclDelim());else if (checkAtrule(pos)) content.push(getAtrule());else if (checkMixin(pos)) content.push(getMixin());else throwError(pos);
}
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkTset(i) {
var l = void 0;
if (l = checkVhash(i)) tokens[i].tset_child = 1;else if (l = checkAny(i)) tokens[i].tset_child = 2;else if (l = checkSC(i)) tokens[i].tset_child = 3;else if (l = checkOperator(i)) tokens[i].tset_child = 4;
return l;
}
/**
* @returns {Array}
*/
function getTset() {
var childType = tokens[pos].tset_child;
if (childType === 1) return getVhash();
if (childType === 2) return getAny();
if (childType === 3) return getSC();
if (childType === 4) return getOperator();
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkTsets(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
while (l = checkTset(i)) {
i += l;
}
return i - start;
}
/**
* @returns {Array}
*/
function getTsets() {
var content = [];
var t = void 0;
while (checkTset(pos)) {
t = getTset();
if (typeof t.content === 'string') content.push(t);else content = content.concat(t);
}
return content;
}
/**
* Check if token is an unary (arithmetical) sign (`+` or `-`)
* @param {Number} i Token's index number
* @returns {Number} `1` if token is an unary sign, `0` if not
*/
function checkUnary(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.HyphenMinus || tokens[i].type === TokenType.PlusSign) {
return 1;
}
return 0;
}
/**
* Get node with an unary (arithmetical) sign (`+` or `-`)
* @returns {Array} `['unary', x]` where `x` is an unary sign
* converted to string.
*/
function getUnary() {
var type = NodeType.OperatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a unicode range (single or multiple <urange> nodes)
* @param {number} i Token's index
* @return {number} Unicode range node's length
*/
function checkUnicodeRange(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkUrange(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
var comma = checkDelim(i + spaceBefore);
if (!comma) break;
var spaceAfter = checkSC(i + spaceBefore + comma);
if (l = checkUrange(i + spaceBefore + comma + spaceAfter)) {
i += spaceBefore + comma + spaceAfter + l;
} else break;
}
return i - start;
}
/**
* Get a unicode range node
* @return {Node}
*/
function getUnicodeRange() {
var type = NodeType.UnicodeRangeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
while (pos < tokensLength) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkDelim(pos)) content.push(getDelim());else if (checkUrange(pos)) content.push(getUrange());else break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is unit
* @param {Number} i Token's index number
* @return {Number}
*/
function checkUnit(i) {
var units = ['em', 'ex', 'ch', 'rem', 'vh', 'vw', 'vmin', 'vmax', 'px', 'mm', 'q', 'cm', 'in', 'pt', 'pc', 'deg', 'grad', 'rad', 'turn', 's', 'ms', 'Hz', 'kHz', 'dpi', 'dpcm', 'dppx'];
return units.indexOf(tokens[i].value) !== -1 ? 1 : 0;
}
/**
* Get unit node of type ident
* @return {Node} An ident node containing the unit value
*/
function getUnit() {
var type = NodeType.IdentType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a u-range (part of a unicode-range)
* (1) `U+416`
* (2) `U+400-4ff`
* (3) `U+4??`
* @param {number} i Token's index
* @return {number} Urange node's length
*/
function checkUrange(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
// Check for unicode prefix (u+ or U+)
if (tokens[i].value === 'U' || tokens[i].value === 'u') i += 1;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].value === '+') i += 1;else return 0;
while (i < tokensLength) {
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = _checkUnicodeWildcard(i)) i += l;else break;
}
tokens[start].urangeEnd = i - 1;
return i - start;
}
/**
* Get a u-range node (part of a unicode-range)
* @return {Node}
*/
function getUrange() {
var startPos = pos;
var type = NodeType.UrangeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content = joinValues(startPos, tokens[startPos].urangeEnd);
pos = tokens[startPos].urangeEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check for unicode wildcard characters `?`
* @param {number} i Token's index
* @return {number} Wildcard length
*/
function _checkUnicodeWildcard(i) {
var start = i;
if (i >= tokensLength) return 0;
while (i < tokensLength) {
if (tokens[i].type === TokenType.QuestionMark) i += 1;else break;
}
return i - start;
}
/**
* Check if token is part of URI (e.g. `url('/css/styles.css')`)
* @param {Number} i Token's index number
* @returns {Number} Length of URI
*/
function checkUri(i) {
var start = i;
if (i >= tokensLength || tokens[i++].value !== 'url' || i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
return tokens[i].right - start + 1;
}
/**
* Get node with URI
* @returns {Array} `['uri', x]` where `x` is URI's nodes (without `url`
* and braces, e.g. `['string', ''/css/styles.css'']`).
*/
function getUri() {
var startPos = pos;
var uriExcluding = {};
var uri = void 0;
var token = void 0;
var l = void 0;
var raw = void 0;
pos += 2;
uriExcluding[TokenType.Space] = 1;
uriExcluding[TokenType.Tab] = 1;
uriExcluding[TokenType.Newline] = 1;
uriExcluding[TokenType.LeftParenthesis] = 1;
uriExcluding[TokenType.RightParenthesis] = 1;
if (checkUri1(pos)) {
uri = [].concat(getSC()).concat([getString()]).concat(getSC());
} else {
uri = getSC();
l = checkExcluding(uriExcluding, pos);
token = tokens[pos];
raw = newNode(NodeType.RawType, joinValues(pos, pos + l), token.ln, token.col);
uri.push(raw);
pos += l + 1;
uri = uri.concat(getSC());
}
token = tokens[startPos];
var line = token.ln;
var column = token.col;
var end = getLastPosition(uri, line, column, 1);
pos++;
return newNode(NodeType.UriType, uri, line, column, end);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkUri1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type !== TokenType.StringDQ && tokens[i].type !== TokenType.StringSQ) {
return 0;
}
i++;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* Check if token is part of a value
* @param {Number} i Token's index number
* @returns {Number} Length of the value
*/
function checkValue(i) {
var start = i;
var l = void 0;
var s = void 0;
var _i = void 0;
while (i < tokensLength) {
s = checkSC(i);
_i = i + s;
if (l = _checkValue(_i)) i += l + s;
if (!l || checkBlock(_i)) break;
}
tokens[start].value_end = i;
return i - start;
}
/**
* @returns {Array}
*/
function getValue() {
var type = NodeType.ValueType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = tokens[pos].value_end;
var content = [];
var _pos = void 0;
var s = void 0;
while (pos < end) {
s = checkSC(pos);
_pos = pos + s;
if (!_checkValue(_pos)) break;
if (s) content = content.concat(getSC());
content.push(_getValue());
}
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function _checkValue(i) {
var l = void 0;
if (l = checkEscapedString(i)) tokens[i].value_child = 1;else if (l = checkInterpolatedVariable(i)) tokens[i].value_child = 2;else if (l = checkVariable(i)) tokens[i].value_child = 3;else if (l = checkVhash(i)) tokens[i].value_child = 4;else if (l = checkBlock(i)) tokens[i].value_child = 5;else if (l = checkProgid(i)) tokens[i].value_child = 6;else if (l = checkAny(i)) tokens[i].value_child = 7;else if (l = checkAtkeyword(i)) tokens[i].value_child = 8;else if (l = checkOperator(i)) tokens[i].value_child = 9;else if (l = checkImportant(i)) tokens[i].value_child = 10;
return l;
}
/**
* @returns {Array}
*/
function _getValue() {
var childType = tokens[pos].value_child;
if (childType === 1) return getEscapedString();
if (childType === 2) return getInterpolatedVariable();
if (childType === 3) return getVariable();
if (childType === 4) return getVhash();
if (childType === 5) return getBlock();
if (childType === 6) return getProgid();
if (childType === 7) return getAny();
if (childType === 8) return getAtkeyword();
if (childType === 9) return getOperator();
if (childType === 10) return getImportant();
}
/**
* Check if token is part of LESS variable
* @param {Number} i Token's index number
* @returns {Number} Length of the variable
*/
function checkVariable(i) {
var l = void 0;
if (i >= tokensLength || tokens[i].type !== TokenType.CommercialAt) return 0;
if (tokens[i - 1] && tokens[i - 1].type === TokenType.CommercialAt && tokens[i - 2] && tokens[i - 2].type === TokenType.CommercialAt) return 0;
return (l = checkVariable(i + 1) || checkIdent(i + 1)) ? l + 1 : 0;
}
/**
* Get node with a variable
* @returns {Array} `['variable', ['ident', x]]` where `x` is
* a variable name.
*/
function getVariable() {
var type = NodeType.VariableType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `$`.
pos++;
if (checkVariable(pos)) content.push(getVariable());else content.push(getIdent());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a variables list (e.g. `@rest...`).
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkVariablesList(i) {
var d = 0; // Number of dots
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkVariable(i)) i += l;else return 0;
while (tokens[i] && tokens[i].type === TokenType.FullStop) {
d++;
i++;
}
return d === 3 ? l + d : 0;
}
/**
* Get node with a variables list
* @returns {Array} `['variableslist', ['variable', ['ident', x]]]` where
* `x` is a variable name.
*/
function getVariablesList() {
var type = NodeType.VariablesListType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getVariable()];
var end = getLastPosition(content, line, column, 3);
// Skip `...`.
pos += 3;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
* some value
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkVhash(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
// Skip `#`.
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
if (l = checkNmName2(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a hexadecimal number (e.g. `#fff`) inside some value
* @returns {Array} `['vhash', x]` where `x` is a hexadecimal number
* converted to string (without `#`, e.g. `'fff'`).
*/
function getVhash() {
var type = NodeType.VhashType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `#`.
pos++;
var content = getNmName2();
var end = getLastPosition(content, line, column + 1);
return newNode(type, content, line, column, end);
}
function checkSelectorsGroup(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkSelector(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
var comma = checkDelim(i + spaceBefore);
if (!comma) break;
var spaceAfter = checkSC(i + spaceBefore + comma);
if (l = checkSelector(i + spaceBefore + comma + spaceAfter)) {
i += spaceBefore + comma + spaceAfter + l;
} else break;
}
tokens[start].selectorsGroupEnd = i;
return i - start;
}
function getSelectorsGroup() {
var selectorsGroup = [];
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
selectorsGroup.push(getSelector());
while (pos < selectorsGroupEnd) {
selectorsGroup = selectorsGroup.concat(getSC(), getDelim(), getSC(), getSelector());
}
return selectorsGroup;
}
function checkSelector(i) {
var l = void 0;
if (l = checkSelector1(i)) tokens[i].selectorType = 1;else if (l = checkSelector2(i)) tokens[i].selectorType = 2;
return l;
}
function getSelector() {
var selectorType = tokens[pos].selectorType;
if (selectorType === 1) return getSelector1();else return getSelector2();
}
/**
* Checks for selector which starts with a compound selector.
*/
function checkSelector1(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkCompoundSelector(i)) i += l;else return 0;
while (i < tokensLength) {
var space = checkSC(i);
var comma = checkCombinator(i + space);
if (!space && !comma) break;
if (comma) {
i += space + comma;
space = checkSC(i);
}
if (l = checkCompoundSelector(i + space)) i += space + l;else break;
}
tokens[start].selectorEnd = i;
return i - start;
}
function getSelector1() {
var type = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var selectorEnd = token.selectorEnd;
var content = getCompoundSelector();
while (pos < selectorEnd) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
}
return newNode(type, content, line, column);
}
/**
* Checks for a selector that starts with a combinator.
*/
function checkSelector2(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkCombinator(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
if (l = checkCompoundSelector(i + spaceBefore)) i += spaceBefore + l;else break;
var spaceAfter = checkSC(i);
var comma = checkCombinator(i + spaceAfter);
if (!spaceAfter && !comma) break;
if (comma) {
i += spaceAfter + comma;
}
}
tokens[start].selectorEnd = i;
return i - start;
}
function getSelector2() {
var type = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var selectorEnd = token.selectorEnd;
var content = [getCombinator()];
while (pos < selectorEnd) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
}
return newNode(type, content, line, column);
}
function checkCompoundSelector(i) {
var l = void 0;
if (l = checkCompoundSelector1(i)) {
tokens[i].compoundSelectorType = 1;
} else if (l = checkCompoundSelector2(i)) {
tokens[i].compoundSelectorType = 2;
}
return l;
}
function getCompoundSelector() {
var type = tokens[pos].compoundSelectorType;
if (type === 1) return getCompoundSelector1();
if (type === 2) return getCompoundSelector2();
}
function checkCompoundSelector1(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkUniversalSelector(i) || checkTypeSelector(i) || checkParentSelectorWithExtension(i)) i += l;else return 0;
while (i < tokensLength) {
var _l2 = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i);
if (_l2) i += _l2;else break;
}
tokens[start].compoundSelectorEnd = i;
return i - start;
}
function getCompoundSelector1() {
var sequence = [];
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
if (checkUniversalSelector(pos)) sequence.push(getUniversalSelector());else if (checkTypeSelector(pos)) sequence.push(getTypeSelector());else if (checkParentSelectorWithExtension(pos)) sequence = sequence.concat(getParentSelectorWithExtension());
while (pos < compoundSelectorEnd) {
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());
}
return sequence;
}
function checkCompoundSelector2(i) {
if (i >= tokensLength) return 0;
var start = i;
while (i < tokensLength) {
var l = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i);
if (l) i += l;else break;
}
tokens[start].compoundSelectorEnd = i;
return i - start;
}
function getCompoundSelector2() {
var sequence = [];
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
while (pos < compoundSelectorEnd) {
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());
}
return sequence;
}
function checkUniversalSelector(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (tokens[i].type === TokenType.Asterisk) i++;else return 0;
return i - start;
}
function getUniversalSelector() {
var type = NodeType.UniversalSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var end = void 0;
if (checkNamePrefix(pos)) {
content.push(getNamePrefix());
end = getLastPosition(content, line, column, 1);
}
pos++;
return newNode(type, content, line, column, end);
}
function checkTypeSelector(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
function getTypeSelector() {
var type = NodeType.TypeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkNamePrefix(pos)) content.push(getNamePrefix());
content.push(getIdent());
return newNode(type, content, line, column);
}
function checkAttributeSelector(i) {
var l = void 0;
if (l = checkAttributeSelector1(i)) tokens[i].attributeSelectorType = 1;else if (l = checkAttributeSelector2(i)) tokens[i].attributeSelectorType = 2;
return l;
}
function getAttributeSelector() {
var type = tokens[pos].attributeSelectorType;
if (type === 1) return getAttributeSelector1();else return getAttributeSelector2();
}
/**
* (1) `[panda=nani]`
* (2) `[panda='nani']`
* (3) `[panda='nani' i]`
*
*/
function checkAttributeSelector1(i) {
var start = i;
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeName(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeMatch(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeValue(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeFlags(i)) {
i += l;
if (l = checkSC(i)) i += l;
}
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
return i - start;
}
function getAttributeSelector1() {
var type = NodeType.AttributeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `[`.
pos++;
content = content.concat(getSC(), getAttributeName(), getSC(), getAttributeMatch(), getSC(), getAttributeValue(), getSC());
if (checkAttributeFlags(pos)) {
content.push(getAttributeFlags());
content = content.concat(getSC());
}
// Skip `]`.
pos++;
var end = getLastPosition(content, line, column, 1);
return newNode(type, content, line, column, end);
}
/**
* (1) `[panda]`
*/
function checkAttributeSelector2(i) {
var start = i;
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeName(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
return i - start;
}
function getAttributeSelector2() {
var type = NodeType.AttributeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `[`.
pos++;
content = content.concat(getSC(), getAttributeName(), getSC());
// Skip `]`.
pos++;
var end = getLastPosition(content, line, column, 1);
return newNode(type, content, line, column, end);
}
function checkAttributeName(i) {
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
function getAttributeName() {
var type = NodeType.AttributeNameType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkNamePrefix(pos)) content.push(getNamePrefix());
content.push(getIdent());
return newNode(type, content, line, column);
}
function checkAttributeMatch(i) {
var l = void 0;
if (l = checkAttributeMatch1(i)) tokens[i].attributeMatchType = 1;else if (l = checkAttributeMatch2(i)) tokens[i].attributeMatchType = 2;
return l;
}
function getAttributeMatch() {
var type = tokens[pos].attributeMatchType;
if (type === 1) return getAttributeMatch1();else return getAttributeMatch2();
}
function checkAttributeMatch1(i) {
var start = i;
var type = tokens[i].type;
if (type === TokenType.Tilde || type === TokenType.VerticalLine || type === TokenType.CircumflexAccent || type === TokenType.DollarSign || type === TokenType.Asterisk) i++;else return 0;
if (tokens[i].type === TokenType.EqualsSign) i++;else return 0;
return i - start;
}
function getAttributeMatch1() {
var type = NodeType.AttributeMatchType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos].value + tokens[pos + 1].value;
pos += 2;
return newNode(type, content, line, column);
}
function checkAttributeMatch2(i) {
if (tokens[i].type === TokenType.EqualsSign) return 1;else return 0;
}
function getAttributeMatch2() {
var type = NodeType.AttributeMatchType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '=';
pos++;
return newNode(type, content, line, column);
}
function checkAttributeValue(i) {
return checkString(i) || checkIdent(i);
}
function getAttributeValue() {
var type = NodeType.AttributeValueType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkString(pos)) content.push(getString());else content.push(getIdent());
return newNode(type, content, line, column);
}
function checkAttributeFlags(i) {
return checkIdent(i);
}
function getAttributeFlags() {
var type = NodeType.AttributeFlagsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getIdent()];
return newNode(type, content, line, column);
}
function checkNamePrefix(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkNamePrefix1(i)) tokens[i].namePrefixType = 1;else if (l = checkNamePrefix2(i)) tokens[i].namePrefixType = 2;
return l;
}
function getNamePrefix() {
var type = tokens[pos].namePrefixType;
if (type === 1) return getNamePrefix1();else return getNamePrefix2();
}
/**
* (1) `panda|`
* (2) `panda<comment>|`
*/
function checkNamePrefix1(i) {
var start = i;
var l = void 0;
if (l = checkNamespacePrefix(i)) i += l;else return 0;
if (l = checkCommentML(i)) i += l;
if (l = checkNamespaceSeparator(i)) i += l;else return 0;
return i - start;
}
function getNamePrefix1() {
var type = NodeType.NamePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(getNamespacePrefix());
if (checkCommentML(pos)) content.push(getCommentML());
content.push(getNamespaceSeparator());
return newNode(type, content, line, column);
}
/**
* (1) `|`
*/
function checkNamePrefix2(i) {
return checkNamespaceSeparator(i);
}
function getNamePrefix2() {
var type = NodeType.NamePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getNamespaceSeparator()];
return newNode(type, content, line, column);
}
/**
* (1) `*`
* (2) `panda`
*/
function checkNamespacePrefix(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (tokens[i].type === TokenType.Asterisk) return 1;else if (l = checkIdent(i)) return l;else return 0;
}
function getNamespacePrefix() {
var type = NodeType.NamespacePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (token.type === TokenType.Asterisk) {
var asteriskNode = newNode(NodeType.IdentType, '*', line, column);
content.push(asteriskNode);
pos++;
} else if (checkIdent(pos)) content.push(getIdent());
return newNode(type, content, line, column);
}
/**
* (1) `|`
*/
function checkNamespaceSeparator(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.VerticalLine) return 0;
// Return false if `|=` - [attr|=value]
if (tokens[i + 1] && tokens[i + 1].type === TokenType.EqualsSign) return 0;
return 1;
}
function getNamespaceSeparator() {
var type = NodeType.NamespaceSeparatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '|';
pos++;
return newNode(type, content, line, column);
}
module.exports = function (_tokens, context) {
tokens = _tokens;
tokensLength = tokens.length;
pos = 0;
return contexts[context]();
};
/***/ }),
/* 20 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
module.exports = function (css, tabSize) {
var TokenType = __webpack_require__(13);
var tokens = [];
var urlMode = false;
var c = void 0; // Current character
var cn = void 0; // Next character
var pos = 0;
var tn = 0;
var ln = 1;
var col = 1;
var Punctuation = {
' ': TokenType.Space,
'\n': TokenType.Newline,
'\r': TokenType.Newline,
'\t': TokenType.Tab,
'!': TokenType.ExclamationMark,
'"': TokenType.QuotationMark,
'#': TokenType.NumberSign,
'$': TokenType.DollarSign,
'%': TokenType.PercentSign,
'&': TokenType.Ampersand,
'\'': TokenType.Apostrophe,
'(': TokenType.LeftParenthesis,
')': TokenType.RightParenthesis,
'*': TokenType.Asterisk,
'+': TokenType.PlusSign,
',': TokenType.Comma,
'-': TokenType.HyphenMinus,
'.': TokenType.FullStop,
'/': TokenType.Solidus,
':': TokenType.Colon,
';': TokenType.Semicolon,
'<': TokenType.LessThanSign,
'=': TokenType.EqualsSign,
'>': TokenType.GreaterThanSign,
'?': TokenType.QuestionMark,
'@': TokenType.CommercialAt,
'[': TokenType.LeftSquareBracket,
']': TokenType.RightSquareBracket,
'^': TokenType.CircumflexAccent,
'_': TokenType.LowLine,
'{': TokenType.LeftCurlyBracket,
'|': TokenType.VerticalLine,
'}': TokenType.RightCurlyBracket,
'~': TokenType.Tilde
};
/**
* Add a token to the token list
* @param {string} type
* @param {string} value
*/
function pushToken(type, value, column) {
tokens.push({
tn: tn++,
ln: ln,
col: column,
type: type,
value: value
});
}
/**
* Check if a character is a decimal digit
* @param {string} c Character
* @returns {boolean}
*/
function isDecimalDigit(c) {
return '0123456789'.indexOf(c) >= 0;
}
/**
* Parse spaces
* @param {string} css Unparsed part of CSS string
*/
function parseSpaces(css) {
var start = pos;
// Read the string until we meet a non-space character:
for (; pos < css.length; pos++) {
if (css.charAt(pos) !== ' ') break;
}
// Add a substring containing only spaces to tokens:
pushToken(TokenType.Space, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Parse a string within quotes
* @param {string} css Unparsed part of CSS string
* @param {string} q Quote (either `'` or `"`)
*/
function parseString(css, q) {
var start = pos;
// Read the string until we meet a matching quote:
for (pos++; pos < css.length; pos++) {
// Skip escaped quotes:
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) === q) break;
}
// Add the string (including quotes) to tokens:
var type = q === '"' ? TokenType.StringDQ : TokenType.StringSQ;
pushToken(type, css.substring(start, pos + 1), col);
col += pos - start;
}
/**
* Parse numbers
* @param {string} css Unparsed part of CSS string
*/
function parseDecimalNumber(css) {
var start = pos;
// Read the string until we meet a character that's not a digit:
for (; pos < css.length; pos++) {
if (!isDecimalDigit(css.charAt(pos))) break;
}
// Add the number to tokens:
pushToken(TokenType.DecimalNumber, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Parse identifier
* @param {string} css Unparsed part of CSS string
*/
function parseIdentifier(css) {
var start = pos;
// Skip all opening slashes:
while (css.charAt(pos) === '/') {
pos++;
} // Read the string until we meet a punctuation mark:
for (; pos < css.length; pos++) {
// Skip all '\':
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) in Punctuation) break;
}
var ident = css.substring(start, pos--);
// Enter url mode if parsed substring is `url`:
if (!urlMode && ident === 'url' && css.charAt(pos + 1) === '(') {
urlMode = true;
}
// Add identifier to tokens:
pushToken(TokenType.Identifier, ident, col);
col += pos - start;
}
/**
* Parse a multiline comment
* @param {string} css Unparsed part of CSS string
*/
function parseMLComment(css) {
var start = pos;
// Read the string until we meet `*/`.
// Since we already know first 2 characters (`/*`), start reading
// from `pos + 2`:
for (pos = pos + 2; pos < css.length; pos++) {
if (css.charAt(pos) === '*' && css.charAt(pos + 1) === '/') {
pos++;
break;
}
}
// Add full comment (including `/*` and `*/`) to the list of tokens:
var comment = css.substring(start, pos + 1);
pushToken(TokenType.CommentML, comment, col);
var newlines = comment.split('\n');
if (newlines.length > 1) {
ln += newlines.length - 1;
col = newlines[newlines.length - 1].length;
} else {
col += pos - start;
}
}
/**
* Parse a single line comment
* @param {string} css Unparsed part of CSS string
*/
function parseSLComment(css) {
var start = pos;
// Read the string until we meet line break.
// Since we already know first 2 characters (`//`), start reading
// from `pos + 2`:
for (pos += 2; pos < css.length; pos++) {
if (css.charAt(pos) === '\n' || css.charAt(pos) === '\r') {
break;
}
}
// Add comment (including `//` and line break) to the list of tokens:
pushToken(TokenType.CommentSL, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Convert a CSS string to a list of tokens
* @param {string} css CSS string
* @returns {Array} List of tokens
* @private
*/
function getTokens(css) {
// Parse string, character by character:
for (pos = 0; pos < css.length; col++, pos++) {
c = css.charAt(pos);
cn = css.charAt(pos + 1);
// If we meet `/*`, it's a start of a multiline comment.
// Parse following characters as a multiline comment:
if (c === '/' && cn === '*') {
parseMLComment(css);
}
// If we meet `//` and it is not a part of url:
else if (!urlMode && c === '/' && cn === '/') {
// If we're currently inside a block, treat `//` as a start
// of identifier. Else treat `//` as a start of a single-line
// comment:
parseSLComment(css);
}
// If current character is a double or single quote, it's a start
// of a string:
else if (c === '"' || c === "'") {
parseString(css, c);
}
// If current character is a space:
else if (c === ' ') {
parseSpaces(css);
}
// If current character is a punctuation mark:
else if (c in Punctuation) {
// Add it to the list of tokens:
pushToken(Punctuation[c], c, col);
if (c === '\n' || c === '\r') {
ln++;
col = 0;
} // Go to next line
if (c === ')') urlMode = false; // Exit url mode
else if (c === '\t' && tabSize > 1) col += tabSize - 1;
}
// If current character is a decimal digit:
else if (isDecimalDigit(c)) {
parseDecimalNumber(css);
}
// If current character is anything else:
else {
parseIdentifier(css);
}
}
return tokens;
}
return getTokens(css);
};
/***/ }),
/* 21 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
exports.__esModule = true;
exports.default = {
mark: __webpack_require__(22),
parse: __webpack_require__(23),
stringify: __webpack_require__(5),
tokenizer: __webpack_require__(24)
};
module.exports = exports['default'];
/***/ }),
/* 22 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var TokenType = __webpack_require__(13);
module.exports = function () {
/**
* Mark whitespaces and comments
*/
function markSC(tokens) {
var tokensLength = tokens.length;
var ws = -1; // Flag for whitespaces
var sc = -1; // Flag for whitespaces and comments
var t = void 0; // Current token
// For every token in the token list, mark spaces and line breaks
// as spaces (set both `ws` and `sc` flags). Mark multiline comments
// with `sc` flag.
// If there are several spaces or tabs or line breaks or multiline
// comments in a row, group them: take the last one's index number
// and save it to the first token in the group as a reference:
// e.g., `ws_last = 7` for a group of whitespaces or `sc_last = 9`
// for a group of whitespaces and comments.
for (var i = 0; i < tokensLength; i++) {
t = tokens[i];
switch (t.type) {
case TokenType.Space:
case TokenType.Tab:
t.ws = true;
t.sc = true;
if (ws === -1) ws = i;
if (sc === -1) sc = i;
break;
case TokenType.Newline:
t.ws = true;
t.sc = true;
ws = ws === -1 ? i : ws;
sc = sc === -1 ? i : ws;
tokens[ws].ws_last = i - 1;
tokens[sc].sc_last = i - 1;
tokens[i].ws_last = i;
tokens[i].sc_last = i;
ws = -1;
sc = -1;
break;
case TokenType.CommentML:
case TokenType.CommentSL:
if (ws !== -1) {
tokens[ws].ws_last = i - 1;
ws = -1;
}
t.sc = true;
break;
default:
if (ws !== -1) {
tokens[ws].ws_last = i - 1;
ws = -1;
}
if (sc !== -1) {
tokens[sc].sc_last = i - 1;
sc = -1;
}
}
}
if (ws !== -1) tokens[ws].ws_last = i - 1;
if (sc !== -1) tokens[sc].sc_last = i - 1;
}
/**
* Pair brackets
*/
function markBrackets(tokens) {
var tokensLength = tokens.length;
var ps = []; // Parentheses
var sbs = []; // Square brackets
var cbs = []; // Curly brackets
var t = void 0; // Current token
// For every token in the token list, if we meet an opening (left)
// bracket, push its index number to a corresponding array.
// If we then meet a closing (right) bracket, look at the corresponding
// array. If there are any elements (records about previously met
// left brackets), take a token of the last left bracket (take
// the last index number from the array and find a token with
// this index number) and save right bracket's index as a reference:
for (var i = 0; i < tokensLength; i++) {
t = tokens[i];
switch (t.type) {
case TokenType.LeftParenthesis:
ps.push(i);
break;
case TokenType.RightParenthesis:
if (ps.length) {
t.left = ps.pop();
tokens[t.left].right = i;
}
break;
case TokenType.LeftSquareBracket:
sbs.push(i);
break;
case TokenType.RightSquareBracket:
if (sbs.length) {
t.left = sbs.pop();
tokens[t.left].right = i;
}
break;
case TokenType.LeftCurlyBracket:
cbs.push(i);
break;
case TokenType.RightCurlyBracket:
if (cbs.length) {
t.left = cbs.pop();
tokens[t.left].right = i;
}
break;
}
}
}
function markBlocks(tokens) {
var i = 0;
var l = tokens.length;
var lines = [];
var whitespaceOnlyLines = [];
for (i = 0; i < l; i++) {
var lineStart = i;
var currentLineIndent = 0;
// Get all spaces.
while (i < l && (tokens[i].type === TokenType.Space || tokens[i].type === TokenType.Tab)) {
currentLineIndent += tokens[i].value.length;
i++;
}
lines.push([lineStart, currentLineIndent]);
var x = i;
while (i < l && tokens[i].type !== TokenType.Newline) {
i++;
}
if (x === i) {
whitespaceOnlyLines.push(lines.length - 1);
}
}
var levels = [0];
var blockStarts = [];
for (i = 0; i < lines.length; i++) {
var line = lines[i];
var token = line[0];
var indent = line[1];
var lastLevel = levels[levels.length - 1];
if (indent > lastLevel) {
blockStarts.push(token);
levels.push(indent);
} else {
// Check if line is whitespace-only.
var p = i;
while (true) {
if (whitespaceOnlyLines.indexOf(p) === -1) break;
p++;
}
if (i === p && indent === lastLevel) continue;
if (!lines[p]) {
continue;
}
indent = lines[p][1];
if (indent === lastLevel) {
i = p;
continue;
}
if (indent > lastLevel) {
blockStarts.push(token);
levels.push(lines[p][1]);
i = p;
continue;
}
while (true) {
var _lastLevel = levels.pop();
if (indent < _lastLevel) {
var start = blockStarts.pop();
tokens[start].block_end = token - 1;
} else {
levels.push(indent);
break;
}
}
}
}
blockStarts.forEach(function (start) {
tokens[start].block_end = tokens.length - 1;
});
}
return function (tokens) {
markBrackets(tokens);
markSC(tokens);
markBlocks(tokens);
};
}();
/***/ }),
/* 23 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var Node = __webpack_require__(1);
var NodeType = __webpack_require__(15);
var TokenType = __webpack_require__(13);
var tokens = void 0;
var tokensLength = void 0;
var pos = void 0;
var contexts = {
'arguments': function _arguments() {
return checkArguments(pos) && getArguments();
},
'atkeyword': function atkeyword() {
return checkAtkeyword(pos) && getAtkeyword();
},
'atrule': function atrule() {
return checkAtrule(pos) && getAtrule();
},
'attributeSelector': function attributeSelector() {
return checkAttributeSelector(pos) && getAttributeSelector();
},
'block': function block() {
return checkBlock(pos) && getBlock();
},
'brackets': function brackets() {
return checkBrackets(pos) && getBrackets();
},
'class': function _class() {
return checkClass(pos) && getClass();
},
'combinator': function combinator() {
return checkCombinator(pos) && getCombinator();
},
'commentML': function commentML() {
return checkCommentML(pos) && getCommentML();
},
'commentSL': function commentSL() {
return checkCommentSL(pos) && getCommentSL();
},
'condition': function condition() {
return checkCondition(pos) && getCondition();
},
'conditionalStatement': function conditionalStatement() {
return checkConditionalStatement(pos) && getConditionalStatement();
},
'declaration': function declaration() {
return checkDeclaration(pos) && getDeclaration();
},
'declDelim': function declDelim() {
return checkDeclDelim(pos) && getDeclDelim();
},
'default': function _default() {
return checkDefault(pos) && getDefault();
},
'delim': function delim() {
return checkDelim(pos) && getDelim();
},
'dimension': function dimension() {
return checkDimension(pos) && getDimension();
},
'expression': function expression() {
return checkExpression(pos) && getExpression();
},
'extend': function extend() {
return checkExtend(pos) && getExtend();
},
'function': function _function() {
return checkFunction(pos) && getFunction();
},
'global': function global() {
return checkGlobal(pos) && getGlobal();
},
'ident': function ident() {
return checkIdent(pos) && getIdent();
},
'important': function important() {
return checkImportant(pos) && getImportant();
},
'include': function include() {
return checkInclude(pos) && getInclude();
},
'interpolation': function interpolation() {
return checkInterpolation(pos) && getInterpolation();
},
'loop': function loop() {
return checkLoop(pos) && getLoop();
},
'mixin': function mixin() {
return checkMixin(pos) && getMixin();
},
'namespace': function namespace() {
return checkNamespace(pos) && getNamespace();
},
'number': function number() {
return checkNumber(pos) && getNumber();
},
'operator': function operator() {
return checkOperator(pos) && getOperator();
},
'optional': function optional() {
return checkOptional(pos) && getOptional();
},
'parentheses': function parentheses() {
return checkParentheses(pos) && getParentheses();
},
'parentselector': function parentselector() {
return checkParentSelector(pos) && getParentSelector();
},
'percentage': function percentage() {
return checkPercentage(pos) && getPercentage();
},
'placeholder': function placeholder() {
return checkPlaceholder(pos) && getPlaceholder();
},
'progid': function progid() {
return checkProgid(pos) && getProgid();
},
'property': function property() {
return checkProperty(pos) && getProperty();
},
'propertyDelim': function propertyDelim() {
return checkPropertyDelim(pos) && getPropertyDelim();
},
'pseudoc': function pseudoc() {
return checkPseudoc(pos) && getPseudoc();
},
'pseudoe': function pseudoe() {
return checkPseudoe(pos) && getPseudoe();
},
'ruleset': function ruleset() {
return checkRuleset(pos) && getRuleset();
},
's': function s() {
return checkS(pos) && getS();
},
'selector': function selector() {
return checkSelector(pos) && getSelector();
},
'shash': function shash() {
return checkShash(pos) && getShash();
},
'string': function string() {
return checkString(pos) && getString();
},
'stylesheet': function stylesheet() {
return checkStylesheet(pos) && getStylesheet();
},
'typeSelector': function typeSelector() {
return checkTypeSelector(pos) && getTypeSelector();
},
'unary': function unary() {
return checkUnary(pos) && getUnary();
},
'unicodeRange': function unicodeRange() {
return checkUnicodeRange(pos) && getUnicodeRange();
},
'universalSelector': function universalSelector() {
return checkUniversalSelector(pos) && getUniversalSelector();
},
'urange': function urange() {
return checkUrange(pos) && getUrange();
},
'uri': function uri() {
return checkUri(pos) && getUri();
},
'value': function value() {
return checkValue(pos) && getValue();
},
'variable': function variable() {
return checkVariable(pos) && getVariable();
},
'variableslist': function variableslist() {
return checkVariablesList(pos) && getVariablesList();
},
'vhash': function vhash() {
return checkVhash(pos) && getVhash();
}
};
/**
* Stops parsing and display error.
*
* @param {number=} i Token's index number
*/
function throwError(i) {
var ln = tokens[i].ln;
throw { line: ln, syntax: 'sass' };
}
/**
* @param {number} start
* @param {number} finish
* @return {string}
*/
function joinValues(start, finish) {
var s = '';
for (var i = start; i < finish + 1; i++) {
s += tokens[i].value;
}
return s;
}
/**
* @param {number} start
* @param {number} num
* @return {string}
*/
function joinValues2(start, num) {
if (start + num - 1 >= tokensLength) return;
var s = '';
for (var i = 0; i < num; i++) {
s += tokens[start + i].value;
}
return s;
}
/**
* @param {string|!Array} content
* @param {number} line
* @param {number} column
* @param {number} colOffset
*/
function getLastPosition(content, line, column, colOffset) {
return typeof content === 'string' ? getLastPositionForString(content, line, column, colOffset) : getLastPositionForArray(content, line, column, colOffset);
}
/**
* @param {string} content
* @param {number} line
* @param {number} column
* @param {number} colOffset
*/
function getLastPositionForString(content, line, column, colOffset) {
var position = [];
if (!content) {
position = [line, column];
if (colOffset) position[1] += colOffset - 1;
return position;
}
var lastLinebreak = content.lastIndexOf('\n');
var endsWithLinebreak = lastLinebreak === content.length - 1;
var splitContent = content.split('\n');
var linebreaksCount = splitContent.length - 1;
var prevLinebreak = linebreaksCount === 0 || linebreaksCount === 1 ? -1 : content.length - splitContent[linebreaksCount - 1].length - 2;
// Line:
var offset = endsWithLinebreak ? linebreaksCount - 1 : linebreaksCount;
position[0] = line + offset;
// Column:
if (endsWithLinebreak) {
offset = prevLinebreak !== -1 ? content.length - prevLinebreak : content.length - 1;
} else {
offset = linebreaksCount !== 0 ? content.length - lastLinebreak - column - 1 : content.length - 1;
}
position[1] = column + offset;
if (!colOffset) return position;
if (endsWithLinebreak) {
position[0]++;
position[1] = colOffset;
} else {
position[1] += colOffset;
}
return position;
}
/**
* @param {!Array} content
* @param {number} line
* @param {number} column
* @param {number} colOffset
*/
function getLastPositionForArray(content, line, column, colOffset) {
var position = void 0;
if (content.length === 0) {
position = [line, column];
} else {
var c = content[content.length - 1];
if (c.hasOwnProperty('end')) {
position = [c.end.line, c.end.column];
} else {
position = getLastPosition(c.content, line, column);
}
}
if (!colOffset) return position;
if (tokens[pos - 1] && tokens[pos - 1].type !== 'Newline') {
position[1] += colOffset;
} else {
position[0]++;
position[1] = 1;
}
return position;
}
/**
* @param {string} type
* @param {string|!Array} content
* @param {number} line
* @param {number} column
* @param {!Array} end
*/
function newNode(type, content, line, column, end) {
if (!end) end = getLastPosition(content, line, column);
return new Node({
type: type,
content: content,
start: {
line: line,
column: column
},
end: {
line: end[0],
column: end[1]
},
syntax: 'sass'
});
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkAny(i) {
var l = void 0;
if (l = checkBrackets(i)) tokens[i].any_child = 1;else if (l = checkParentheses(i)) tokens[i].any_child = 2;else if (l = checkString(i)) tokens[i].any_child = 3;else if (l = checkVariablesList(i)) tokens[i].any_child = 4;else if (l = checkVariable(i)) tokens[i].any_child = 5;else if (l = checkPlaceholder(i)) tokens[i].any_child = 6;else if (l = checkPercentage(i)) tokens[i].any_child = 7;else if (l = checkDimension(i)) tokens[i].any_child = 8;else if (l = checkUnicodeRange(i)) tokens[i].any_child = 9;else if (l = checkNumber(i)) tokens[i].any_child = 10;else if (l = checkUri(i)) tokens[i].any_child = 11;else if (l = checkExpression(i)) tokens[i].any_child = 12;else if (l = checkFunctionsList(i)) tokens[i].any_child = 13;else if (l = checkFunction(i)) tokens[i].any_child = 14;else if (l = checkInterpolation(i)) tokens[i].any_child = 15;else if (l = checkIdent(i)) tokens[i].any_child = 16;else if (l = checkClass(i)) tokens[i].any_child = 17;else if (l = checkUnary(i)) tokens[i].any_child = 18;else if (l = checkParentSelector(i)) tokens[i].any_child = 19;else if (l = checkImportant(i)) tokens[i].any_child = 20;else if (l = checkGlobal(i)) tokens[i].any_child = 21;else if (l = checkDefault(i)) tokens[i].any_child = 22;else if (l = checkOptional(i)) tokens[i].any_child = 23;
return l;
}
/**
* @return {!Node}
*/
function getAny() {
var childType = tokens[pos].any_child;
if (childType === 1) return getBrackets();
if (childType === 2) return getParentheses();
if (childType === 3) return getString();
if (childType === 4) return getVariablesList();
if (childType === 5) return getVariable();
if (childType === 6) return getPlaceholder();
if (childType === 7) return getPercentage();
if (childType === 8) return getDimension();
if (childType === 9) return getUnicodeRange();
if (childType === 10) return getNumber();
if (childType === 11) return getUri();
if (childType === 12) return getExpression();
if (childType === 13) return getFunctionsList();
if (childType === 14) return getFunction();
if (childType === 15) return getInterpolation();
if (childType === 16) return getIdent();
if (childType === 17) return getClass();
if (childType === 18) return getUnary();
if (childType === 19) return getParentSelector();
if (childType === 20) return getImportant();
if (childType === 21) return getGlobal();
if (childType === 22) return getDefault();
if (childType === 23) return getOptional();
}
/**
* Checks if token is part of mixin's arguments.
*
* @param {number} i Token's index number
* @return {number} Length of arguments
*/
function checkArguments(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
// Skip `(`.
i++;
while (i < tokens[start].right) {
if (l = checkArgument(i)) i += l;else return 0;
}
return tokens[start].right - start + 1;
}
/**
* @return {Array}
*/
function getArguments() {
var type = NodeType.ArgumentsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var body = void 0;
// Skip `(`.
pos++;
while (pos < tokensLength && tokens[pos].type !== TokenType.RightParenthesis) {
if (checkSingleValueDeclaration(pos)) {
content.push(getSingleValueDeclaration());
} else if (checkArgument(pos)) {
body = getArgument();
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
} else if (checkClass(pos)) content.push(getClass());else throwError(pos);
}
var end = getLastPosition(content, line, column, 1);
// Skip `)`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Checks if token is valid to be part of arguments list
* @param {number} i Token's index number
* @return {number} Length of argument
*/
function checkArgument(i) {
var l = void 0;
if (l = checkBrackets(i)) tokens[i].argument_child = 1;else if (l = checkParentheses(i)) tokens[i].argument_child = 2;else if (l = checkSingleValueDeclaration(i)) tokens[i].argument_child = 3;else if (l = checkFunctionsList(i)) tokens[i].argument_child = 4;else if (l = checkFunction(i)) tokens[i].argument_child = 5;else if (l = checkVariablesList(i)) tokens[i].argument_child = 6;else if (l = checkVariable(i)) tokens[i].argument_child = 7;else if (l = checkSC(i)) tokens[i].argument_child = 8;else if (l = checkDelim(i)) tokens[i].argument_child = 9;else if (l = checkDeclDelim(i)) tokens[i].argument_child = 10;else if (l = checkString(i)) tokens[i].argument_child = 11;else if (l = checkPercentage(i)) tokens[i].argument_child = 12;else if (l = checkDimension(i)) tokens[i].argument_child = 13;else if (l = checkNumber(i)) tokens[i].argument_child = 14;else if (l = checkUri(i)) tokens[i].argument_child = 15;else if (l = checkInterpolation(i)) tokens[i].argument_child = 16;else if (l = checkIdent(i)) tokens[i].argument_child = 17;else if (l = checkVhash(i)) tokens[i].argument_child = 18;else if (l = checkCustomProperty(i)) tokens[i].argument_child = 19;else if (l = checkOperator(i)) tokens[i].argument_child = 20;else if (l = checkUnary(i)) tokens[i].argument_child = 21;else if (l = checkParentSelector(i)) tokens[i].argument_child = 22;else if (l = checkImportant(i)) tokens[i].argument_child = 23;else if (l = checkGlobal(i)) tokens[i].argument_child = 24;else if (l = checkDefault(i)) tokens[i].argument_child = 25;else if (l = checkOptional(i)) tokens[i].argument_child = 26;
return l;
}
/**
* @return {!Node}
*/
function getArgument() {
var childType = tokens[pos].argument_child;
if (childType === 1) return getBrackets();
if (childType === 2) return getParentheses();
if (childType === 3) return getSingleValueDeclaration();
if (childType === 4) return getFunctionsList();
if (childType === 5) return getFunction();
if (childType === 6) return getVariablesList();
if (childType === 7) return getVariable();
if (childType === 8) return getSC();
if (childType === 9) return getDelim();
if (childType === 10) return getDeclDelim();
if (childType === 11) return getString();
if (childType === 12) return getPercentage();
if (childType === 13) return getDimension();
if (childType === 14) return getNumber();
if (childType === 15) return getUri();
if (childType === 16) return getInterpolation();
if (childType === 17) return getIdent();
if (childType === 18) return getVhash();
if (childType === 19) return getCustomProperty();
if (childType === 20) return getOperator();
if (childType === 21) return getUnary();
if (childType === 22) return getParentSelector();
if (childType === 23) return getImportant();
if (childType === 24) return getGlobal();
if (childType === 25) return getDefault();
if (childType === 26) return getOptional();
}
/**
* Checks if token is part of an @-word (e.g. `@import`, `@include`).
*
* @param {number} i Token's index number
* @return {number}
*/
function checkAtkeyword(i) {
var l = void 0;
// Check that token is `@`:
if (i >= tokensLength || tokens[i++].type !== TokenType.CommercialAt) return 0;
return (l = checkIdentOrInterpolation(i)) ? l + 1 : 0;
}
/**
* Gets node with @-word.
*
* @return {!Node}
*/
function getAtkeyword() {
var type = NodeType.AtkeywordType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `@`.
pos++;
var content = getIdentOrInterpolation();
return newNode(type, content, line, column);
}
/**
* Checks if token is a part of an @-rule.
*
* @param {number} i Token's index number
* @return {number} Length of @-rule
*/
function checkAtrule(i) {
var l = void 0;
if (i >= tokensLength) return 0;
// If token already has a record of being part of an @-rule,
// return the @-rule's length:
if (tokens[i].atrule_l !== undefined) return tokens[i].atrule_l;
// If token is part of an @-rule, save the rule's type to token.
// @keyframes:
if (l = checkKeyframesRule(i)) tokens[i].atrule_type = 4;
// @-rule with ruleset:
else if (l = checkAtruler(i)) tokens[i].atrule_type = 1;
// Block @-rule:
else if (l = checkAtruleb(i)) tokens[i].atrule_type = 2;
// Single-line @-rule:
else if (l = checkAtrules(i)) tokens[i].atrule_type = 3;else return 0;
// If token is part of an @-rule, save the rule's length to token:
tokens[i].atrule_l = l;
return l;
}
/**
* Gets node with @-rule.
*
* @return {!Node}
*/
function getAtrule() {
var childType = tokens[pos].atrule_type;
if (childType === 1) return getAtruler(); // @-rule with ruleset
if (childType === 2) return getAtruleb(); // Block @-rule
if (childType === 3) return getAtrules(); // Single-line @-rule
if (childType === 4) return getKeyframesRule();
}
/**
* Checks if token is part of a block @-rule.
*
* @param {number} i Token's index number
* @return {number} Length of the @-rule
*/
function checkAtruleb(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Gets node with a block @-rule.
*
* @return {!Node}
*/
function getAtruleb() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets(), getBlock());
return newNode(type, content, line, column);
}
/**
* Checks if token is part of an @-rule with ruleset.
*
* @param {number} i Token's index number
* @return {number} Length of the @-rule
*/
function checkAtruler(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
if (l = checkAtrulers(i)) i += l;else return 0;
return i - start;
}
/**
* Gets node with an @-rule with ruleset.
*
* @return {!Node}
*/
function getAtruler() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets(), getAtrulers());
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkAtrulers(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
var blockEnd = tokens[i].block_end;
if (!blockEnd) return 0;
while (i < blockEnd) {
if (l = checkSC(i)) tokens[i].atrulers_child = 1;else if (l = checkAtrule(i)) tokens[i].atrulers_child = 2;else if (l = checkRuleset(i)) tokens[i].atrulers_child = 3;else return 0;
i += l;
}
if (i < tokensLength) tokens[i].atrulers_end = 1;
return i - start;
}
/**
* @return {!Node}
*/
function getAtrulers() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = getSC();
while (pos < tokensLength && !tokens[pos].atrulers_end) {
var childType = tokens[pos].atrulers_child;
if (childType === 1) content = content.concat(getSC());else if (childType === 2) content.push(getAtrule());else if (childType === 3) content.push(getRuleset());else break;
}
var end = getLastPosition(content, line, column);
return newNode(type, content, line, column, end);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkAtrules(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
return i - start;
}
/**
* @return {!Node}
*/
function getAtrules() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets());
return newNode(type, content, line, column);
}
/**
* Checks if token is part of a block (e.g. `{...}`).
*
* @param {number} i Token's index number
* @return {number} Length of the block
*/
function checkBlock(i) {
return i < tokensLength && tokens[i].block_end ? tokens[i].block_end - i + 1 : 0;
}
/**
* Gets node with a block.
*
* @return {!Node}
*/
function getBlock() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = tokens[pos].block_end;
var content = [];
while (pos < end) {
if (checkBlockdecl(pos)) content = content.concat(getBlockdecl());else throwError(pos);
}
return newNode(type, content, line, column);
}
/**
* Checks if token is part of a declaration (property-value pair).
*
* @param {number} i Token's index number
* @return {number} Length of the declaration
*/
function checkBlockdecl(i) {
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkBlockdecl7(i)) tokens[i].bd_type = 7;else if (l = checkBlockdecl5(i)) tokens[i].bd_type = 5;else if (l = checkBlockdecl6(i)) tokens[i].bd_type = 6;else if (l = checkBlockdecl1(i)) tokens[i].bd_type = 1;else if (l = checkBlockdecl2(i)) tokens[i].bd_type = 2;else if (l = checkBlockdecl3(i)) tokens[i].bd_type = 3;else if (l = checkBlockdecl4(i)) tokens[i].bd_type = 4;else return 0;
return l;
}
/**
* @return {!Array}
*/
function getBlockdecl() {
var childType = tokens[pos].bd_type;
if (childType === 1) return getBlockdecl1();
if (childType === 2) return getBlockdecl2();
if (childType === 3) return getBlockdecl3();
if (childType === 4) return getBlockdecl4();
if (childType === 5) return getBlockdecl5();
if (childType === 6) return getBlockdecl6();
if (childType === 7) return getBlockdecl7();
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkBlockdecl1(i) {
var start = i;
var l = void 0;
if (l = checkInclude(i)) tokens[i].bd_kind = 2;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 5;else if (l = checkAtrule(i)) tokens[i].bd_kind = 6;else return 0;
i += l;
if (tokens[start].bd_kind === 2 && [2, 4, 6, 8].indexOf(tokens[start].include_type) === -1) return 0;
if (tokens[start].bd_kind === 6 && tokens[start].atrule_type === 3) return 0;
while (i < tokensLength) {
if (l = checkDeclDelim(i)) return i + l - start;
if (l = checkS(i)) i += l;else if (l = checkCommentSL(i)) i += l;else break;
}
return 0;
}
/**
* @return {!Array}
*/
function getBlockdecl1() {
var content = [];
var _content = [];
switch (tokens[pos].bd_kind) {
case 2:
content.push(getInclude());
break;
case 5:
content.push(getDeclaration());
break;
case 6:
content.push(getAtrule());
break;
}
while (pos < tokensLength) {
var _pos = pos;
if (checkDeclDelim(pos)) {
_content.push(getDeclDelim());
content = content.concat(_content);
break;
}
if (checkS(pos)) _content.push(getS());else if (checkCommentSL(pos)) _content.push(getCommentSL());else {
pos = _pos;
break;
}
}
return content;
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkBlockdecl2(i) {
var start = i;
var l = void 0;
if (l = checkConditionalStatement(i)) tokens[i].bd_kind = 1;else if (l = checkInclude(i)) tokens[i].bd_kind = 2;else if (l = checkExtend(i)) tokens[i].bd_kind = 4;else if (l = checkMixin(i)) tokens[i].bd_kind = 8;else if (l = checkLoop(i)) tokens[i].bd_kind = 3;else if (l = checkRuleset(i)) tokens[i].bd_kind = 7;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 5;else if (l = checkAtrule(i)) tokens[i].bd_kind = 6;else return 0;
i += l;
while (i < tokensLength) {
if (l = checkS(i)) i += l;else if (l = checkCommentSL(i)) i += l;else break;
}
return i - start;
}
/**
* @return {!Array}
*/
function getBlockdecl2() {
var content = [];
switch (tokens[pos].bd_kind) {
case 1:
content.push(getConditionalStatement());
break;
case 2:
content.push(getInclude());
break;
case 3:
content.push(getLoop());
break;
case 4:
content.push(getExtend());
break;
case 5:
content.push(getDeclaration());
break;
case 6:
content.push(getAtrule());
break;
case 7:
content.push(getRuleset());
break;
case 8:
content.push(getMixin());
break;
}
while (pos < tokensLength) {
if (checkS(pos)) content.push(getS());else if (checkCommentSL(pos)) content.push(getCommentSL());else break;
}
return content;
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkBlockdecl3(i) {
var start = i;
var l = void 0;
if (l = checkConditionalStatement(i)) tokens[i].bd_kind = 1;else if (l = checkInclude(i)) tokens[i].bd_kind = 2;else if (l = checkExtend(i)) tokens[i].bd_kind = 4;else if (l = checkLoop(i)) tokens[i].bd_kind = 3;else if (l = checkRuleset(i)) tokens[i].bd_kind = 7;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 5;else if (l = checkAtrule(i)) tokens[i].bd_kind = 6;else return 0;
i += l;
return i - start;
}
/**
* @return {!Array}
*/
function getBlockdecl3() {
var content = void 0;
switch (tokens[pos].bd_kind) {
case 1:
content = getConditionalStatement();
break;
case 2:
content = getInclude();
break;
case 3:
content = getLoop();
break;
case 4:
content = getExtend();
break;
case 5:
content = getDeclaration();
break;
case 6:
content = getAtrule();
break;
case 7:
content = getRuleset();
break;
}
return [content];
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkBlockdecl4(i) {
return checkSC(i);
}
/**
* @return {!Array}
*/
function getBlockdecl4() {
return getSC();
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkBlockdecl5(i) {
var start = i;
var l = void 0;
if (l = checkInclude(i)) i += l;else if (l = checkRuleset(i)) i += l;else return 0;
while (i < tokensLength) {
if (l = checkS(i)) i += l;else if (l = checkCommentSL(i)) i += l;else break;
}
return i - start;
}
/**
* @return {!Array}
*/
function getBlockdecl5() {
var content = [];
if (checkInclude(pos)) content.push(getInclude());else content.push(getRuleset());
while (pos < tokensLength) {
if (checkS(pos)) content.push(getS());else if (checkCommentSL(pos)) content.push(getCommentSL());else break;
}
return content;
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkBlockdecl6(i) {
var start = i;
var l = void 0;
if (l = checkInclude(i)) i += l;else if (l = checkRuleset(i)) i += l;else return 0;
return i - start;
}
/**
* @return {!Array}
*/
function getBlockdecl6() {
var content = void 0;
if (checkInclude(pos)) content = getInclude();else content = getRuleset();
return [content];
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkBlockdecl7(i) {
var start = i;
var l = void 0;
if (l = checkInclude(i)) i += l;else return 0;
if ([2, 4, 6, 8].indexOf(tokens[start].include_type) === -1) return 0;
while (i < tokensLength) {
if (l = checkDeclDelim(i)) return i + l - start;
if (l = checkS(i)) i += l;else if (l = checkCommentSL(i)) i += l;else break;
}
return 0;
}
/**
* @return {!Array}
*/
function getBlockdecl7() {
var content = [];
var _content = [];
content.push(getInclude());
while (pos < tokensLength) {
var _pos = pos;
if (checkDeclDelim(pos)) {
_content.push(getDeclDelim());
content = content.concat(_content);
break;
}
if (checkS(pos)) _content.push(getS());else if (checkCommentSL(pos)) _content.push(getCommentSL());else {
pos = _pos;
break;
}
}
return content;
}
/**
* Checks if token is part of text inside square brackets, e.g. `[1]`.
*
* @param {number} i Token's index number
* @return {number}
*/
function checkBrackets(i) {
if (i >= tokensLength) return 0;
var start = i;
// Skip `[`.
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
if (i < tokens[start].right) {
var l = checkTsets(i);
if (l) i += l;else return 0;
}
// Skip `]`.
i++;
return i - start;
}
/**
* Gets node with text inside square brackets, e.g. `[1]`.
*
* @return {!Node}
*/
function getBrackets() {
var type = NodeType.BracketsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var right = token.right;
var content = [];
// Skip `[`.
pos++;
if (pos < right) {
content = getTsets();
}
var end = getLastPosition(content, line, column, 1);
// Skip `]`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Checks if token is part of a class selector (e.g. `.abc`).
*
* @param {number} i Token's index number
* @return {number} Length of the class selector
*/
function checkClass(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].class_l) return tokens[i].class_l;
// Skip `.`.
if (tokens[i].type === TokenType.FullStop) i++;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
while (i < tokensLength) {
if (l = checkIdentOrInterpolation(i)) {
tokens[start].class_l = l + 1;
i += l;
} else break;
}
tokens[start].classEnd = i;
return i - start;
}
/**
* Gets node with a class selector.
*
* @return {!Node}
*/
function getClass() {
var type = NodeType.ClassType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = token.classEnd;
var content = [];
// Skip `.`
pos++;
while (pos < end) {
if (checkIdentOrInterpolation(pos)) {
content = content.concat(getIdentOrInterpolation());
} else break;
}
return newNode(type, content, line, column);
}
/**
* @param {number} i
* @return {number}
*/
function checkCombinator(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkCombinator1(i)) tokens[i].combinatorType = 1;else if (l = checkCombinator2(i)) tokens[i].combinatorType = 2;else if (l = checkCombinator3(i)) tokens[i].combinatorType = 3;else if (l = checkCombinator4(i)) tokens[i].combinatorType = 4;
return l;
}
/**
* @return {!Node}
*/
function getCombinator() {
var type = tokens[pos].combinatorType;
if (type === 1) return getCombinator1();
if (type === 2) return getCombinator2();
if (type === 3) return getCombinator3();
if (type === 4) return getCombinator4();
}
/**
* (1) `>>>`
*
* @param {Number} i
* @return {Number}
*/
function checkCombinator1(i) {
if (i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign) return 3;
return 0;
}
/**
* @return {Node}
*/
function getCombinator1() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '>>>';
// Skip combinator
pos += 3;
return newNode(type, content, line, column);
}
/**
* (1) `||`
* (2) `>>`
*
* @param {number} i
* @return {number}
*/
function checkCombinator2(i) {
if (i + 1 >= tokensLength) return 0;
if (tokens[i].type === TokenType.VerticalLine && tokens[i + 1].type === TokenType.VerticalLine) return 2;
if (tokens[i].type === TokenType.GreaterThanSign && tokens[i + 1].type === TokenType.GreaterThanSign) return 2;
return 0;
}
/**
* @return {!Node}
*/
function getCombinator2() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '' + token.value + tokens[pos + 1].value;
// Skip combinator
pos += 2;
return newNode(type, content, line, column);
}
/**
* (1) `>`
* (2) `+`
* (3) `~`
*
* @param {number} i
* @return {number}
*/
function checkCombinator3(i) {
var type = tokens[i].type;
if (type === TokenType.PlusSign || type === TokenType.GreaterThanSign || type === TokenType.Tilde) return 1;else return 0;
}
/**
* @return {Node}
*/
function getCombinator3() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
// Skip combinator
pos++;
return newNode(type, content, line, column);
}
/**
* (1) `/panda/`
*/
function checkCombinator4(i) {
var start = i;
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
var l = void 0;
if (l = checkIdent(i)) i += l;else return 0;
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getCombinator4() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `/`.
pos++;
var ident = getIdent();
// Skip `/`.
pos++;
var content = '/' + ident.content + '/';
return newNode(type, content, line, column);
}
/**
* Check if token is a multiline comment.
* @param {number} i Token's index number
* @return {number} `1` if token is a multiline comment, otherwise `0`
*/
function checkCommentML(i) {
return i < tokensLength && tokens[i].type === TokenType.CommentML ? 1 : 0;
}
/**
* Get node with a multiline comment
* @return {Array} `['commentML', x]` where `x`
* is the comment's text (without `/*` and `* /`).
*/
function getCommentML() {
var type = NodeType.CommentMLType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos].value.substring(2);
var end = getLastPosition(content, line, column + 2);
if (content.endsWith('*/')) {
content = content.substring(0, content.length - 2);
}
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a single-line comment.
* @param {number} i Token's index number
* @return {number} `1` if token is a single-line comment, otherwise `0`
*/
function checkCommentSL(i) {
return i < tokensLength && tokens[i].type === TokenType.CommentSL ? 1 : 0;
}
/**
* Get node with a single-line comment.
* @return {Array} `['commentSL', x]` where `x` is comment's message
* (without `//`)
*/
function getCommentSL() {
var type = NodeType.CommentSLType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos++].value.substring(2);
var end = !content ? [line, column + 1] : getLastPosition(content, line, column + 2);
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a condition
* (e.g. `@if ...`, `@else if ...` or `@else ...`).
* @param {number} i Token's index number
* @return {number} Length of the condition
*/
function checkCondition(i) {
var start = i;
var l = void 0;
var _i = void 0;
var s = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (['if', 'else'].indexOf(tokens[start + 1].value) < 0) return 0;
while (i < tokensLength) {
if (l = checkBlock(i)) break;
s = checkSC(i);
_i = i + s;
if (l = _checkCondition(_i)) i += l + s;else break;
}
return i - start;
}
function _checkCondition(i) {
return checkVariable(i) || checkNumber(i) || checkInterpolation(i) || checkIdent(i) || checkOperator(i) || checkCombinator(i) || checkString(i);
}
/**
* Get node with a condition.
* @return {Array} `['condition', x]`
*/
function getCondition() {
var type = NodeType.ConditionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var s = void 0;
var _pos = void 0;
content.push(getAtkeyword());
while (pos < tokensLength) {
if (checkBlock(pos)) break;
s = checkSC(pos);
_pos = pos + s;
if (!_checkCondition(_pos)) break;
if (s) content = content.concat(getSC());
content.push(_getCondition());
}
return newNode(type, content, line, column);
}
function _getCondition() {
if (checkVariable(pos)) return getVariable();
if (checkNumber(pos)) return getNumber();
if (checkInterpolation(pos)) return getInterpolation();
if (checkIdent(pos)) return getIdent();
if (checkOperator(pos)) return getOperator();
if (checkCombinator(pos)) return getCombinator();
if (checkString(pos)) return getString();
}
/**
* Check if token is part of a conditional statement
* (e.g. `@if ... {} @else if ... {} @else ... {}`).
* @param {number} i Token's index number
* @return {number} Length of the condition
*/
function checkConditionalStatement(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkCondition(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a condition.
* @return {Array} `['condition', x]`
*/
function getConditionalStatement() {
var type = NodeType.ConditionalStatementType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getCondition(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a declaration (property-value pair)
* @param {number} i Token's index number
* @return {number} Length of the declaration
*/
function checkDeclaration(i) {
return checkDeclaration1(i) || checkDeclaration2(i);
}
/**
* Get node with a declaration
* @return {Array} `['declaration', ['property', x], ['propertyDelim'],
* ['value', y]]`
*/
function getDeclaration() {
return checkDeclaration1(pos) ? getDeclaration1() : getDeclaration2();
}
/**
* Check if token is part of a declaration (property-value pair)
* @param {number} i Token's index number
* @return {number} Length of the declaration
*/
function checkDeclaration1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkProperty(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkPropertyDelim(i)) i++;else return 0;
if (l = checkValue(i)) return i + l - start;
if (l = checkS(i)) i += l;
if (l = checkValue(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a declaration
* @return {Array} `['declaration', ['property', x], ['propertyDelim'],
* ['value', y]]`
*/
function getDeclaration1() {
var type = NodeType.DeclarationType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(getProperty());
if (checkS(pos)) content.push(getS());
content.push(getPropertyDelim());
if (checkS(pos)) content.push(getS());
content.push(getValue());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a declaration (property-value pair)
* @param {number} i Token's index number
* @return {number} Length of the declaration
*/
function checkDeclaration2(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkPropertyDelim(i)) i++;else return 0;
if (l = checkProperty(i)) i += l;else return 0;
if (l = checkValue(i)) return i + l - start;
if (l = checkSC(i)) i += l;
if (l = checkValue(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a declaration
* @return {Array} `['declaration', ['propertyDelim'], ['property', x],
* ['value', y]]`
*/
function getDeclaration2() {
var type = NodeType.DeclarationType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getPropertyDelim(), getProperty(), getSC(), getValue());
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @returns {number} Length of the declaration
*/
function checkSingleValueDeclaration(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkProperty(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkPropertyDelim(i)) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkSingleValue(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a declaration
* @returns {Array} `['declaration', ['property', x], ['propertyDelim'],
* ['value', y]]`
*/
function getSingleValueDeclaration() {
var type = NodeType.DeclarationType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getProperty(), getSC(), getPropertyDelim(), getSC(), getSingleValue());
return newNode(type, content, line, column);
}
/**
* Check if token is a semicolon
* @param {number} i Token's index number
* @return {number} `1` if token is a semicolon, otherwise `0`
*/
function checkDeclDelim(i) {
if (i >= tokensLength) return 0;
return tokens[i].type === TokenType.Newline || tokens[i].type === TokenType.Semicolon ? 1 : 0;
}
/**
* Get node with a semicolon
* @return {Array} `['declDelim']`
*/
function getDeclDelim() {
var type = NodeType.DeclDelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '\n';
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token if part of `!default` word.
* @param {number} i Token's index number
* @return {number} Length of the `!default` word
*/
function checkDefault(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].value === 'default') {
tokens[start].defaultEnd = i;
return i - start + 1;
} else {
return 0;
}
}
/**
* Get node with a `!default` word
* @return {Array} `['default', sc]` where `sc` is optional whitespace
*/
function getDefault() {
var type = NodeType.DefaultType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, token.defaultEnd);
pos = token.defaultEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is a comma
* @param {number} i Token's index number
* @return {number} `1` if token is a comma, otherwise `0`
*/
function checkDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Comma ? 1 : 0;
}
/**
* Get node with a comma
* @return {Array} `['delim']`
*/
function getDelim() {
var type = NodeType.DelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ',';
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is part of a number with dimension unit (e.g. `10px`)
* @param {Number} i Token's index number
* @return {Number}
*/
function checkDimension(i) {
var ln = checkNumber(i);
var li = void 0;
if (i >= tokensLength || !ln || i + ln >= tokensLength) return 0;
return (li = checkUnit(i + ln)) ? ln + li : 0;
}
/**
* Get node of a number with dimension unit
* @return {Node}
*/
function getDimension() {
var type = NodeType.DimensionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getNumber(), getUnit()];
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkExpression(i) {
var start = i;
if (i >= tokensLength || tokens[i++].value !== 'expression' || i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) {
return 0;
}
return tokens[i].right - start + 1;
}
/**
* @return {Array}
*/
function getExpression() {
var type = NodeType.ExpressionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
pos++;
var content = joinValues(pos + 1, tokens[pos].right - 1);
var end = getLastPosition(content, line, column, 1);
if (end[0] === line) end[1] += 11;
pos = tokens[pos].right + 1;
return newNode(type, content, line, column, end);
}
function checkExtend(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkExtend1(i)) tokens[i].extend_child = 1;else if (l = checkExtend2(i)) tokens[i].extend_child = 2;
return l;
}
function getExtend() {
var childType = tokens[pos].extend_child;
if (childType === 1) return getExtend1();
if (childType === 2) return getExtend2();
}
/**
* Checks if token is part of an extend with `!optional` flag.
* @param {number} i
*/
function checkExtend1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'extend') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkOptional(i)) i += l;else return 0;
return i - start;
}
function getExtend1() {
var type = NodeType.ExtendType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getSelectorsGroup(), getSC(), getOptional());
return newNode(type, content, line, column);
}
/**
* Checks if token is part of an extend without `!optional` flag.
* @param {number} i
*/
function checkExtend2(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'extend') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
return i - start;
}
function getExtend2() {
var type = NodeType.ExtendType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getSelectorsGroup());
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkFunction(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i < tokensLength && tokens[i].type === TokenType.LeftParenthesis ? tokens[i].right - start + 1 : 0;
}
/**
* @return {Array}
*/
function getFunction() {
var type = NodeType.FunctionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getIdentOrInterpolation(), getArguments());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a functions list (e.g. `function(value)...`).
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkFunctionsList(i) {
var d = 0; // Number of dots
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkFunction(i)) i += l;else return 0;
while (i < tokensLength && tokens[i].type === TokenType.FullStop) {
d++;
i++;
}
return d === 3 ? l + d : 0;
}
/**
* Get node with a functions list
* @returns {Array}
*/
function getFunctionsList() {
var type = NodeType.FunctionsListType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getFunction()];
var end = getLastPosition(content, line, column, 3);
// Skip `...`.
pos += 3;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of `!global` word
* @param {number} i Token's index number
* @return {number}
*/
function checkGlobal(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].value === 'global') {
tokens[start].globalEnd = i;
return i - start + 1;
} else {
return 0;
}
}
/**
* Get node with `!global` word
*/
function getGlobal() {
var type = NodeType.GlobalType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, token.globalEnd);
pos = token.globalEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of an identifier
* @param {number} i Token's index number
* @return {number} Length of the identifier
*/
function checkIdent(i) {
var start = i;
if (i >= tokensLength) return 0;
// Check if token is part of a negative number
if (tokens[i].type === TokenType.HyphenMinus && tokens[i + 1].type === TokenType.DecimalNumber) return 0;
if (tokens[i].type === TokenType.HyphenMinus) i++;
if (checkInterpolation(i)) {
tokens[start].ident_last = i - 1;
return i - start;
}
if (tokens[i].type === TokenType.LowLine || tokens[i].type === TokenType.Identifier) i++;else return 0;
for (; i < tokensLength; i++) {
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
}
tokens[start].ident_last = i - 1;
return i - start;
}
/**
* Get node with an identifier
* @return {Array} `['ident', x]` where `x` is identifier's name
*/
function getIdent() {
var type = NodeType.IdentType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, tokens[pos].ident_last);
pos = tokens[pos].ident_last + 1;
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @returns {number} Length of the identifier
*/
function checkPartialIdent(i) {
var start = i;
if (i >= tokensLength) return 0;
for (; i < tokensLength; i++) {
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
}
tokens[start].ident_last = i - 1;
return i - start;
}
/**
* @param {number} i Token's index number
* @returns {number} Length of the identifier
*/
function checkIdentOrInterpolation(i) {
var start = i;
var l = void 0;
var prevIsInterpolation = false;
while (i < tokensLength) {
if (l = checkInterpolation(i)) {
tokens[i].ii_type = 1;
i += l;
prevIsInterpolation = true;
} else if (l = checkIdent(i)) {
tokens[i].ii_type = 2;
i += l;
prevIsInterpolation = false;
} else if (prevIsInterpolation && (l = checkPartialIdent(i))) {
tokens[i].ii_type = 3;
i += l;
prevIsInterpolation = false;
} else break;
}
return i - start;
}
function getIdentOrInterpolation() {
var content = [];
while (pos < tokensLength) {
var tokenType = tokens[pos].ii_type;
if (tokenType === 1) {
content.push(getInterpolation());
} else if (tokenType === 2 || tokenType === 3) {
content.push(getIdent());
} else break;
}
return content;
}
/**
* Check if token is part of `!important` word
* @param {number} i Token's index number
* @return {number}
*/
function checkImportant(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].value === 'important') {
tokens[start].importantEnd = i;
return i - start + 1;
} else {
return 0;
}
}
/**
* Get node with `!important` word
* @return {Array} `['important', sc]` where `sc` is optional whitespace
*/
function getImportant() {
var type = NodeType.ImportantType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, token.importantEnd);
pos = token.importantEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of an included mixin (`@include` or `@extend`
* directive).
* @param {number} i Token's index number
* @return {number} Length of the included mixin
*/
function checkInclude(i) {
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIncludeWithKeyframes1(i)) tokens[i].include_type = 9;else if (l = checkInclude1(i)) tokens[i].include_type = 1;else if (l = checkInclude2(i)) tokens[i].include_type = 2;else if (l = checkInclude3(i)) tokens[i].include_type = 3;else if (l = checkInclude4(i)) tokens[i].include_type = 4;else if (l = checkIncludeWithKeyframes2(i)) tokens[i].include_type = 10;else if (l = checkInclude5(i)) tokens[i].include_type = 5;else if (l = checkInclude6(i)) tokens[i].include_type = 6;else if (l = checkInclude7(i)) tokens[i].include_type = 7;else if (l = checkInclude8(i)) tokens[i].include_type = 8;
return l;
}
/**
* Get node with included mixin
* @return {Array} `['include', x]`
*/
function getInclude() {
var type = tokens[pos].include_type;
if (type === 1) return getInclude1();
if (type === 2) return getInclude2();
if (type === 3) return getInclude3();
if (type === 4) return getInclude4();
if (type === 5) return getInclude5();
if (type === 6) return getInclude6();
if (type === 7) return getInclude7();
if (type === 8) return getInclude8();
if (type === 9) return getIncludeWithKeyframes1();
if (type === 10) return getIncludeWithKeyframes2();
}
/**
* Check if token is part of an included mixin like `@include nani(foo) {...}`
* @param {number} i Token's index number
* @return {number} Length of the include
*/
function checkInclude1(i) {
var start = i;
var l = void 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'include') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with included mixin like `@include nani(foo) {...}`
* @return {Array} `['include', ['atkeyword', x], sc, ['selector', y], sc,
* ['arguments', z], sc, ['block', q], sc` where `x` is `include` or
* `extend`, `y` is mixin's identifier (selector), `z` are arguments
* passed to the mixin, `q` is block passed to the mixin and `sc`
* are optional whitespaces
*/
function getInclude1() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an included mixin like `@include nani(foo)`
* @param {number} i Token's index number
* @return {number} Length of the include
*/
function checkInclude2(i) {
var start = i;
var l = void 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'include') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with included mixin like `@include nani(foo)`
* @return {Array} `['include', ['atkeyword', x], sc, ['selector', y], sc,
* ['arguments', z], sc]` where `x` is `include` or `extend`, `y` is
* mixin's identifier (selector), `z` are arguments passed to the
* mixin and `sc` are optional whitespaces
*/
function getInclude2() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an included mixin with a content block passed
* as an argument (e.g. `@include nani {...}`)
* @param {number} i Token's index number
* @return {number} Length of the mixin
*/
function checkInclude3(i) {
var start = i;
var l = void 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'include') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with an included mixin with a content block passed
* as an argument (e.g. `@include nani {...}`)
* @return {Array} `['include', x]`
*/
function getInclude3() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkInclude4(i) {
var start = i;
var l = void 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'include') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
/**
* @return {Array} `['include', x]`
*/
function getInclude4() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an included mixin like `+nani(foo) {...}`
* @param {number} i Token's index number
* @return {number} Length of the include
*/
function checkInclude5(i) {
var start = i;
var l = void 0;
if (tokens[i].type === TokenType.PlusSign) i++;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with included mixin like `+nani(foo) {...}`
* @return {Array} `['include', ['operator', '+'], ['selector', x], sc,
* ['arguments', y], sc, ['block', z], sc` where `x` is
* mixin's identifier (selector), `y` are arguments passed to the
* mixin, `z` is block passed to mixin and `sc` are optional whitespaces
*/
function getInclude5() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getOperator(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an included mixin like `+nani(foo)`
* @param {number} i Token's index number
* @return {number} Length of the include
*/
function checkInclude6(i) {
var start = i;
var l = void 0;
if (tokens[i].type === TokenType.PlusSign) i++;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with included mixin like `+nani(foo)`
* @return {Array} `['include', ['operator', '+'], ['selector', y], sc,
* ['arguments', z], sc]` where `y` is
* mixin's identifier (selector), `z` are arguments passed to the
* mixin and `sc` are optional whitespaces
*/
function getInclude6() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getOperator(), getIdentOrInterpolation(), getSC(), getArguments());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an included mixin with a content block passed
* as an argument (e.g. `+nani {...}`)
* @param {number} i Token's index number
* @return {number} Length of the mixin
*/
function checkInclude7(i) {
var start = i;
var l = void 0;
if (tokens[i].type === TokenType.PlusSign) i++;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with an included mixin with a content block passed
* as an argument (e.g. `+nani {...}`)
* @return {Array} `['include', x]`
*/
function getInclude7() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getOperator(), getIdentOrInterpolation(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkInclude8(i) {
var start = i;
var l = void 0;
if (tokens[i].type === TokenType.PlusSign) i++;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
/**
* @return {Array} `['include', x]`
*/
function getInclude8() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getOperator(), getIdentOrInterpolation());
return newNode(type, content, line, column);
}
/**
* Get node with included mixin with keyfames selector like
* `@include nani(foo) { 0% {}}`
* @param {number} i Token's index number
* @returns {number} Length of the include
*/
function checkIncludeWithKeyframes1(i) {
var start = i;
var l = void 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'include') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with included mixin with keyfames selector like
* `@include nani(foo) { 0% {}}`
* @return {!Node}
*/
function getIncludeWithKeyframes1() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getKeyframesBlocks());
return newNode(type, content, line, column);
}
/**
* Get node with included mixin with keyfames selector like
* `+nani(foo) { 0% {}}`
* @param {number} i Token's index number
* @returns {number} Length of the include
*/
function checkIncludeWithKeyframes2(i) {
var start = i;
var l = void 0;
if (tokens[i].type === TokenType.PlusSign) i++;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with included mixin with keyfames selector like
* `+nani(foo) { 0% {}}`
* @return {!Node}
*/
function getIncludeWithKeyframes2() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getOperator(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getKeyframesBlocks());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an interpolated variable (e.g. `#{$nani}`).
* @param {number} i Token's index number
* @return {number}
*/
function checkInterpolation(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.NumberSign || !tokens[i + 1] || tokens[i + 1].type !== TokenType.LeftCurlyBracket) return 0;
i += 2;
while (tokens[i].type !== TokenType.RightCurlyBracket) {
if (l = checkArgument(i)) i += l;else return 0;
}
return tokens[i].type === TokenType.RightCurlyBracket ? i - start + 1 : 0;
}
/**
* Get node with an interpolated variable
* @return {Array} `['interpolation', x]`
*/
function getInterpolation() {
var type = NodeType.InterpolationType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `#{`:
pos += 2;
while (pos < tokensLength && tokens[pos].type !== TokenType.RightCurlyBracket) {
var body = getArgument();
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
}
var end = getLastPosition(content, line, column, 1);
// Skip `}`:
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check a single keyframe block - `5% {}`
* @param {number} i
* @return {number}
*/
function checkKeyframesBlock(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkKeyframesSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get a single keyframe block - `5% {}`
* @return {Node}
*/
function getKeyframesBlock() {
var type = NodeType.RulesetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getKeyframesSelectorsGroup(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check all keyframe blocks - `5% {} 100% {}`
* @param {number} i
* @return {number}
*/
function checkKeyframesBlocks(i) {
if (i >= tokensLength) return 0;
var blockEnd = tokens[i].block_end;
var start = i;
var l = void 0;
if (!blockEnd) return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlock(i)) i += l;
while (i < blockEnd) {
if (l = checkSC(i)) i += l;else if (l = checkKeyframesBlock(i)) i += l;else if (l = checkAtrule(i)) i += l;else break;
}
if (i !== blockEnd + 1) return 0;
return blockEnd + 1 - start;
}
/**
* Get all keyframe blocks - `5% {} 100% {}`
* @return {Node}
*/
function getKeyframesBlocks() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var keyframesBlocksEnd = token.block_end;
var content = [];
while (pos < keyframesBlocksEnd) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkKeyframesBlock(pos)) content.push(getKeyframesBlock());else if (checkAtrule(pos)) content.push(getAtrule()); // @content
else break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is part of a @keyframes rule.
* @param {number} i Token's index number
* @return {number} Length of the @keyframes rule
*/
function checkKeyframesRule(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
var atruleName = joinValues2(i - l, l);
if (atruleName.toLowerCase().indexOf('keyframes') === -1) return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i) || checkPseudoc(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getKeyframesRule() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC());
if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());else if (checkPseudoc(pos)) {
content = content.concat(getPseudoc());
}
content = content.concat(getSC(), getKeyframesBlocks());
return newNode(type, content, line, column);
}
/**
* Check a single keyframe selector - `5%`, `from` etc
* @param {Number} i
* @return {Number}
*/
function checkKeyframesSelector(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) {
// Valid selectors are only `from` and `to`.
var selector = joinValues2(i, l);
if (selector !== 'from' && selector !== 'to') return 0;
i += l;
tokens[start].keyframesSelectorType = 1;
} else if (l = checkPercentage(i)) {
i += l;
tokens[start].keyframesSelectorType = 2;
} else if (l = checkInterpolation(i)) {
i += l;
tokens[start].keyframesSelectorType = 3;
} else {
return 0;
}
return i - start;
}
/**
* Get a single keyframe selector
* @return {Node}
*/
function getKeyframesSelector() {
var keyframesSelectorType = NodeType.KeyframesSelectorType;
var selectorType = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (token.keyframesSelectorType === 1) {
content.push(getIdent());
} else if (token.keyframesSelectorType === 2) {
content.push(getPercentage());
} else if (token.keyframesSelectorType === 3) {
content.push(getInterpolation());
}
var keyframesSelector = newNode(keyframesSelectorType, content, line, column);
return newNode(selectorType, [keyframesSelector], line, column);
}
/**
* Check the keyframe's selector groups
* @param {number} i
* @return {number}
*/
function checkKeyframesSelectorsGroup(i) {
var start = i;
var l = void 0;
if (l = checkKeyframesSelector(i)) i += l;else return 0;
// Check for trailing space
if (l = checkSC(i) && tokens[i].type !== TokenType.Newline) i += l;
while (i < tokensLength) {
var tempStart = i;
var tempIndex = i;
var tempLength = void 0;
if (tempLength = checkDelim(tempIndex)) tempIndex += tempLength;else break;
// Check for maxmimum space usage - 'space', '\n', 'space'
if (tempLength = checkSC(tempIndex)) tempIndex += tempLength;
if (tempLength = checkSC(tempIndex)) tempIndex += tempLength;
if (tempLength = checkSC(tempIndex)) tempIndex += tempLength;
if (tempLength = checkKeyframesSelector(tempIndex)) tempIndex += tempLength;else break;
// Check for trailing space
if (tempLength = checkSC(tempIndex) && tokens[tempIndex].type !== TokenType.Newline) {
tempIndex += tempLength;
}
i += tempIndex - tempStart;
}
tokens[start].selectorsGroupEnd = i;
return i - start;
}
/**
* Get the keyframe's selector groups
* @return {Array} An array of keyframe selectors
*/
function getKeyframesSelectorsGroup() {
var selectorsGroup = [];
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
selectorsGroup.push(getKeyframesSelector());
if (checkSC(pos) && tokens[pos].type !== TokenType.Newline) {
selectorsGroup = selectorsGroup.concat(getSC());
}
while (pos < selectorsGroupEnd) {
selectorsGroup = selectorsGroup.concat(getDelim(), getSC(), getSC(), getSC(), getKeyframesSelector());
if (checkSC(pos) && tokens[pos].type !== TokenType.Newline) {
selectorsGroup = selectorsGroup.concat(getSC());
}
}
return selectorsGroup;
}
/**
* Check if token is part of a loop.
* @param {number} i Token's index number
* @return {number} Length of the loop
*/
function checkLoop(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (['for', 'each', 'while'].indexOf(tokens[start + 1].value) < 0) return 0;
while (i < tokensLength) {
if (l = checkBlock(i)) {
i += l;
break;
} else if (l = checkVariable(i) || checkNumber(i) || checkInterpolation(i) || checkIdent(i) || checkSC(i) || checkOperator(i) || checkCombinator(i) || checkString(i)) i += l;else return 0;
}
return i - start;
}
/**
* Get node with a loop.
* @return {Array} `['loop', x]`
*/
function getLoop() {
var type = NodeType.LoopType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(getAtkeyword());
while (pos < tokensLength) {
if (checkBlock(pos)) {
content.push(getBlock());
break;
} else if (checkVariable(pos)) content.push(getVariable());else if (checkNumber(pos)) content.push(getNumber());else if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkIdent(pos)) content.push(getIdent());else if (checkOperator(pos)) content.push(getOperator());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkSC(pos)) content = content.concat(getSC());else if (checkString(pos)) content.push(getString());
}
return newNode(type, content, line, column);
}
/**
* Check if token is part of a mixin
* @param {number} i Token's index number
* @return {number} Length of the mixin
*/
function checkMixin(i) {
return checkMixin1(i) || checkMixin2(i);
}
/**
* Get node with a mixin
* @return {Array} `['mixin', x]`
*/
function getMixin() {
return checkMixin1(pos) ? getMixin1() : getMixin2();
}
/**
* Check if token is part of a mixin
* @param {number} i Token's index number
* @return {number} Length of the mixin
*/
function checkMixin1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if ((l = checkAtkeyword(i)) && tokens[i + 1].value === 'mixin') i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else {
if (l = checkArguments(i)) i += l;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
}
return i - start;
}
/**
* Get node with a mixin
* @return {Array} `['mixin', x]`
*/
function getMixin1() {
var type = NodeType.MixinType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC());
if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());
content = content.concat(getSC());
if (checkBlock(pos)) content.push(getBlock());else {
if (checkArguments(pos)) content.push(getArguments());
content = content.concat(getSC());
content.push(getBlock());
}
return newNode(type, content, line, column);
}
/**
* Check if token is part of a mixin
* @param {number} i Token's index number
* @return {number} Length of the mixin
*/
function checkMixin2(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.EqualsSign) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else {
if (l = checkArguments(i)) i += l;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
}
return i - start;
}
/**
* Get node with a mixin
* @return {Array} `['mixin', x]`
*/
function getMixin2() {
var type = NodeType.MixinType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getOperator(), getSC());
if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());
content = content.concat(getSC());
if (checkBlock(pos)) content.push(getBlock());else {
if (checkArguments(pos)) content.push(getArguments());
content = content.concat(getSC());
content.push(getBlock());
}
return newNode(type, content, line, column);
}
/**
* Check if token is a namespace sign (`|`)
* @param {number} i Token's index number
* @return {number} `1` if token is `|`, `0` if not
*/
function checkNamespace(i) {
return i < tokensLength && tokens[i].type === TokenType.VerticalLine ? 1 : 0;
}
/**
* Get node with a namespace sign
* @return {Array} `['namespace']`
*/
function getNamespace() {
var type = NodeType.NamespaceType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '|';
pos++;
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkNmName2(i) {
if (tokens[i].type === TokenType.Identifier) return 1;else if (tokens[i].type !== TokenType.DecimalNumber) return 0;
i++;
return i < tokensLength && tokens[i].type === TokenType.Identifier ? 2 : 1;
}
/**
* @return {string}
*/
function getNmName2() {
var s = tokens[pos].value;
if (tokens[pos++].type === TokenType.DecimalNumber && pos < tokensLength && tokens[pos].type === TokenType.Identifier) s += tokens[pos++].value;
return s;
}
/**
* Check if token is part of a number
* @param {number} i Token's index number
* @return {number} Length of number
*/
function checkNumber(i) {
if (i >= tokensLength) return 0;
if (tokens[i].number_l) return tokens[i].number_l;
// `10`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && (!tokens[i + 1] || tokens[i + 1] && tokens[i + 1].type !== TokenType.FullStop)) {
tokens[i].number_l = 1;
return 1;
}
// `10.`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && (!tokens[i + 2] || tokens[i + 2].type !== TokenType.DecimalNumber)) {
tokens[i].number_l = 2;
return 2;
}
// `.10`:
if (i < tokensLength && tokens[i].type === TokenType.FullStop && tokens[i + 1].type === TokenType.DecimalNumber) {
tokens[i].number_l = 2;
return 2;
}
// `10.10`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && tokens[i + 2] && tokens[i + 2].type === TokenType.DecimalNumber) {
tokens[i].number_l = 3;
return 3;
}
return 0;
}
/**
* Get node with number
* @return {Array} `['number', x]` where `x` is a number converted
* to string.
*/
function getNumber() {
var type = NodeType.NumberType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var l = tokens[pos].number_l;
var content = '';
for (var j = 0; j < l; j++) {
content += tokens[pos + j].value;
}
pos += l;
return newNode(type, content, line, column);
}
/**
* Check if token is an operator (`/`, `%`, `,`, `:` or `=`).
* @param {number} i Token's index number
* @return {number} `1` if token is an operator, otherwise `0`
*/
function checkOperator(i) {
if (i >= tokensLength) return 0;
switch (tokens[i].type) {
case TokenType.Solidus:
case TokenType.PercentSign:
case TokenType.Comma:
case TokenType.Colon:
case TokenType.EqualsSign:
case TokenType.EqualitySign:
case TokenType.InequalitySign:
case TokenType.LessThanSign:
case TokenType.GreaterThanSign:
case TokenType.Asterisk:
return 1;
}
return 0;
}
/**
* Get node with an operator
* @return {Array} `['operator', x]` where `x` is an operator converted
* to string.
*/
function getOperator() {
var type = NodeType.OperatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is part of `!optional` word
* @param {number} i Token's index number
* @return {number}
*/
function checkOptional(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].value === 'optional') {
tokens[start].optionalEnd = i;
return i - start + 1;
} else {
return 0;
}
}
/**
* Get node with `!optional` word
*/
function getOptional() {
var type = NodeType.OptionalType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, token.optionalEnd);
pos = token.optionalEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of text inside parentheses, e.g. `(1)`
* @param {number} i Token's index number
* @return {number}
*/
function checkParentheses(i) {
if (i >= tokensLength) return 0;
var start = i;
var right = tokens[i].right;
var l = void 0;
// Skip `(`.
if (tokens[i].type === TokenType.LeftParenthesis) i++;else return 0;
if (i < right) {
if (l = checkTsets(i)) i += l;else return 0;
}
// Skip `)`.
i++;
return i - start;
}
/**
* Get node with text inside parentheses, e.g. `(1)`
* @return {Node}
*/
function getParentheses() {
var type = NodeType.ParenthesesType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var right = token.right;
var content = [];
// Skip `(`.
pos++;
if (pos < right) {
content = getTsets();
}
var end = getLastPosition(content, line, column, 1);
// Skip `)`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is a parent selector, e.g. `&`
* @param {number} i Token's index number
* @return {number}
*/
function checkParentSelector(i) {
return i < tokensLength && tokens[i].type === TokenType.Ampersand ? 1 : 0;
}
/**
* Get node with a parent selector
* @return {Node}
*/
function getParentSelector() {
var type = NodeType.ParentSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '&';
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a parent selector extension, e.g. `&--foo-bar`
* @param {number} i Token's index number
* @returns {number} Length of the parent selector extension
*/
function checkParentSelectorExtension(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
while (i < tokensLength) {
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else break;
}
return i - start;
}
/**
* Get parent selector extension node
* @return {Node}
*/
function getParentSelectorExtension() {
var type = NodeType.ParentSelectorExtensionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
while (pos < tokensLength) {
if (checkIdentOrInterpolation(pos)) {
content = content.concat(getIdentOrInterpolation());
} else if (checkPartialIdent(pos)) {
content.push(getIdent());
} else break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is a parent selector with an extension or not
* @param {number} i Token's index number
* @return {number} Length of the parent selector and extension if applicable
*/
function checkParentSelectorWithExtension(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkParentSelector(i)) i += l;else return 0;
if (l = checkParentSelectorExtension(i)) i += l;
return i - start;
}
/**
* Get parent selector node and extension node if applicable
* @return {Array}
*/
function getParentSelectorWithExtension() {
var content = [getParentSelector()];
if (checkParentSelectorExtension(pos)) content.push(getParentSelectorExtension());
return content;
}
/**
* Check if token is part of a number or an interpolation with a percent sign
* (e.g. `10%`).
* @param {number} i Token's index number
* @return {number}
*/
function checkPercentage(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkNumberOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
// Skip `%`.
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
return i - start;
}
/**
* Get a percentage node that contains either a number or an interpolation
* @return {Object} The percentage node
*/
function getPercentage() {
var type = NodeType.PercentageType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = getNumberOrInterpolation();
var end = getLastPosition(content, line, column, 1);
// Skip `%`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is a number or an interpolation
* @param {number} i Token's index number
* @return {number}
*/
function checkNumberOrInterpolation(i) {
var start = i;
var l = void 0;
while (i < tokensLength) {
if (l = checkInterpolation(i) || checkNumber(i)) i += l;else break;
}
return i - start;
}
/**
* Get a number and/or interpolation node
* @return {Array} An array containing a single or multiple nodes
*/
function getNumberOrInterpolation() {
var content = [];
while (pos < tokensLength) {
if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkNumber(pos)) content.push(getNumber());else break;
}
return content;
}
/**
* Check if token is part of a placeholder selector (e.g. `%abc`).
* @param {number} i Token's index number
* @return {number} Length of the selector
*/
function checkPlaceholder(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[start].placeholder_l) return tokens[start].placeholder_l;
// Skip `%`.
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
if (l = checkIdentOrInterpolation(i)) {
i += l;
tokens[start].placeholder_l = i - start;
} else return 0;
return i - start;
}
/**
* Get node with a placeholder selector
* @return {Array} `['placeholder', ['ident', x]]` where x is a placeholder's
* identifier (without `%`, e.g. `abc`).
*/
function getPlaceholder() {
var type = NodeType.PlaceholderType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `%`.
pos++;
content = content.concat(getIdentOrInterpolation());
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkProgid(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (joinValues2(i, 6) === 'progid:DXImageTransform.Microsoft.') i += 6;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type === TokenType.LeftParenthesis) {
tokens[start].progid_end = tokens[i].right;
i = tokens[i].right + 1;
} else return 0;
return i - start;
}
/**
* @return {Array}
*/
function getProgid() {
var type = NodeType.ProgidType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var progid_end = token.progid_end;
var content = joinValues(pos, progid_end);
pos = progid_end + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of a property
* @param {Number} i Token's index number
* @return {Number} Length of the property
*/
function checkProperty(i) {
var start = i;
var l = void 0;
if (l = checkProperty1(i)) tokens[start].propertyType = 1;else if (l = checkProperty2(i)) tokens[start].propertyType = 2;else if (l = checkProperty3(i)) tokens[start].propertyType = 3;
return l;
}
/**
* Get node with a property
* @return {!Node}
*/
function getProperty() {
var type = tokens[pos].propertyType;
if (type === 1) return getProperty1();
if (type === 2) return getProperty2();
if (type === 3) return getProperty3();
}
/**
* Check if token is part of a property
* (1) `foo`
* (2) `#{$foo}`
* @param {Number} i Token's index number
* @returns {Number} Length of the property
*/
function checkProperty1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a property
* @returns {Array}
*/
function getProperty1() {
var type = NodeType.PropertyType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = getIdentOrInterpolation();
return newNode(type, content, line, column);
}
/**
* Check if token is part of a custom property
* (1) `--foo-bar`
* @param {Number} i Token's index number
* @return {Number} Length of the property
*/
function checkProperty2(i) {
return checkCustomProperty(i);
}
/**
* Get node with a custom property
* @return {Node}
*/
function getProperty2() {
return getCustomProperty();
}
/**
* Check if token is part of a property
* (1) `$foo`
* @param {Number} i Token's index number
* @returns {Number} Length of the property
*/
function checkProperty3(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkVariable(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a property
* @returns {Array} `['property', x]`
*/
function getProperty3() {
var type = NodeType.PropertyType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getVariable()];
return newNode(type, content, line, column);
}
/**
* Check if token is part of a custom property
* @param {Number} i Token's index number
* @return {Number} Length of the property
*/
function checkCustomProperty(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.HyphenMinus || tokens[i + 1] && tokens[i + 1].type !== TokenType.HyphenMinus) return 0;
// Skip `--`
i += 2;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a custom property
* @return {Node}
*/
function getCustomProperty() {
var type = NodeType.CustomPropertyType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `--`
pos += 2;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* Check if token is a colon
* @param {number} i Token's index number
* @return {number} `1` if token is a colon, otherwise `0`
*/
function checkPropertyDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Colon ? 1 : 0;
}
/**
* Get node with a colon
* @return {Array} `['propertyDelim']`
*/
function getPropertyDelim() {
var type = NodeType.PropertyDelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ':';
// Skip `:`.
pos++;
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkPseudo(i) {
return checkPseudoe(i) || checkPseudoc(i);
}
/**
* @return {Array}
*/
function getPseudo() {
if (checkPseudoe(pos)) return getPseudoe();
if (checkPseudoc(pos)) return getPseudoc();
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkPseudoe(i) {
var l = void 0;
// Check `::`
if (i >= tokensLength || tokens[i].type !== TokenType.Colon || i + 1 >= tokensLength || tokens[i + 1].type !== TokenType.Colon) return 0;
if (l = checkPseudoElement1(i)) tokens[i].pseudoElementType = 1;else if (l = checkPseudoElement2(i)) tokens[i].pseudoElementType = 2;else return 0;
return l;
}
/**
* @return {Node}
*/
function getPseudoe() {
var childType = tokens[pos].pseudoElementType;
if (childType === 1) return getPseudoElement1();
if (childType === 2) return getPseudoElement2();
}
/**
* (1) `::slotted(selector)`
* (2) `::slotted(selector, selector)`
*/
function checkPseudoElement1(i) {
var start = i;
var l = void 0;
// Skip `::`.
i += 2;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
/**
* (1) `::slotted(selector)`
* (2) `::slotted(selector, selector)`
*/
function getPseudoElement1() {
var type = NodeType.PseudoeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `::`.
pos += 2;
content.push(getIdent());
{
var _type = NodeType.ArgumentsType;
var _token = tokens[pos];
var _line = _token.ln;
var _column = _token.col;
// Skip `(`.
pos++;
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
var end = getLastPosition(selectorContent, _line, _column, 1);
var args = newNode(_type, selectorContent, _line, _column, end);
content.push(args);
// Skip `)`.
pos++;
}
return newNode(type, content, line, column);
}
function checkPseudoElement2(i) {
var start = i;
var l = void 0;
// Skip `::`.
i += 2;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getPseudoElement2() {
var type = NodeType.PseudoeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `::`.
pos += 2;
var content = getIdentOrInterpolation();
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkPseudoc(i) {
var l = void 0;
if (i >= tokensLength || tokens[i].type !== TokenType.Colon) return 0;
if (l = checkPseudoClass3(i)) tokens[i].pseudoClassType = 3;else if (l = checkPseudoClass4(i)) tokens[i].pseudoClassType = 4;else if (l = checkPseudoClass5(i)) tokens[i].pseudoClassType = 5;else if (l = checkPseudoClass1(i)) tokens[i].pseudoClassType = 1;else if (l = checkPseudoClass2(i)) tokens[i].pseudoClassType = 2;else if (l = checkPseudoClass6(i)) tokens[i].pseudoClassType = 6;else return 0;
return l;
}
/**
* @return {Array}
*/
function getPseudoc() {
var childType = tokens[pos].pseudoClassType;
if (childType === 1) return getPseudoClass1();
if (childType === 2) return getPseudoClass2();
if (childType === 3) return getPseudoClass3();
if (childType === 4) return getPseudoClass4();
if (childType === 5) return getPseudoClass5();
if (childType === 6) return getPseudoClass6();
}
/**
* (-) `:not(panda)`
*/
function checkPseudoClass1(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
/**
* (-) `:not(panda)`
*/
function getPseudoClass1() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content = content.concat(getIdentOrInterpolation());
{
var _type2 = NodeType.ArgumentsType;
var _token2 = tokens[pos];
var _line2 = _token2.ln;
var _column2 = _token2.col;
// Skip `(`.
pos++;
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
var end = getLastPosition(selectorContent, _line2, _column2, 1);
var args = newNode(_type2, selectorContent, _line2, _column2, end);
content.push(args);
// Skip `)`.
pos++;
}
return newNode(type, content, line, column);
}
/**
* (1) `:nth-child(odd)`
* (2) `:nth-child(even)`
* (3) `:lang(de-DE)`
*/
function checkPseudoClass2(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass2() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content = content.concat(getIdentOrInterpolation());
var l = tokens[pos].ln;
var c = tokens[pos].col;
// Skip `(`.
pos++;
var value = [].concat(getSC(), getIdentOrInterpolation(), getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(-3n + 2)`
*/
function checkPseudoClass3(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (l = checkNumberOrInterpolation(i)) i += l;
if (i >= tokensLength) return 0;
if (tokens[i].value === 'n') i++;
if (l = checkSC(i)) i += l;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.PlusSign || tokens[i].type === TokenType.HyphenMinus) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkNumberOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass3() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content = content.concat(getIdentOrInterpolation());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkNumberOrInterpolation(pos)) value = value.concat(getNumberOrInterpolation());
{
var _token3 = tokens[pos];
if (_token3.value === 'n') {
var _l = _token3.ln;
var _c = _token3.col;
var _content2 = _token3.value;
var ident = newNode(NodeType.IdentType, _content2, _l, _c);
value.push(ident);
pos++;
}
}
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
value = value.concat(getSC());
if (checkNumberOrInterpolation(pos)) value = value.concat(getNumberOrInterpolation());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(-3n)`
*/
function checkPseudoClass4(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (l = checkInterpolation(i)) i += l;
if (tokens[i].type === TokenType.DecimalNumber) i++;
if (tokens[i].value === 'n') i++;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass4() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content = content.concat(getIdentOrInterpolation());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkInterpolation(pos)) value.push(getInterpolation());
if (checkNumber(pos)) value.push(getNumber());
if (checkIdent(pos)) value.push(getIdent());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(+8)`
*/
function checkPseudoClass5(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass5() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content = content.concat(getIdentOrInterpolation());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkNumber(pos)) value.push(getNumber());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:checked`
*/
function checkPseudoClass6(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
function getPseudoClass6() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `:`.
pos++;
var content = getIdentOrInterpolation();
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkRuleset(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) {
i += l;
} else if (l = checkSC(i)) {
i += l;
if (l = checkBlock(i)) i += l;else return 0;
} else return 0;
return i - start;
}
function getRuleset() {
var type = NodeType.RulesetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getSelectorsGroup(), getSC());
if (checkBlock(pos)) {
content.push(getBlock());
} else {
content = content.concat(getSC(), getBlock());
}
return newNode(type, content, line, column);
}
/**
* Check if token is marked as a space (if it's a space or a tab
* or a line break).
* @param {number} i
* @return {number} Number of spaces in a row starting with the given token.
*/
function checkS(i) {
return i < tokensLength && tokens[i].ws ? tokens[i].ws_last - i + 1 : 0;
}
/**
* Get node with spaces
* @return {Array} `['s', x]` where `x` is a string containing spaces
*/
function getS() {
var type = NodeType.SType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, tokens[pos].ws_last);
pos = tokens[pos].ws_last + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is a space, newline, or a comment.
* @param {number} i Token's index number
* @return {number} Number of similar (space, newline, or comment) tokens
* in a row starting with the given token.
*/
function checkMultilineSC(i) {
if (!tokens[i]) return 0;
var l = void 0;
var lsc = 0;
while (i < tokensLength) {
if (!(l = checkS(i)) && !(l = checkCommentML(i)) && !(l = checkCommentSL(i))) break;
i += l;
lsc += l;
}
return lsc || 0;
}
/**
* Get node with spaces newlines and comments
* @return {!Node}
*/
function getMultilineSC() {
var sc = [];
if (pos >= tokensLength) return sc;
while (pos < tokensLength) {
if (checkS(pos)) sc.push(getS());else if (checkCommentML(pos)) sc.push(getCommentML());else if (checkCommentSL(pos)) sc.push(getCommentSL());else break;
}
return sc;
}
/**
* Check if token is a space or a comment.
* @param {number} i Token's index number
* @return {number} Number of similar (space or comment) tokens
* in a row starting with the given token.
*/
function checkSC(i) {
if (i >= tokensLength) return 0;
var l = void 0;
var lsc = 0;
var ln = tokens[i].ln;
while (i < tokensLength) {
if (tokens[i].ln !== ln) break;
if (l = checkS(i)) tokens[i].sc_child = 1;else if (l = checkCommentML(i)) tokens[i].sc_child = 2;else if (l = checkCommentSL(i)) tokens[i].sc_child = 3;else break;
i += l;
lsc += l;
if (tokens[i] && tokens[i].type === TokenType.Newline) break;
}
return lsc || 0;
}
/**
* Get node with spaces and comments
* @return {Array} Array containing nodes with spaces (if there are any)
* and nodes with comments (if there are any):
* `[['s', x]*, ['comment', y]*]` where `x` is a string of spaces
* and `y` is a comment's text (without `/*` and `* /`).
*/
function getSC() {
var sc = [];
if (pos >= tokensLength) return sc;
var ln = tokens[pos].ln;
while (pos < tokensLength) {
if (tokens[pos].ln !== ln) break;else if (checkS(pos)) sc.push(getS());else if (checkCommentML(pos)) sc.push(getCommentML());else if (checkCommentSL(pos)) sc.push(getCommentSL());else break;
if (tokens[pos] && tokens[pos].type === TokenType.Newline) break;
}
return sc;
}
/**
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside a simple
* selector
* @param {number} i Token's index number
* @return {number}
*/
function checkShash(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else return 0;
while (i < tokensLength) {
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else break;
}
tokens[start].shashEnd = i;
return i - start;
}
/**
* Get node with a hexadecimal number (e.g. `#fff`) inside a simple selector
* @return {Node}
*/
function getShash() {
var type = NodeType.ShashType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = token.shashEnd;
var content = [];
// Skip `#`.
pos++;
while (pos < end) {
if (checkIdentOrInterpolation(pos)) {
content = content.concat(getIdentOrInterpolation());
} else if (checkPartialIdent(pos)) {
content.push(getIdent());
} else break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is part of a string (text wrapped in quotes)
* @param {number} i Token's index number
* @return {number} `1` if token is part of a string, `0` if not
*/
function checkString(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.StringSQ || tokens[i].type === TokenType.StringDQ) {
return 1;
}
return 0;
}
/**
* Get string's node
* @return {Array} `['string', x]` where `x` is a string (including
* quotes).
*/
function getString() {
var type = NodeType.StringType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Validate stylesheet: it should consist of any number (0 or more) of
* rulesets (sets of rules with selectors), @-rules, whitespaces or
* comments.
* @param {number} i Token's index number
* @return {number}
*/
function checkStylesheet(i) {
var start = i;
var l = void 0;
while (i < tokensLength) {
if (l = checkSC(i) || checkDeclaration(i) || checkDeclDelim(i) || checkInclude(i) || checkExtend(i) || checkMixin(i) || checkLoop(i) || checkConditionalStatement(i) || checkAtrule(i) || checkRuleset(i)) i += l;else throwError(i);
}
return i - start;
}
/**
* @return {Array} `['stylesheet', x]` where `x` is all stylesheet's
* nodes.
*/
function getStylesheet() {
var type = NodeType.StylesheetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var node = void 0;
var wasDeclaration = false;
while (pos < tokensLength) {
if (wasDeclaration && checkDeclDelim(pos)) node = getDeclDelim();else if (checkSC(pos)) node = getSC();else if (checkRuleset(pos)) node = getRuleset();else if (checkInclude(pos)) node = getInclude();else if (checkExtend(pos)) node = getExtend();else if (checkMixin(pos)) node = getMixin();else if (checkLoop(pos)) node = getLoop();else if (checkConditionalStatement(pos)) node = getConditionalStatement();else if (checkAtrule(pos)) node = getAtrule();else if (checkDeclaration(pos)) node = getDeclaration();else throwError(pos);
wasDeclaration = node.type === NodeType.DeclarationType;
if (Array.isArray(node)) content = content.concat(node);else content.push(node);
}
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @return {Number}
*/
function checkTset(i) {
return checkVhash(i) || checkOperator(i) || checkAny(i) || checkSC(i);
}
/**
* @return {Array}
*/
function getTset() {
if (checkVhash(pos)) return getVhash();else if (checkOperator(pos)) return getOperator();else if (checkAny(pos)) return getAny();else if (checkSC(pos)) return getSC();
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function checkTsets(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
while (tokens[i - 1].type !== TokenType.Newline && (l = checkTset(i))) {
i += l;
}
return i - start;
}
/**
* @return {Array}
*/
function getTsets() {
var content = [];
var t = void 0;
while (tokens[pos - 1].type !== TokenType.Newline && (t = getTset())) {
if (typeof t.content === 'string') content.push(t);else content = content.concat(t);
}
return content;
}
/**
* Check if token is an unary (arithmetical) sign (`+` or `-`)
* @param {number} i Token's index number
* @return {number} `1` if token is an unary sign, `0` if not
*/
function checkUnary(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.HyphenMinus || tokens[i].type === TokenType.PlusSign) {
return 1;
}
return 0;
}
/**
* Get node with an unary (arithmetical) sign (`+` or `-`)
* @return {Array} `['unary', x]` where `x` is an unary sign
* converted to string.
*/
function getUnary() {
var type = NodeType.OperatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a unicode range (single or multiple <urange> nodes)
* @param {number} i Token's index
* @return {number} Unicode range node's length
*/
function checkUnicodeRange(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkUrange(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
var comma = checkDelim(i + spaceBefore);
if (!comma) break;
var spaceAfter = checkSC(i + spaceBefore + comma);
if (l = checkUrange(i + spaceBefore + comma + spaceAfter)) {
i += spaceBefore + comma + spaceAfter + l;
} else break;
}
return i - start;
}
/**
* Get a unicode range node
* @return {Node}
*/
function getUnicodeRange() {
var type = NodeType.UnicodeRangeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
while (pos < tokensLength) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkDelim(pos)) content.push(getDelim());else if (checkUrange(pos)) content.push(getUrange());else break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is unit
* @param {Number} i Token's index number
* @return {Number}
*/
function checkUnit(i) {
var units = ['em', 'ex', 'ch', 'rem', 'vh', 'vw', 'vmin', 'vmax', 'px', 'mm', 'q', 'cm', 'in', 'pt', 'pc', 'deg', 'grad', 'rad', 'turn', 's', 'ms', 'Hz', 'kHz', 'dpi', 'dpcm', 'dppx'];
return units.indexOf(tokens[i].value) !== -1 ? 1 : 0;
}
/**
* Get unit node of type ident
* @return {Node} An ident node containing the unit value
*/
function getUnit() {
var type = NodeType.IdentType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a u-range (part of a unicode-range)
* (1) `U+416`
* (2) `U+400-4ff`
* (3) `U+4??`
* @param {number} i Token's index
* @return {number} Urange node's length
*/
function checkUrange(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
// Check for unicode prefix (u+ or U+)
if (tokens[i].value === 'U' || tokens[i].value === 'u') i += 1;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].value === '+') i += 1;else return 0;
while (i < tokensLength) {
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = _checkUnicodeWildcard(i)) i += l;else break;
}
tokens[start].urangeEnd = i - 1;
return i - start;
}
/**
* Get a u-range node (part of a unicode-range)
* @return {Node}
*/
function getUrange() {
var startPos = pos;
var type = NodeType.UrangeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content = joinValues(startPos, tokens[startPos].urangeEnd);
pos = tokens[startPos].urangeEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check for unicode wildcard characters `?`
* @param {number} i Token's index
* @return {number} Wildcard length
*/
function _checkUnicodeWildcard(i) {
var start = i;
if (i >= tokensLength) return 0;
while (i < tokensLength) {
if (tokens[i].type === TokenType.QuestionMark) i += 1;else break;
}
return i - start;
}
/**
* Check if token is part of URI, e.g. `url('/css/styles.css')`
* @param {number} i Token's index number
* @returns {number} Length of URI
*/
function checkUri(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i].value !== 'url') return 0;
// Skip `url`.
i++;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
// Store the opening parenthesis token as we will reference it's `right`
// property to determine when the parentheses close
var leftParenthesis = tokens[i];
// Skip `(`.
i++;
// Determine the type of URI
while (i < leftParenthesis.right) {
if (l = checkUri1(i)) {
i += l;
tokens[start].uriType = 1; // Raw based URI (without quotes)
} else if (l = checkUri2(i)) {
i += l;
tokens[start].uriType = 2; // Non-raw based URI (with quotes)
} else return 0;
}
// Skip `)`.
i++;
return i - start;
}
/**
* Get specific type of URI node
* @return {Node} Specific type of URI node
*/
function getUri() {
var startPos = pos;
var type = NodeType.UriType;
var token = tokens[startPos];
var line = token.ln;
var column = token.col;
var content = [];
var end = void 0;
var uriType = tokens[startPos].uriType;
// Skip `url` and `(`.
pos += 2;
if (uriType === 1) content = content.concat(getUri1());else if (uriType === 2) content = content.concat(getUri2());else end = getLastPosition(content, line, column, 4);
if (!end) end = getLastPosition(content, line, column, 1);
// Skip `)`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token type is valid URI character
* @param {number} i Token's index number
* @return {number} Length of raw node
*/
function checkUriRawCharacters(i) {
var start = i;
var l = void 0;
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else {
switch (tokens[i].type) {
case TokenType.ExclamationMark:
case TokenType.NumberSign:
case TokenType.DollarSign:
case TokenType.PercentSign:
case TokenType.Ampersand:
case TokenType.Asterisk:
case TokenType.PlusSign:
case TokenType.Comma:
case TokenType.HyphenMinus:
case TokenType.FullStop:
case TokenType.Solidus:
case TokenType.Colon:
case TokenType.Semicolon:
case TokenType.LessThanSign:
case TokenType.EqualsSign:
case TokenType.GreaterThanSign:
case TokenType.QuotationMark:
case TokenType.CommercialAt:
case TokenType.LeftSquareBracket:
case TokenType.RightSquareBracket:
case TokenType.CircumflexAccent:
case TokenType.LowLine:
case TokenType.LeftCurlyBracket:
case TokenType.VerticalLine:
case TokenType.RightCurlyBracket:
case TokenType.Tilde:
i += 1;
break;
default:
return 0;
}
}
return i - start;
}
/**
* Check if content of URI can be contained within a raw node
* @param {number} i Token's index number
* @return {number} Length of raw node
*/
function checkUriRaw(i) {
var start = i;
var l = void 0;
while (i < tokensLength) {
if (checkInterpolation(i) || checkVariable(i)) break;else if (l = checkUriRawCharacters(i)) i += l;else break;
}
tokens[start].uri_raw_end = i;
return i - start;
}
/**
* Get a raw node
* @return {Node}
*/
function getUriRaw() {
var startPos = pos;
var type = NodeType.RawType;
var token = tokens[startPos];
var line = token.ln;
var column = token.col;
var content = [];
var l = void 0;
while (pos < tokens[startPos].uri_raw_end) {
if (checkInterpolation(pos) || checkVariable(pos)) break;else if (l = checkUriRawCharacters(pos)) pos += l;else break;
}
content = joinValues(startPos, pos - 1);
return newNode(type, content, line, column);
}
/**
* Check for a raw (without quotes) URI
* (1) http://foo.com/bar.png
* (2) http://foo.com/#{$bar}.png
* (3) #{$foo}/bar.png
* (4) #{$foo}
* @param {number} i Token's index number
* @return {number} Length of URI node
*/
function checkUri1(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
while (i < tokensLength) {
if (l = checkInterpolation(i) || checkUriRaw(i)) i += l;else break;
}
if (l = checkSC(i)) i += l;
// Check that we are at the end of the uri
if (i < tokens[start - 1].right) return 0;
tokens[start].uri_end = i;
return i - start;
}
/**
* Get a raw (without quotes) URI
node
* @return {Array}
*/
function getUri1() {
var startPos = pos;
var content = [];
if (checkSC(pos)) content = content.concat(getSC());
while (pos < tokens[startPos].uri_end) {
if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkUriRaw(pos)) content.push(getUriRaw());else break;
}
if (checkSC(pos)) content = content.concat(getSC());
return content;
}
/**
* Check for a non-raw (with quotes) URI
* (1) 'http://foo.com/bar.png'
* (2) 'http://foo.com/'#{$bar}.png
* (3) #{$foo}'/bar.png'
* @param {number} i Token's index number
* @return {number} Length of URI node
*/
function checkUri2(i) {
var start = i;
var l = void 0;
while (i < tokensLength) {
if (l = checkSC(i)) i += l;else if (l = checkString(i)) i += l;else if (l = checkFunction(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = checkIdentOrInterpolation(i)) i += l;else if (l = checkVariable(i)) i += l;else break;
}
// Check that we are at the end of the uri
if (i < tokens[start - 1].right) return 0;
tokens[start].uri_end = i;
return i - start;
}
/**
* Get a non-raw (with quotes) URI node
* @return {Array}
*/
function getUri2() {
var startPos = pos;
var content = [];
while (pos < tokens[startPos].uri_end) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkUnary(pos)) content.push(getUnary());else if (_checkValue(pos)) content.push(_getValue());else break;
}
return content;
}
/**
* Check if token is part of a value
* @param {number} i Token's index number
* @return {number} Length of the value
*/
function checkValue(i) {
var start = i;
var l = void 0;
var s = void 0;
var _i = void 0;
while (i < tokensLength) {
if (checkDeclDelim(i)) break;
if (l = checkBlock(i)) {
i += l;
break;
}
s = checkS(i);
_i = i + s;
if (l = _checkValue(_i)) i += l + s;
if (!l || checkBlock(i - l)) break;
}
return i - start;
}
/**
* @return {Array}
*/
function getValue() {
var type = NodeType.ValueType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var _pos = void 0;
var s = void 0;
while (pos < tokensLength) {
if (checkDeclDelim(pos)) break;
s = checkS(pos);
_pos = pos + s;
if (checkDeclDelim(_pos)) break;
if (checkBlock(pos)) {
content.push(getBlock());
break;
}
if (!_checkValue(_pos)) break;
if (s) content.push(getS());
content.push(_getValue());
if (checkBlock(_pos)) break;
}
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @return {number}
*/
function _checkValue(i) {
var l = void 0;
if (l = checkInterpolation(i)) tokens[i].value_child = 1;else if (l = checkVariable(i)) tokens[i].value_child = 2;else if (l = checkVhash(i)) tokens[i].value_child = 3;else if (l = checkBlock(i)) tokens[i].value_child = 4;else if (l = checkAtkeyword(i)) tokens[i].value_child = 5;else if (l = checkOperator(i)) tokens[i].value_child = 6;else if (l = checkImportant(i)) tokens[i].value_child = 7;else if (l = checkGlobal(i)) tokens[i].value_child = 8;else if (l = checkDefault(i)) tokens[i].value_child = 9;else if (l = checkProgid(i)) tokens[i].value_child = 10;else if (l = checkAny(i)) tokens[i].value_child = 11;else if (l = checkParentSelector(i)) tokens[i].value_child = 12;
return l;
}
/**
* @return {Array}
*/
function _getValue() {
var childType = tokens[pos].value_child;
if (childType === 1) return getInterpolation();
if (childType === 2) return getVariable();
if (childType === 3) return getVhash();
if (childType === 4) return getBlock();
if (childType === 5) return getAtkeyword();
if (childType === 6) return getOperator();
if (childType === 7) return getImportant();
if (childType === 8) return getGlobal();
if (childType === 9) return getDefault();
if (childType === 10) return getProgid();
if (childType === 11) return getAny();
if (childType === 12) return getParentSelector();
}
/**
* @param {number} i Token's index number
* @returns {number} Length of the value
*/
function checkSingleValue(i) {
var start = i;
var l = void 0;
var s = void 0;
var _i = void 0;
while (i < tokensLength) {
if (checkDeclDelim(i) || checkDelim(i)) break;
if (l = checkBlock(i)) {
i += l;
break;
}
s = checkSC(i);
_i = i + s;
if (l = _checkValue(_i)) i += l + s;
if (!l || checkBlock(i - l)) break;
}
return i - start;
}
/**
* @returns {Array}
*/
function getSingleValue() {
var type = NodeType.ValueType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var _pos = void 0;
var s = void 0;
while (pos < tokensLength) {
if (checkDeclDelim(pos) || checkDelim(pos)) break;
s = checkSC(pos);
_pos = pos + s;
if (checkDeclDelim(_pos) || checkDelim(_pos)) break;
if (checkBlock(pos)) {
content.push(getBlock());
break;
}
if (!_checkValue(_pos)) break;
if (s) content.push(getS());
content.push(_getValue());
if (checkBlock(_pos)) break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is part of a variable
* @param {number} i Token's index number
* @return {number} Length of the variable
*/
function checkVariable(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
// Skip `$`.
if (tokens[i].type === TokenType.DollarSign) i++;else return 0;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a variable
* @return {Array} `['variable', ['ident', x]]` where `x` is
* a variable name.
*/
function getVariable() {
var type = NodeType.VariableType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `$`.
pos++;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* Check if token is part of a variables list (e.g. `$values...`).
* @param {number} i Token's index number
* @return {number}
*/
function checkVariablesList(i) {
var d = 0; // Number of dots
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkVariable(i)) i += l;else return 0;
while (i < tokensLength && tokens[i].type === TokenType.FullStop) {
d++;
i++;
}
return d === 3 ? l + d : 0;
}
/**
* Get node with a variables list
* @return {Array} `['variableslist', ['variable', ['ident', x]]]` where
* `x` is a variable name.
*/
function getVariablesList() {
var type = NodeType.VariablesListType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getVariable()];
var end = getLastPosition(content, line, column, 3);
// Skip `...`.
pos += 3;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
* some value
* @param {number} i Token's index number
* @return {number}
*/
function checkVhash(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
// Skip `#`.
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
if (l = checkNmName2(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a hexadecimal number (e.g. `#fff`) inside some value
* @return {Array} `['vhash', x]` where `x` is a hexadecimal number
* converted to string (without `#`, e.g. `'fff'`).
*/
function getVhash() {
var type = NodeType.VhashType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `#`.
pos++;
var content = getNmName2();
var end = getLastPosition(content, line, column + 1);
return newNode(type, content, line, column, end);
}
function checkSelectorsGroup(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkSelector(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
var comma = checkDelim(i + spaceBefore);
if (!comma) break;
var spaceAfter = checkMultilineSC(i + spaceBefore + comma);
var spaceEnd = spaceAfter ? checkMultilineSC(i + spaceBefore + comma + spaceAfter) : 0;
if (l = checkSelector(i + spaceBefore + comma + spaceAfter + spaceEnd)) i += spaceBefore + comma + spaceAfter + spaceEnd + l;else break;
}
tokens[start].selectorsGroupEnd = i;
return i - start;
}
function getSelectorsGroup() {
var selectorsGroup = [];
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
selectorsGroup.push(getSelector());
while (pos < selectorsGroupEnd) {
selectorsGroup = selectorsGroup.concat(getMultilineSC(), getDelim(), getMultilineSC(), getSelector());
}
return selectorsGroup;
}
function checkSelector(i) {
var l = void 0;
if (l = checkSelector1(i)) tokens[i].selectorType = 1;else if (l = checkSelector2(i)) tokens[i].selectorType = 2;
return l;
}
function getSelector() {
var selectorType = tokens[pos].selectorType;
if (selectorType === 1) return getSelector1();else return getSelector2();
}
/**
* Checks for selector which starts with a compound selector.
*/
function checkSelector1(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkCompoundSelector(i)) i += l;else return 0;
while (i < tokensLength) {
var space = checkSC(i);
var comma = checkCombinator(i + space);
if (!space && !comma) break;
if (comma) {
i += space + comma;
space = checkSC(i);
}
if (l = checkCompoundSelector(i + space)) i += space + l;else break;
}
tokens[start].selectorEnd = i;
return i - start;
}
function getSelector1() {
var type = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var selectorEnd = token.selectorEnd;
var content = getCompoundSelector();
while (pos < selectorEnd) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
}
return newNode(type, content, line, column);
}
/**
* Checks for a selector that starts with a combinator.
*/
function checkSelector2(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkCombinator(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
if (l = checkCompoundSelector(i + spaceBefore)) i += spaceBefore + l;else break;
var spaceAfter = checkSC(i);
var comma = checkCombinator(i + spaceAfter);
if (!spaceAfter && !comma) break;
if (comma) {
i += spaceAfter + comma;
}
}
tokens[start].selectorEnd = i;
return i - start;
}
function getSelector2() {
var type = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var selectorEnd = token.selectorEnd;
var content = [getCombinator()];
while (pos < selectorEnd) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
}
return newNode(type, content, line, column);
}
function checkCompoundSelector(i) {
var l = void 0;
if (l = checkCompoundSelector1(i)) {
tokens[i].compoundSelectorType = 1;
} else if (l = checkCompoundSelector2(i)) {
tokens[i].compoundSelectorType = 2;
}
return l;
}
function getCompoundSelector() {
var type = tokens[pos].compoundSelectorType;
if (type === 1) return getCompoundSelector1();
if (type === 2) return getCompoundSelector2();
}
/**
* Check for compound selectors that start with either a type selector,
* placeholder or parent selector with extension
* (1) `foo.bar`
* (2) `foo[attr=val]`
* (3) `foo:first-of-type`
* (4) `foo%bar`
* @param {number} i Token's index
* @return {number} Compound selector's length
*/
function checkCompoundSelector1(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkUniversalSelector(i) || checkTypeSelector(i) || checkPlaceholder(i) || checkParentSelectorWithExtension(i)) i += l;else return 0;
while (i < tokensLength) {
var _l2 = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i) || checkPlaceholder(i) || checkInterpolation(i);
if (_l2) i += _l2;else break;
}
tokens[start].compoundSelectorEnd = i;
return i - start;
}
/**
* @return {Array} An array of nodes that make up the compound selector
*/
function getCompoundSelector1() {
var sequence = [];
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
if (checkUniversalSelector(pos)) sequence.push(getUniversalSelector());else if (checkTypeSelector(pos)) sequence.push(getTypeSelector());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkParentSelectorWithExtension(pos)) sequence = sequence.concat(getParentSelectorWithExtension());
while (pos < compoundSelectorEnd) {
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkInterpolation(pos)) sequence.push(getInterpolation());else break;
}
return sequence;
}
/**
* Check for all other compound selectors
* (1) `.foo.bar`
* (2) `.foo[attr=val]`
* (3) `.foo:first-of-type`
* (4) `.foo%bar`
* (5) `.foo#{$bar}`
* @param {number} i Token's index
* @return {number} Compound selector's length
*/
function checkCompoundSelector2(i) {
if (i >= tokensLength) return 0;
var start = i;
while (i < tokensLength) {
var l = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i) || checkPlaceholder(i) || checkInterpolation(i);
if (l) i += l;else break;
}
tokens[start].compoundSelectorEnd = i;
return i - start;
}
/**
* @return {Array} An array of nodes that make up the compound selector
*/
function getCompoundSelector2() {
var sequence = [];
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
while (pos < compoundSelectorEnd) {
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkInterpolation(pos)) sequence.push(getInterpolation());else break;
}
return sequence;
}
function checkUniversalSelector(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (tokens[i].type === TokenType.Asterisk) i++;else return 0;
return i - start;
}
function getUniversalSelector() {
var type = NodeType.UniversalSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var end = void 0;
if (checkNamePrefix(pos)) {
content.push(getNamePrefix());
end = getLastPosition(content, line, column, 1);
}
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a type selector
* @param {number} i Token's index
* @return {number} Type selector's length
*/
function checkTypeSelector(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
/**
* Get type selector node
* @return {Node}
*/
function getTypeSelector() {
var type = NodeType.TypeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkNamePrefix(pos)) content.push(getNamePrefix());
content = content.concat(getIdentOrInterpolation());
return newNode(type, content, line, column);
}
function checkAttributeSelector(i) {
var l = void 0;
if (l = checkAttributeSelector1(i)) tokens[i].attributeSelectorType = 1;else if (l = checkAttributeSelector2(i)) tokens[i].attributeSelectorType = 2;
return l;
}
function getAttributeSelector() {
var type = tokens[pos].attributeSelectorType;
if (type === 1) return getAttributeSelector1();else return getAttributeSelector2();
}
/**
* (1) `[panda=nani]`
* (2) `[panda='nani']`
* (3) `[panda='nani' i]`
*
*/
function checkAttributeSelector1(i) {
var start = i;
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeName(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeMatch(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeValue(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeFlags(i)) {
i += l;
if (l = checkSC(i)) i += l;
}
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
return i - start;
}
function getAttributeSelector1() {
var type = NodeType.AttributeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `[`.
pos++;
content = content.concat(getSC(), getAttributeName(), getSC(), getAttributeMatch(), getSC(), getAttributeValue(), getSC());
if (checkAttributeFlags(pos)) {
content.push(getAttributeFlags());
content = content.concat(getSC());
}
// Skip `]`.
pos++;
var end = getLastPosition(content, line, column, 1);
return newNode(type, content, line, column, end);
}
/**
* (1) `[panda]`
*/
function checkAttributeSelector2(i) {
var start = i;
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeName(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
return i - start;
}
function getAttributeSelector2() {
var type = NodeType.AttributeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `[`.
pos++;
content = content.concat(getSC(), getAttributeName(), getSC());
// Skip `]`.
pos++;
var end = getLastPosition(content, line, column, 1);
return newNode(type, content, line, column, end);
}
function checkAttributeName(i) {
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
function getAttributeName() {
var type = NodeType.AttributeNameType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkNamePrefix(pos)) content.push(getNamePrefix());
content = content.concat(getIdentOrInterpolation());
return newNode(type, content, line, column);
}
function checkAttributeMatch(i) {
var l = void 0;
if (l = checkAttributeMatch1(i)) tokens[i].attributeMatchType = 1;else if (l = checkAttributeMatch2(i)) tokens[i].attributeMatchType = 2;
return l;
}
function getAttributeMatch() {
var type = tokens[pos].attributeMatchType;
if (type === 1) return getAttributeMatch1();else return getAttributeMatch2();
}
function checkAttributeMatch1(i) {
var start = i;
var type = tokens[i].type;
if (type === TokenType.Tilde || type === TokenType.VerticalLine || type === TokenType.CircumflexAccent || type === TokenType.DollarSign || type === TokenType.Asterisk) i++;else return 0;
if (tokens[i].type === TokenType.EqualsSign) i++;else return 0;
return i - start;
}
function getAttributeMatch1() {
var type = NodeType.AttributeMatchType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos].value + tokens[pos + 1].value;
pos += 2;
return newNode(type, content, line, column);
}
function checkAttributeMatch2(i) {
if (tokens[i].type === TokenType.EqualsSign) return 1;else return 0;
}
function getAttributeMatch2() {
var type = NodeType.AttributeMatchType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '=';
pos++;
return newNode(type, content, line, column);
}
function checkAttributeValue(i) {
return checkString(i) || checkIdentOrInterpolation(i);
}
function getAttributeValue() {
var type = NodeType.AttributeValueType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkString(pos)) content.push(getString());else content = content.concat(getIdentOrInterpolation());
return newNode(type, content, line, column);
}
function checkAttributeFlags(i) {
return checkIdentOrInterpolation(i);
}
function getAttributeFlags() {
var type = NodeType.AttributeFlagsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = getIdentOrInterpolation();
return newNode(type, content, line, column);
}
function checkNamePrefix(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkNamePrefix1(i)) tokens[i].namePrefixType = 1;else if (l = checkNamePrefix2(i)) tokens[i].namePrefixType = 2;
return l;
}
function getNamePrefix() {
var type = tokens[pos].namePrefixType;
if (type === 1) return getNamePrefix1();else return getNamePrefix2();
}
/**
* (1) `panda|`
* (2) `panda<comment>|`
*/
function checkNamePrefix1(i) {
var start = i;
var l = void 0;
if (l = checkNamespacePrefix(i)) i += l;else return 0;
if (l = checkCommentML(i)) i += l;
if (l = checkNamespaceSeparator(i)) i += l;else return 0;
return i - start;
}
function getNamePrefix1() {
var type = NodeType.NamePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(getNamespacePrefix());
if (checkCommentML(pos)) content.push(getCommentML());
content.push(getNamespaceSeparator());
return newNode(type, content, line, column);
}
/**
* (1) `|`
*/
function checkNamePrefix2(i) {
return checkNamespaceSeparator(i);
}
function getNamePrefix2() {
var type = NodeType.NamePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getNamespaceSeparator()];
return newNode(type, content, line, column);
}
/**
* (1) `*`
* (2) `panda`
*/
function checkNamespacePrefix(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (tokens[i].type === TokenType.Asterisk) return 1;else if (l = checkIdentOrInterpolation(i)) return l;else return 0;
}
function getNamespacePrefix() {
var type = NodeType.NamespacePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (token.type === TokenType.Asterisk) {
var asteriskNode = newNode(NodeType.IdentType, '*', line, column);
content.push(asteriskNode);
pos++;
} else if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());
return newNode(type, content, line, column);
}
/**
* (1) `|`
*/
function checkNamespaceSeparator(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.VerticalLine) return 0;
// Return false if `|=` - [attr|=value]
if (tokens[i + 1] && tokens[i + 1].type === TokenType.EqualsSign) return 0;
return 1;
}
function getNamespaceSeparator() {
var type = NodeType.NamespaceSeparatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '|';
pos++;
return newNode(type, content, line, column);
}
module.exports = function (_tokens, context) {
tokens = _tokens;
tokensLength = tokens.length;
pos = 0;
return contexts[context]();
};
/***/ }),
/* 24 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
module.exports = function (css, tabSize) {
var TokenType = __webpack_require__(13);
var tokens = [];
var urlMode = false;
var c = void 0; // Current character
var cn = void 0; // Next character
var pos = 0;
var tn = 0;
var ln = 1;
var col = 1;
var Punctuation = {
' ': TokenType.Space,
'\n': TokenType.Newline,
'\r': TokenType.Newline,
'\t': TokenType.Tab,
'!': TokenType.ExclamationMark,
'"': TokenType.QuotationMark,
'#': TokenType.NumberSign,
'$': TokenType.DollarSign,
'%': TokenType.PercentSign,
'&': TokenType.Ampersand,
'\'': TokenType.Apostrophe,
'(': TokenType.LeftParenthesis,
')': TokenType.RightParenthesis,
'*': TokenType.Asterisk,
'+': TokenType.PlusSign,
',': TokenType.Comma,
'-': TokenType.HyphenMinus,
'.': TokenType.FullStop,
'/': TokenType.Solidus,
':': TokenType.Colon,
';': TokenType.Semicolon,
'<': TokenType.LessThanSign,
'=': TokenType.EqualsSign,
'==': TokenType.EqualitySign,
'!=': TokenType.InequalitySign,
'>': TokenType.GreaterThanSign,
'?': TokenType.QuestionMark,
'@': TokenType.CommercialAt,
'[': TokenType.LeftSquareBracket,
']': TokenType.RightSquareBracket,
'^': TokenType.CircumflexAccent,
'_': TokenType.LowLine,
'{': TokenType.LeftCurlyBracket,
'|': TokenType.VerticalLine,
'}': TokenType.RightCurlyBracket,
'~': TokenType.Tilde,
'`': TokenType.Backtick
};
/**
* Add a token to the token list
* @param {string} type
* @param {string} value
*/
function pushToken(type, value, column) {
tokens.push({
tn: tn++,
ln: ln,
col: column,
type: type,
value: value
});
}
/**
* Check if a character is a decimal digit
* @param {string} c Character
* @returns {boolean}
*/
function isDecimalDigit(c) {
return '0123456789'.indexOf(c) >= 0;
}
/**
* Parse spaces
* @param {string} css Unparsed part of CSS string
*/
function parseSpaces(css) {
var start = pos;
// Read the string until we meet a non-space character:
for (; pos < css.length; pos++) {
if (css.charAt(pos) !== ' ') break;
}
// Add a substring containing only spaces to tokens:
pushToken(TokenType.Space, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Parse a string within quotes
* @param {string} css Unparsed part of CSS string
* @param {string} q Quote (either `'` or `"`)
*/
function parseString(css, q) {
var start = pos;
// Read the string until we meet a matching quote:
for (pos++; pos < css.length; pos++) {
// Skip escaped quotes:
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) === q) break;
}
// Add the string (including quotes) to tokens:
var type = q === '"' ? TokenType.StringDQ : TokenType.StringSQ;
pushToken(type, css.substring(start, pos + 1), col);
col += pos - start;
}
/**
* Parse numbers
* @param {string} css Unparsed part of CSS string
*/
function parseDecimalNumber(css) {
var start = pos;
// Read the string until we meet a character that's not a digit:
for (; pos < css.length; pos++) {
if (!isDecimalDigit(css.charAt(pos))) break;
}
// Add the number to tokens:
pushToken(TokenType.DecimalNumber, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Parse identifier
* @param {string} css Unparsed part of CSS string
*/
function parseIdentifier(css) {
var start = pos;
// Skip all opening slashes:
while (css.charAt(pos) === '/') {
pos++;
} // Read the string until we meet a punctuation mark:
for (; pos < css.length; pos++) {
// Skip all '\':
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) in Punctuation) break;
}
var ident = css.substring(start, pos--);
// Enter url mode if parsed substring is `url`:
if (!urlMode && ident === 'url' && css.charAt(pos + 1) === '(') {
urlMode = true;
}
// Add identifier to tokens:
pushToken(TokenType.Identifier, ident, col);
col += pos - start;
}
/**
* Parse equality sign
*/
function parseEquality() {
pushToken(TokenType.EqualitySign, '==', col);
pos++;
col++;
}
/**
* Parse inequality sign
*/
function parseInequality() {
pushToken(TokenType.InequalitySign, '!=', col);
pos++;
col++;
}
/**
* Parse a multiline comment
* @param {string} css Unparsed part of CSS string
*/
function parseMLComment(css) {
var start = pos;
var col_ = col;
// Get current indent level:
var il = 0;
for (var _pos = pos - 1; _pos > -1; _pos--) {
// TODO: Can be tabs:
if (css.charAt(_pos) === ' ') il++;else break;
}
for (pos += 2; pos < css.length; pos++) {
var ch = css.charAt(pos);
if (ch === '\n') {
var _pos2 = void 0;
// Get new line's indent level:
var _il = 0;
for (_pos2 = pos + 1; _pos2 < css.length; _pos2++) {
if (css.charAt(_pos2) === ' ') _il++;else break;
}
if (_il > il) {
col = 0;
pos += _pos2 - pos;
} else {
pos--;
break;
}
} else if (ch === '*' && css.charAt(pos + 1) === '/') {
pos++;
break;
}
}
// If CRLF is used, we need to adjust pos
if (css.charAt(pos) === '\r') pos--;
// Add full comment (including `/*`) to the list of tokens:
var comment = css.substring(start, pos + 1);
pushToken(TokenType.CommentML, comment, col_);
var newlines = comment.split('\n');
if (newlines.length > 1) {
ln += newlines.length - 1;
col = newlines[newlines.length - 1].length;
} else {
col += pos - start;
}
}
/**
* Parse a single line comment
* @param {string} css Unparsed part of CSS string
*/
function parseSLComment(css) {
var start = pos;
var col_ = col;
var _pos;
// Check if comment is the only token on the line, and if so,
// get current indent level:
var il = 0;
var onlyToken = false;
for (_pos = pos - 1; _pos > -1; _pos--) {
// TODO: Can be tabs:
if (css.charAt(_pos) === ' ') il++;else if (css.charAt(_pos) === '\n') {
onlyToken = true;
break;
} else break;
}
if (_pos === -1) onlyToken = true;
// Read the string until we meet comment end.
// Since we already know first 2 characters (`//`), start reading
// from `pos + 2`:
if (!onlyToken) {
for (pos += 2; pos < css.length; pos++) {
if (css.charAt(pos) === '\n' || css.charAt(pos) === '\r') {
break;
}
}
} else {
for (pos += 2; pos < css.length; pos++) {
var ch = css.charAt(pos);
if (ch === '\n') {
// Get new line's indent level:
var _il = 0;
for (_pos = pos + 1; _pos < css.length; _pos++) {
if (css.charAt(_pos) === ' ') _il++;else break;
}
if (_il > il) {
col = 0;
pos += _pos - pos;
} else {
break;
}
}
}
}
// If CRLF is used, we need to adjust pos
if (css.charAt(pos - 1) === '\r') pos--;
// Add comment (including `//` and line break) to the list of tokens:
var comment = css.substring(start, pos--);
pushToken(TokenType.CommentSL, comment, col_);
var newlines = comment.split('\n');
if (newlines.length > 1) {
ln += newlines.length - 1;
col = newlines[newlines.length - 1].length;
} else {
col += pos - start;
}
}
/**
* Convert a CSS string to a list of tokens
* @param {string} css CSS string
* @returns {Array} List of tokens
* @private
*/
function getTokens(css) {
// Parse string, character by character:
for (pos = 0; pos < css.length; col++, pos++) {
c = css.charAt(pos);
cn = css.charAt(pos + 1);
// If we meet `/*`, it's a start of a multiline comment.
// Parse following characters as a multiline comment:
if (c === '/' && cn === '*') {
parseMLComment(css);
}
// If we meet `//` and it is not a part of url:
else if (!urlMode && c === '/' && cn === '/') {
// If we're currently inside a block, treat `//` as a start
// of identifier. Else treat `//` as a start of a single-line
// comment:
parseSLComment(css);
}
// If current character is a double or single quote, it's a start
// of a string:
else if (c === '"' || c === "'") {
parseString(css, c);
}
// If current character is a space:
else if (c === ' ') {
parseSpaces(css);
}
// If current character is `=`, it must be combined with next `=`
else if (c === '=' && cn === '=') {
parseEquality(css);
}
// If we meet `!=`, this must be inequality
else if (c === '!' && cn === '=') {
parseInequality(css);
}
// If current character is a punctuation mark:
else if (c in Punctuation) {
// Check for CRLF here or just LF
if (c === '\r' && cn === '\n' || c === '\n') {
// If \r we know the next character is \n due to statement above
// so we push a CRLF token type to the token list and importantly
// skip the next character so as not to double count newlines or
// columns etc
if (c === '\r') {
pushToken(TokenType.Newline, '\r\n', col);
pos++; // If CRLF skip the next character and push crlf token
} else if (c === '\n') {
// If just a LF newline and not part of CRLF newline we can just
// push punctuation as usual
pushToken(Punctuation[c], c, col);
}
ln++; // Go to next line
col = 0; // Reset the column count
} else if (c !== '\r' && c !== '\n') {
// Handle all other punctuation and add to list of tokens
pushToken(Punctuation[c], c, col);
} // Go to next line
if (c === ')') urlMode = false; // Exit url mode
else if (c === '\t' && tabSize > 1) col += tabSize - 1;
}
// If current character is a decimal digit:
else if (isDecimalDigit(c)) {
parseDecimalNumber(css);
}
// If current character is anything else:
else {
parseIdentifier(css);
}
}
return tokens;
}
return getTokens(css);
};
/***/ }),
/* 25 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
exports.__esModule = true;
exports.default = {
mark: __webpack_require__(26),
parse: __webpack_require__(27),
stringify: __webpack_require__(6),
tokenizer: __webpack_require__(28)
};
module.exports = exports['default'];
/***/ }),
/* 26 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var TokenType = __webpack_require__(13);
module.exports = function () {
/**
* Mark whitespaces and comments
*/
function markSC(tokens) {
var tokensLength = tokens.length;
var ws = -1; // Flag for whitespaces
var sc = -1; // Flag for whitespaces and comments
var t = void 0; // Current token
// For every token in the token list, mark spaces and line breaks
// as spaces (set both `ws` and `sc` flags). Mark multiline comments
// with `sc` flag.
// If there are several spaces or tabs or line breaks or multiline
// comments in a row, group them: take the last one's index number
// and save it to the first token in the group as a reference:
// e.g., `ws_last = 7` for a group of whitespaces or `sc_last = 9`
// for a group of whitespaces and comments.
for (var i = 0; i < tokensLength; i++) {
t = tokens[i];
switch (t.type) {
case TokenType.Space:
case TokenType.Tab:
case TokenType.Newline:
t.ws = true;
t.sc = true;
if (ws === -1) ws = i;
if (sc === -1) sc = i;
break;
case TokenType.CommentML:
case TokenType.CommentSL:
if (ws !== -1) {
tokens[ws].ws_last = i - 1;
ws = -1;
}
t.sc = true;
break;
default:
if (ws !== -1) {
tokens[ws].ws_last = i - 1;
ws = -1;
}
if (sc !== -1) {
tokens[sc].sc_last = i - 1;
sc = -1;
}
}
}
if (ws !== -1) tokens[ws].ws_last = i - 1;
if (sc !== -1) tokens[sc].sc_last = i - 1;
}
/**
* Pair brackets
*/
function markBrackets(tokens) {
var tokensLength = tokens.length;
var ps = []; // Parentheses
var sbs = []; // Square brackets
var cbs = []; // Curly brackets
var t = void 0; // Current token
// For every token in the token list, if we meet an opening (left)
// bracket, push its index number to a corresponding array.
// If we then meet a closing (right) bracket, look at the corresponding
// array. If there are any elements (records about previously met
// left brackets), take a token of the last left bracket (take
// the last index number from the array and find a token with
// this index number) and save right bracket's index as a reference:
for (var i = 0; i < tokensLength; i++) {
t = tokens[i];
switch (t.type) {
case TokenType.LeftParenthesis:
ps.push(i);
break;
case TokenType.RightParenthesis:
if (ps.length) {
t.left = ps.pop();
tokens[t.left].right = i;
}
break;
case TokenType.LeftSquareBracket:
sbs.push(i);
break;
case TokenType.RightSquareBracket:
if (sbs.length) {
t.left = sbs.pop();
tokens[t.left].right = i;
}
break;
case TokenType.LeftCurlyBracket:
cbs.push(i);
break;
case TokenType.RightCurlyBracket:
if (cbs.length) {
t.left = cbs.pop();
tokens[t.left].right = i;
}
break;
}
}
}
return function (tokens) {
markBrackets(tokens);
markSC(tokens);
};
}();
/***/ }),
/* 27 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var Node = __webpack_require__(1);
var NodeType = __webpack_require__(15);
var TokenType = __webpack_require__(13);
var tokens = void 0;
var tokensLength = void 0;
var pos = void 0;
var contexts = {
'arguments': function _arguments() {
return checkArguments(pos) && getArguments();
},
'atkeyword': function atkeyword() {
return checkAtkeyword(pos) && getAtkeyword();
},
'atrule': function atrule() {
return checkAtrule(pos) && getAtrule();
},
'attributeSelector': function attributeSelector() {
return checkAttributeSelector(pos) && getAttributeSelector();
},
'block': function block() {
return checkBlock(pos) && getBlock();
},
'brackets': function brackets() {
return checkBrackets(pos) && getBrackets();
},
'class': function _class() {
return checkClass(pos) && getClass();
},
'combinator': function combinator() {
return checkCombinator(pos) && getCombinator();
},
'commentML': function commentML() {
return checkCommentML(pos) && getCommentML();
},
'commentSL': function commentSL() {
return checkCommentSL(pos) && getCommentSL();
},
'condition': function condition() {
return checkCondition(pos) && getCondition();
},
'conditionalStatement': function conditionalStatement() {
return checkConditionalStatement(pos) && getConditionalStatement();
},
'declaration': function declaration() {
return checkDeclaration(pos) && getDeclaration();
},
'declDelim': function declDelim() {
return checkDeclDelim(pos) && getDeclDelim();
},
'default': function _default() {
return checkDefault(pos) && getDefault();
},
'delim': function delim() {
return checkDelim(pos) && getDelim();
},
'dimension': function dimension() {
return checkDimension(pos) && getDimension();
},
'expression': function expression() {
return checkExpression(pos) && getExpression();
},
'extend': function extend() {
return checkExtend(pos) && getExtend();
},
'function': function _function() {
return checkFunction(pos) && getFunction();
},
'global': function global() {
return checkGlobal(pos) && getGlobal();
},
'ident': function ident() {
return checkIdent(pos) && getIdent();
},
'important': function important() {
return checkImportant(pos) && getImportant();
},
'include': function include() {
return checkInclude(pos) && getInclude();
},
'interpolation': function interpolation() {
return checkInterpolation(pos) && getInterpolation();
},
'loop': function loop() {
return checkLoop(pos) && getLoop();
},
'mixin': function mixin() {
return checkMixin(pos) && getMixin();
},
'namespace': function namespace() {
return checkNamespace(pos) && getNamespace();
},
'number': function number() {
return checkNumber(pos) && getNumber();
},
'operator': function operator() {
return checkOperator(pos) && getOperator();
},
'optional': function optional() {
return checkOptional(pos) && getOptional();
},
'parentheses': function parentheses() {
return checkParentheses(pos) && getParentheses();
},
'parentselector': function parentselector() {
return checkParentSelector(pos) && getParentSelector();
},
'percentage': function percentage() {
return checkPercentage(pos) && getPercentage();
},
'placeholder': function placeholder() {
return checkPlaceholder(pos) && getPlaceholder();
},
'progid': function progid() {
return checkProgid(pos) && getProgid();
},
'property': function property() {
return checkProperty(pos) && getProperty();
},
'propertyDelim': function propertyDelim() {
return checkPropertyDelim(pos) && getPropertyDelim();
},
'pseudoc': function pseudoc() {
return checkPseudoc(pos) && getPseudoc();
},
'pseudoe': function pseudoe() {
return checkPseudoe(pos) && getPseudoe();
},
'ruleset': function ruleset() {
return checkRuleset(pos) && getRuleset();
},
's': function s() {
return checkS(pos) && getS();
},
'selector': function selector() {
return checkSelector(pos) && getSelector();
},
'shash': function shash() {
return checkShash(pos) && getShash();
},
'string': function string() {
return checkString(pos) && getString();
},
'stylesheet': function stylesheet() {
return checkStylesheet(pos) && getStylesheet();
},
'typeSelector': function typeSelector() {
return checkTypeSelector(pos) && getTypeSelector();
},
'unary': function unary() {
return checkUnary(pos) && getUnary();
},
'unicodeRange': function unicodeRange() {
return checkUnicodeRange(pos) && getUnicodeRange();
},
'universalSelector': function universalSelector() {
return checkUniversalSelector(pos) && getUniversalSelector();
},
'urange': function urange() {
return checkUrange(pos) && getUrange();
},
'uri': function uri() {
return checkUri(pos) && getUri();
},
'value': function value() {
return checkValue(pos) && getValue();
},
'variable': function variable() {
return checkVariable(pos) && getVariable();
},
'variableslist': function variableslist() {
return checkVariablesList(pos) && getVariablesList();
},
'vhash': function vhash() {
return checkVhash(pos) && getVhash();
}
};
/**
* Stop parsing and display error
* @param {Number=} i Token's index number
*/
function throwError(i) {
var ln = tokens[i].ln;
throw { line: ln, syntax: 'scss' };
}
/**
* @param {Number} start
* @param {Number} finish
* @returns {String}
*/
function joinValues(start, finish) {
var s = '';
for (var i = start; i < finish + 1; i++) {
s += tokens[i].value;
}
return s;
}
/**
* @param {Number} start
* @param {Number} num
* @returns {String}
*/
function joinValues2(start, num) {
if (start + num - 1 >= tokensLength) return;
var s = '';
for (var i = 0; i < num; i++) {
s += tokens[start + i].value;
}
return s;
}
function getLastPosition(content, line, column, colOffset) {
return typeof content === 'string' ? getLastPositionForString(content, line, column, colOffset) : getLastPositionForArray(content, line, column, colOffset);
}
function getLastPositionForString(content, line, column, colOffset) {
var position = [];
if (!content) {
position = [line, column];
if (colOffset) position[1] += colOffset - 1;
return position;
}
var lastLinebreak = content.lastIndexOf('\n');
var endsWithLinebreak = lastLinebreak === content.length - 1;
var splitContent = content.split('\n');
var linebreaksCount = splitContent.length - 1;
var prevLinebreak = linebreaksCount === 0 || linebreaksCount === 1 ? -1 : content.length - splitContent[linebreaksCount - 1].length - 2;
// Line:
var offset = endsWithLinebreak ? linebreaksCount - 1 : linebreaksCount;
position[0] = line + offset;
// Column:
if (endsWithLinebreak) {
offset = prevLinebreak !== -1 ? content.length - prevLinebreak : content.length - 1;
} else {
offset = linebreaksCount !== 0 ? content.length - lastLinebreak - column - 1 : content.length - 1;
}
position[1] = column + offset;
if (!colOffset) return position;
if (endsWithLinebreak) {
position[0]++;
position[1] = colOffset;
} else {
position[1] += colOffset;
}
return position;
}
function getLastPositionForArray(content, line, column, colOffset) {
var position = void 0;
if (content.length === 0) {
position = [line, column];
} else {
var c = content[content.length - 1];
if (c.hasOwnProperty('end')) {
position = [c.end.line, c.end.column];
} else {
position = getLastPosition(c.content, line, column);
}
}
if (!colOffset) return position;
if (tokens[pos - 1] && tokens[pos - 1].type !== 'Newline') {
position[1] += colOffset;
} else {
position[0]++;
position[1] = 1;
}
return position;
}
function newNode(type, content, line, column, end) {
if (!end) end = getLastPosition(content, line, column);
return new Node({
type: type,
content: content,
start: {
line: line,
column: column
},
end: {
line: end[0],
column: end[1]
},
syntax: 'scss'
});
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkAny(i) {
var l = void 0;
if (l = checkBrackets(i)) tokens[i].any_child = 1;else if (l = checkParentheses(i)) tokens[i].any_child = 2;else if (l = checkString(i)) tokens[i].any_child = 3;else if (l = checkVariablesList(i)) tokens[i].any_child = 4;else if (l = checkVariable(i)) tokens[i].any_child = 5;else if (l = checkPlaceholder(i)) tokens[i].any_child = 6;else if (l = checkPercentage(i)) tokens[i].any_child = 7;else if (l = checkDimension(i)) tokens[i].any_child = 8;else if (l = checkUnicodeRange(i)) tokens[i].any_child = 9;else if (l = checkNumber(i)) tokens[i].any_child = 10;else if (l = checkUri(i)) tokens[i].any_child = 11;else if (l = checkExpression(i)) tokens[i].any_child = 12;else if (l = checkFunctionsList(i)) tokens[i].any_child = 13;else if (l = checkFunction(i)) tokens[i].any_child = 14;else if (l = checkInterpolation(i)) tokens[i].any_child = 15;else if (l = checkIdent(i)) tokens[i].any_child = 16;else if (l = checkClass(i)) tokens[i].any_child = 17;else if (l = checkUnary(i)) tokens[i].any_child = 18;else if (l = checkParentSelector(i)) tokens[i].any_child = 19;else if (l = checkImportant(i)) tokens[i].any_child = 20;else if (l = checkGlobal(i)) tokens[i].any_child = 21;else if (l = checkDefault(i)) tokens[i].any_child = 22;else if (l = checkOptional(i)) tokens[i].any_child = 23;
return l;
}
/**
* @returns {!Node}
*/
function getAny() {
var childType = tokens[pos].any_child;
if (childType === 1) return getBrackets();
if (childType === 2) return getParentheses();
if (childType === 3) return getString();
if (childType === 4) return getVariablesList();
if (childType === 5) return getVariable();
if (childType === 6) return getPlaceholder();
if (childType === 7) return getPercentage();
if (childType === 8) return getDimension();
if (childType === 9) return getUnicodeRange();
if (childType === 10) return getNumber();
if (childType === 11) return getUri();
if (childType === 12) return getExpression();
if (childType === 13) return getFunctionsList();
if (childType === 14) return getFunction();
if (childType === 15) return getInterpolation();
if (childType === 16) return getIdent();
if (childType === 17) return getClass();
if (childType === 18) return getUnary();
if (childType === 19) return getParentSelector();
if (childType === 20) return getImportant();
if (childType === 21) return getGlobal();
if (childType === 22) return getDefault();
if (childType === 23) return getOptional();
}
/**
* Check if token is part of mixin's arguments.
* @param {Number} i Token's index number
* @returns {Number} Length of arguments
*/
function checkArguments(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
// Skip `(`.
i++;
while (i < tokens[start].right) {
if (l = checkArgument(i)) i += l;else return 0;
}
return tokens[start].right - start + 1;
}
/**
* @returns {Array}
*/
function getArguments() {
var type = NodeType.ArgumentsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var body = void 0;
// Skip `(`.
pos++;
while (pos < tokensLength && tokens[pos].type !== TokenType.RightParenthesis) {
if (checkSingleValueDeclaration(pos)) {
content.push(getSingleValueDeclaration());
} else if (checkArgument(pos)) {
body = getArgument();
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
} else if (checkClass(pos)) content.push(getClass());else throwError(pos);
}
var end = getLastPosition(content, line, column, 1);
// Skip `)`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is valid to be part of arguments list
* @param {Number} i Token's index number
* @returns {Number} Length of argument
*/
function checkArgument(i) {
var l = void 0;
if (l = checkBrackets(i)) tokens[i].argument_child = 1;else if (l = checkParentheses(i)) tokens[i].argument_child = 2;else if (l = checkSingleValueDeclaration(i)) tokens[i].argument_child = 3;else if (l = checkFunctionsList(i)) tokens[i].argument_child = 4;else if (l = checkFunction(i)) tokens[i].argument_child = 5;else if (l = checkVariablesList(i)) tokens[i].argument_child = 6;else if (l = checkVariable(i)) tokens[i].argument_child = 7;else if (l = checkSC(i)) tokens[i].argument_child = 8;else if (l = checkDelim(i)) tokens[i].argument_child = 9;else if (l = checkDeclDelim(i)) tokens[i].argument_child = 10;else if (l = checkString(i)) tokens[i].argument_child = 11;else if (l = checkPercentage(i)) tokens[i].argument_child = 12;else if (l = checkDimension(i)) tokens[i].argument_child = 13;else if (l = checkNumber(i)) tokens[i].argument_child = 14;else if (l = checkUri(i)) tokens[i].argument_child = 15;else if (l = checkInterpolation(i)) tokens[i].argument_child = 16;else if (l = checkIdent(i)) tokens[i].argument_child = 17;else if (l = checkVhash(i)) tokens[i].argument_child = 18;else if (l = checkCustomProperty(i)) tokens[i].argument_child = 19;else if (l = checkOperator(i)) tokens[i].argument_child = 20;else if (l = checkUnary(i)) tokens[i].argument_child = 21;else if (l = checkParentSelector(i)) tokens[i].argument_child = 22;else if (l = checkImportant(i)) tokens[i].argument_child = 23;else if (l = checkGlobal(i)) tokens[i].argument_child = 24;else if (l = checkDefault(i)) tokens[i].argument_child = 25;else if (l = checkOptional(i)) tokens[i].argument_child = 26;
return l;
}
/**
* @returns {Array} Node that is part of arguments list
*/
function getArgument() {
var childType = tokens[pos].argument_child;
if (childType === 1) return getBrackets();
if (childType === 2) return getParentheses();
if (childType === 3) return getSingleValueDeclaration();
if (childType === 4) return getFunctionsList();
if (childType === 5) return getFunction();
if (childType === 6) return getVariablesList();
if (childType === 7) return getVariable();
if (childType === 8) return getSC();
if (childType === 9) return getDelim();
if (childType === 10) return getDeclDelim();
if (childType === 11) return getString();
if (childType === 12) return getPercentage();
if (childType === 13) return getDimension();
if (childType === 14) return getNumber();
if (childType === 15) return getUri();
if (childType === 16) return getInterpolation();
if (childType === 17) return getIdent();
if (childType === 18) return getVhash();
if (childType === 19) return getCustomProperty();
if (childType === 20) return getOperator();
if (childType === 21) return getUnary();
if (childType === 22) return getParentSelector();
if (childType === 23) return getImportant();
if (childType === 24) return getGlobal();
if (childType === 25) return getDefault();
if (childType === 26) return getOptional();
}
/**
* Check if token is part of an @-word (e.g. `@import`, `@include`)
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkAtkeyword(i) {
var l = void 0;
// Check that token is `@`:
if (i >= tokensLength || tokens[i++].type !== TokenType.CommercialAt) return 0;
return (l = checkIdentOrInterpolation(i)) ? l + 1 : 0;
}
/**
* Get node with @-word
* @return {Node}
*/
function getAtkeyword() {
var type = NodeType.AtkeywordType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `@`.
pos++;
var content = getIdentOrInterpolation();
return newNode(type, content, line, column);
}
/**
* Check if token is a part of an @-rule
* @param {Number} i Token's index number
* @returns {Number} Length of @-rule
*/
function checkAtrule(i) {
var l = void 0;
if (i >= tokensLength) return 0;
// If token already has a record of being part of an @-rule,
// return the @-rule's length:
if (tokens[i].atrule_l !== undefined) return tokens[i].atrule_l;
// If token is part of an @-rule, save the rule's type to token.
// @keyframes:
if (l = checkKeyframesRule(i)) tokens[i].atrule_type = 4;
// @-rule with ruleset:
else if (l = checkAtruler(i)) tokens[i].atrule_type = 1;
// Block @-rule:
else if (l = checkAtruleb(i)) tokens[i].atrule_type = 2;
// Single-line @-rule:
else if (l = checkAtrules(i)) tokens[i].atrule_type = 3;else return 0;
// If token is part of an @-rule, save the rule's length to token:
tokens[i].atrule_l = l;
return l;
}
/**
* Get node with @-rule
* @returns {Array}
*/
function getAtrule() {
var childType = tokens[pos].atrule_type;
if (childType === 1) return getAtruler(); // @-rule with ruleset
if (childType === 2) return getAtruleb(); // Block @-rule
if (childType === 3) return getAtrules(); // Single-line @-rule
if (childType === 4) return getKeyframesRule();
}
/**
* Check if token is part of a block @-rule
* @param {Number} i Token's index number
* @returns {Number} Length of the @-rule
*/
function checkAtruleb(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a block @-rule
* @returns {Array} `['atruleb', ['atkeyword', x], y, ['block', z]]`
*/
function getAtruleb() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an @-rule with ruleset
* @param {Number} i Token's index number
* @returns {Number} Length of the @-rule
*/
function checkAtruler(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
if (l = checkAtrulers(i)) i += l;
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
return i - start;
}
/**
* Get node with an @-rule with ruleset
* @returns {Array} ['atruler', ['atkeyword', x], y, z]
*/
function getAtruler() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets(), getAtrulers());
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkAtrulers(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkSC(i)) i += l;
while (i < tokensLength) {
if (l = checkSC(i)) tokens[i].atrulers_child = 1;else if (l = checkAtrule(i)) tokens[i].atrulers_child = 2;else if (l = checkRuleset(i)) tokens[i].atrulers_child = 3;else break;
i += l;
}
if (i < tokensLength) tokens[i].atrulers_end = 1;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* @returns {Array} `['atrulers', x]`
*/
function getAtrulers() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `{`.
pos++;
content = content.concat(getSC());
while (pos < tokensLength && !tokens[pos].atrulers_end) {
var childType = tokens[pos].atrulers_child;
if (childType === 1) content = content.concat(getSC());else if (childType === 2) content.push(getAtrule());else if (childType === 3) content.push(getRuleset());else break;
}
content = content.concat(getSC());
var end = getLastPosition(content, line, column, 1);
// Skip `}`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkAtrules(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (l = checkTsets(i)) i += l;
return i - start;
}
/**
* @returns {Array} `['atrules', ['atkeyword', x], y]`
*/
function getAtrules() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getTsets());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a block (e.g. `{...}`).
* @param {Number} i Token's index number
* @returns {Number} Length of the block
*/
function checkBlock(i) {
return i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket ? tokens[i].right - i + 1 : 0;
}
/**
* Get node with a block
* @returns {Array} `['block', x]`
*/
function getBlock() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = tokens[pos].right;
var content = [];
// Skip `{`.
pos++;
while (pos < end) {
if (checkBlockdecl(pos)) content = content.concat(getBlockdecl());else throwError(pos);
}
var end_ = getLastPosition(content, line, column, 1);
pos = end + 1;
return newNode(type, content, line, column, end_);
}
/**
* Check if token is part of a declaration (property-value pair)
* @param {Number} i Token's index number
* @returns {Number} Length of the declaration
*/
function checkBlockdecl(i) {
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkBlockdecl1(i)) tokens[i].bd_type = 1;else if (l = checkBlockdecl2(i)) tokens[i].bd_type = 2;else if (l = checkBlockdecl3(i)) tokens[i].bd_type = 3;else if (l = checkBlockdecl4(i)) tokens[i].bd_type = 4;else return 0;
return l;
}
/**
* @returns {Array}
*/
function getBlockdecl() {
var childType = tokens[pos].bd_type;
if (childType === 1) return getBlockdecl1();
if (childType === 2) return getBlockdecl2();
if (childType === 3) return getBlockdecl3();
if (childType === 4) return getBlockdecl4();
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkBlockdecl1(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkConditionalStatement(i)) tokens[i].bd_kind = 1;else if (l = checkInclude(i)) tokens[i].bd_kind = 2;else if (l = checkExtend(i)) tokens[i].bd_kind = 4;else if (l = checkLoop(i)) tokens[i].bd_kind = 3;else if (l = checkAtrule(i)) tokens[i].bd_kind = 6;else if (l = checkRuleset(i)) tokens[i].bd_kind = 7;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 5;else return 0;
i += l;
if (i < tokensLength && (l = checkDeclDelim(i))) i += l;else return 0;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* @returns {Array}
*/
function getBlockdecl1() {
var sc = getSC();
var content = void 0;
switch (tokens[pos].bd_kind) {
case 1:
content = getConditionalStatement();
break;
case 2:
content = getInclude();
break;
case 3:
content = getLoop();
break;
case 4:
content = getExtend();
break;
case 5:
content = getDeclaration();
break;
case 6:
content = getAtrule();
break;
case 7:
content = getRuleset();
break;
}
return sc.concat(content, getSC(), getDeclDelim(), getSC());
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkBlockdecl2(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkConditionalStatement(i)) tokens[i].bd_kind = 1;else if (l = checkInclude(i)) tokens[i].bd_kind = 2;else if (l = checkExtend(i)) tokens[i].bd_kind = 4;else if (l = checkMixin(i)) tokens[i].bd_kind = 8;else if (l = checkLoop(i)) tokens[i].bd_kind = 3;else if (l = checkAtrule(i)) tokens[i].bd_kind = 6;else if (l = checkRuleset(i)) tokens[i].bd_kind = 7;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 5;else return 0;
i += l;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* @returns {Array}
*/
function getBlockdecl2() {
var sc = getSC();
var content = void 0;
switch (tokens[pos].bd_kind) {
case 1:
content = getConditionalStatement();
break;
case 2:
content = getInclude();
break;
case 3:
content = getLoop();
break;
case 4:
content = getExtend();
break;
case 5:
content = getDeclaration();
break;
case 6:
content = getAtrule();
break;
case 7:
content = getRuleset();
break;
case 8:
content = getMixin();
break;
}
return sc.concat(content, getSC());
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkBlockdecl3(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkDeclDelim(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
return i - start;
}
/**
* @returns {Array} `[s0, ['declDelim'], s1]` where `s0` and `s1` are
* are optional whitespaces.
*/
function getBlockdecl3() {
return [].concat(getSC(), getDeclDelim(), getSC());
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkBlockdecl4(i) {
return checkSC(i);
}
/**
* @returns {Array}
*/
function getBlockdecl4() {
return getSC();
}
/**
* Check if token is part of text inside square brackets, e.g. `[1]`
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkBrackets(i) {
if (i >= tokensLength) return 0;
var start = i;
// Skip `[`.
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
if (i < tokens[start].right) {
var l = checkTsets(i);
if (l) i += l;else return 0;
}
// Skip `]`.
i++;
return i - start;
}
/**
* Get node with text inside parentheses or square brackets (e.g. `(1)`)
* @return {Node}
*/
function getBrackets() {
var type = NodeType.BracketsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var right = token.right;
var content = [];
// Skip `[`.
pos++;
if (pos < right) {
content = getTsets();
}
var end = getLastPosition(content, line, column, 1);
// Skip `]`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a class selector (e.g. `.abc`)
* @param {Number} i Token's index number
* @returns {Number} Length of the class selector
*/
function checkClass(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].class_l) return tokens[i].class_l;
// Skip `.`.
if (tokens[i].type === TokenType.FullStop) i++;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
while (i < tokensLength) {
if (l = checkIdentOrInterpolation(i)) {
tokens[start].class_l = l + 1;
i += l;
} else break;
}
tokens[start].classEnd = i;
return i - start;
}
/**
* Get node with a class selector
* @returns {Array} `['class', ['ident', x]]` where x is a class's
* identifier (without `.`, e.g. `abc`).
*/
function getClass() {
var type = NodeType.ClassType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = token.classEnd;
var content = [];
// Skip `.`
pos++;
while (pos < end) {
if (checkIdentOrInterpolation(pos)) {
content = content.concat(getIdentOrInterpolation());
} else break;
}
return newNode(type, content, line, column);
}
function checkCombinator(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkCombinator1(i)) tokens[i].combinatorType = 1;else if (l = checkCombinator2(i)) tokens[i].combinatorType = 2;else if (l = checkCombinator3(i)) tokens[i].combinatorType = 3;else if (l = checkCombinator4(i)) tokens[i].combinatorType = 4;
return l;
}
function getCombinator() {
var type = tokens[pos].combinatorType;
if (type === 1) return getCombinator1();
if (type === 2) return getCombinator2();
if (type === 3) return getCombinator3();
if (type === 4) return getCombinator4();
}
/**
* (1) `>>>`
*
* @param {Number} i
* @return {Number}
*/
function checkCombinator1(i) {
if (i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign) return 3;
return 0;
}
/**
* @return {Node}
*/
function getCombinator1() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '>>>';
// Skip combinator
pos += 3;
return newNode(type, content, line, column);
}
/**
* (1) `||`
* (2) `>>`
*
* @param {Number} i
* @return {Number}
*/
function checkCombinator2(i) {
if (i + 1 >= tokensLength) return 0;
if (tokens[i].type === TokenType.VerticalLine && tokens[i + 1].type === TokenType.VerticalLine) return 2;
if (tokens[i].type === TokenType.GreaterThanSign && tokens[i + 1].type === TokenType.GreaterThanSign) return 2;
return 0;
}
/**
* @return {Node}
*/
function getCombinator2() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '' + token.value + tokens[pos + 1].value;
// Skip combinator
pos += 2;
return newNode(type, content, line, column);
}
/**
* (1) `>`
* (2) `+`
* (3) `~`
*
* @param {Number} i
* @return {Number}
*/
function checkCombinator3(i) {
var type = tokens[i].type;
if (type === TokenType.PlusSign || type === TokenType.GreaterThanSign || type === TokenType.Tilde) return 1;else return 0;
}
/**
* @return {Node}
*/
function getCombinator3() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
// Skip combinator
pos++;
return newNode(type, content, line, column);
}
/**
* (1) `/panda/`
*/
function checkCombinator4(i) {
var start = i;
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
var l = void 0;
if (l = checkIdent(i)) i += l;else return 0;
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getCombinator4() {
var type = NodeType.CombinatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `/`.
pos++;
var ident = getIdent();
// Skip `/`.
pos++;
var content = '/' + ident.content + '/';
return newNode(type, content, line, column);
}
/**
* Check if token is a multiline comment.
* @param {Number} i Token's index number
* @returns {Number} `1` if token is a multiline comment, otherwise `0`
*/
function checkCommentML(i) {
return i < tokensLength && tokens[i].type === TokenType.CommentML ? 1 : 0;
}
/**
* Get node with a multiline comment
* @returns {Array} `['commentML', x]` where `x`
* is the comment's text (without `/*` and `* /`).
*/
function getCommentML() {
var type = NodeType.CommentMLType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos].value.substring(2);
var l = content.length;
if (content.charAt(l - 2) === '*' && content.charAt(l - 1) === '/') content = content.substring(0, l - 2);
var end = getLastPosition(content, line, column, 2);
if (end[0] === line) end[1] += 2;
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a single-line comment.
* @param {Number} i Token's index number
* @returns {Number} `1` if token is a single-line comment, otherwise `0`
*/
function checkCommentSL(i) {
return i < tokensLength && tokens[i].type === TokenType.CommentSL ? 1 : 0;
}
/**
* Get node with a single-line comment.
* @returns {Array} `['commentSL', x]` where `x` is comment's message
* (without `//`)
*/
function getCommentSL() {
var type = NodeType.CommentSLType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos++].value.substring(2);
var end = getLastPosition(content, line, column + 2);
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a condition
* (e.g. `@if ...`, `@else if ...` or `@else ...`).
* @param {Number} i Token's index number
* @returns {Number} Length of the condition
*/
function checkCondition(i) {
var start = i;
var l = void 0;
var _i = void 0;
var s = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (['if', 'else'].indexOf(tokens[start + 1].value) < 0) return 0;
while (i < tokensLength) {
if (l = checkBlock(i)) break;
s = checkSC(i);
_i = i + s;
if (l = _checkCondition(_i)) i += l + s;else break;
}
return i - start;
}
function _checkCondition(i) {
return checkVariable(i) || checkNumber(i) || checkInterpolation(i) || checkIdent(i) || checkOperator(i) || checkCombinator(i) || checkString(i);
}
/**
* Get node with a condition.
* @returns {Array} `['condition', x]`
*/
function getCondition() {
var type = NodeType.ConditionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var s = void 0;
var _pos = void 0;
content.push(getAtkeyword());
while (pos < tokensLength) {
if (checkBlock(pos)) break;
s = checkSC(pos);
_pos = pos + s;
if (!_checkCondition(_pos)) break;
if (s) content = content.concat(getSC());
content.push(_getCondition());
}
return newNode(type, content, line, column);
}
function _getCondition() {
if (checkVariable(pos)) return getVariable();
if (checkNumber(pos)) return getNumber();
if (checkInterpolation(pos)) return getInterpolation();
if (checkIdent(pos)) return getIdent();
if (checkOperator(pos)) return getOperator();
if (checkCombinator(pos)) return getCombinator();
if (checkString(pos)) return getString();
}
/**
* Check if token is part of a conditional statement
* (e.g. `@if ... {} @else if ... {} @else ... {}`).
* @param {Number} i Token's index number
* @returns {Number} Length of the condition
*/
function checkConditionalStatement(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkCondition(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a condition.
* @returns {Array} `['condition', x]`
*/
function getConditionalStatement() {
var type = NodeType.ConditionalStatementType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getCondition(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a declaration (property-value pair)
* @param {Number} i Token's index number
* @returns {Number} Length of the declaration
*/
function checkDeclaration(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkProperty(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkPropertyDelim(i)) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkValue(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a declaration
* @returns {Array} `['declaration', ['property', x], ['propertyDelim'],
* ['value', y]]`
*/
function getDeclaration() {
var type = NodeType.DeclarationType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getProperty(), getSC(), getPropertyDelim(), getSC(), getValue());
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @returns {number} Length of the declaration
*/
function checkSingleValueDeclaration(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkProperty(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkPropertyDelim(i)) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkSingleValue(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a declaration
* @returns {Array} `['declaration', ['property', x], ['propertyDelim'],
* ['value', y]]`
*/
function getSingleValueDeclaration() {
var type = NodeType.DeclarationType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getProperty(), getSC(), getPropertyDelim(), getSC(), getSingleValue());
return newNode(type, content, line, column);
}
/**
* Check if token is a semicolon
* @param {Number} i Token's index number
* @returns {Number} `1` if token is a semicolon, otherwise `0`
*/
function checkDeclDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Semicolon ? 1 : 0;
}
/**
* Get node with a semicolon
* @returns {Array} `['declDelim']`
*/
function getDeclDelim() {
var type = NodeType.DeclDelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ';';
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token if part of `!default` word.
* @param {Number} i Token's index number
* @returns {Number} Length of the `!default` word
*/
function checkDefault(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].value === 'default') {
tokens[start].defaultEnd = i;
return i - start + 1;
} else {
return 0;
}
}
/**
* Get node with a `!default` word
* @returns {Array} `['default', sc]` where `sc` is optional whitespace
*/
function getDefault() {
var type = NodeType.DefaultType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, token.defaultEnd);
pos = token.defaultEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is a comma
* @param {Number} i Token's index number
* @returns {Number} `1` if token is a comma, otherwise `0`
*/
function checkDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Comma ? 1 : 0;
}
/**
* Get node with a comma
* @returns {Array} `['delim']`
*/
function getDelim() {
var type = NodeType.DelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ',';
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is part of a number with dimension unit (e.g. `10px`)
* @param {Number} i Token's index number
* @return {Number}
*/
function checkDimension(i) {
var ln = checkNumber(i);
var li = void 0;
if (i >= tokensLength || !ln || i + ln >= tokensLength) return 0;
return (li = checkUnit(i + ln)) ? ln + li : 0;
}
/**
* Get node of a number with dimension unit
* @return {Node}
*/
function getDimension() {
var type = NodeType.DimensionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getNumber(), getUnit()];
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkExpression(i) {
var start = i;
if (i >= tokensLength || tokens[i++].value !== 'expression' || i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) {
return 0;
}
return tokens[i].right - start + 1;
}
/**
* @returns {Array}
*/
function getExpression() {
var type = NodeType.ExpressionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
pos++;
var content = joinValues(pos + 1, tokens[pos].right - 1);
var end = getLastPosition(content, line, column, 1);
if (end[0] === line) end[1] += 11;
pos = tokens[pos].right + 1;
return newNode(type, content, line, column, end);
}
function checkExtend(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkExtend1(i)) tokens[i].extend_child = 1;else if (l = checkExtend2(i)) tokens[i].extend_child = 2;
return l;
}
function getExtend() {
var childType = tokens[pos].extend_child;
if (childType === 1) return getExtend1();
if (childType === 2) return getExtend2();
}
/**
* Checks if token is part of an extend with `!optional` flag.
* @param {Number} i
*/
function checkExtend1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'extend') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkOptional(i)) i += l;else return 0;
return i - start;
}
function getExtend1() {
var type = NodeType.ExtendType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getSelectorsGroup(), getSC(), getOptional());
return newNode(type, content, line, column);
}
/**
* Checks if token is part of an extend without `!optional` flag.
* @param {Number} i
*/
function checkExtend2(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'extend') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
return i - start;
}
function getExtend2() {
var type = NodeType.ExtendType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getSelectorsGroup());
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkFunction(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i < tokensLength && tokens[i].type === TokenType.LeftParenthesis ? tokens[i].right - start + 1 : 0;
}
/**
* @returns {Array}
*/
function getFunction() {
var type = NodeType.FunctionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getIdentOrInterpolation(), getArguments());
return newNode(type, content, line, column);
}
/**
* Check if token is part of a functions list (e.g. `function(value)...`).
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkFunctionsList(i) {
var d = 0; // Number of dots
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkFunction(i)) i += l;else return 0;
while (i < tokensLength && tokens[i].type === TokenType.FullStop) {
d++;
i++;
}
return d === 3 ? l + d : 0;
}
/**
* Get node with a functions list
* @returns {Array}
*/
function getFunctionsList() {
var type = NodeType.FunctionsListType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getFunction()];
var end = getLastPosition(content, line, column, 3);
// Skip `...`.
pos += 3;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of `!global` word
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkGlobal(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].value === 'global') {
tokens[start].globalEnd = i;
return i - start + 1;
} else {
return 0;
}
}
/**
* Get node with `!global` word
*/
function getGlobal() {
var type = NodeType.GlobalType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, token.globalEnd);
pos = token.globalEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of an identifier
* @param {Number} i Token's index number
* @returns {Number} Length of the identifier
*/
function checkIdent(i) {
var start = i;
if (i >= tokensLength) return 0;
// Check if token is part of a negative number
if (tokens[i].type === TokenType.HyphenMinus && tokens[i + 1].type === TokenType.DecimalNumber) return 0;
if (tokens[i].type === TokenType.HyphenMinus) i++;
if (checkInterpolation(i)) {
tokens[start].ident_last = i - 1;
return i - start;
}
if (tokens[i].type === TokenType.LowLine || tokens[i].type === TokenType.Identifier) i++;else return 0;
for (; i < tokensLength; i++) {
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
}
tokens[start].ident_last = i - 1;
return i - start;
}
/**
* Get node with an identifier
* @returns {Array} `['ident', x]` where `x` is identifier's name
*/
function getIdent() {
var type = NodeType.IdentType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, tokens[pos].ident_last);
pos = tokens[pos].ident_last + 1;
return newNode(type, content, line, column);
}
/**
* @param {number} i Token's index number
* @returns {number} Length of the identifier
*/
function checkPartialIdent(i) {
var start = i;
if (i >= tokensLength) return 0;
for (; i < tokensLength; i++) {
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
}
tokens[start].ident_last = i - 1;
return i - start;
}
function checkIdentOrInterpolation(i) {
var start = i;
var l = void 0;
var prevIsInterpolation = false;
while (i < tokensLength) {
if (l = checkInterpolation(i)) {
tokens[i].ii_type = 1;
i += l;
prevIsInterpolation = true;
} else if (l = checkIdent(i)) {
tokens[i].ii_type = 2;
i += l;
prevIsInterpolation = false;
} else if (prevIsInterpolation && (l = checkPartialIdent(i))) {
tokens[i].ii_type = 3;
i += l;
prevIsInterpolation = false;
} else break;
}
return i - start;
}
function getIdentOrInterpolation() {
var content = [];
while (pos < tokensLength) {
var tokenType = tokens[pos].ii_type;
if (tokenType === 1) {
content.push(getInterpolation());
} else if (tokenType === 2 || tokenType === 3) {
content.push(getIdent());
} else break;
}
return content;
}
/**
* Check if token is part of `!important` word
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkImportant(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].value === 'important') {
tokens[start].importantEnd = i;
return i - start + 1;
} else {
return 0;
}
}
/**
* Get node with `!important` word
* @returns {Array} `['important', sc]` where `sc` is optional whitespace
*/
function getImportant() {
var type = NodeType.ImportantType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, token.importantEnd);
pos = token.importantEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of an included mixin (`@include` or `@extend`
* directive).
* @param {Number} i Token's index number
* @returns {Number} Length of the included mixin
*/
function checkInclude(i) {
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkInclude1(i)) tokens[i].include_type = 1;else if (l = checkInclude2(i)) tokens[i].include_type = 2;else if (l = checkInclude3(i)) tokens[i].include_type = 3;else if (l = checkInclude4(i)) tokens[i].include_type = 4;else if (l = checkInclude5(i)) tokens[i].include_type = 5;
return l;
}
/**
* Get node with included mixin
* @returns {Array} `['include', x]`
*/
function getInclude() {
var type = tokens[pos].include_type;
if (type === 1) return getInclude1();
if (type === 2) return getInclude2();
if (type === 3) return getInclude3();
if (type === 4) return getInclude4();
if (type === 5) return getInclude5();
}
/**
* Get node with included mixin with keyfames selector like
* `@include nani(foo) { 0% {}}`
* @param {Number} i Token's index number
* @returns {Number} Length of the include
*/
function checkInclude1(i) {
var start = i;
var l = void 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'include') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with included mixin with keyfames selector like
* `@include nani(foo) { 0% {}}`
* @returns {Array} `['include', ['atkeyword', x], sc, ['selector', y], sc,
* ['arguments', z], sc, ['block', q], sc` where `x` is `include` or
* `extend`, `y` is mixin's identifier (selector), `z` are arguments
* passed to the mixin, `q` is block passed to the mixin containing a
* ruleset > selector > keyframesSelector, and `sc` are optional
* whitespaces
*/
function getInclude1() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getKeyframesBlocks());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an included mixin like `@include nani(foo) {...}`
* @param {Number} i Token's index number
* @returns {Number} Length of the include
*/
function checkInclude2(i) {
var start = i;
var l = void 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'include') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with included mixin like `@include nani(foo) {...}`
* @returns {Array} `['include', ['atkeyword', x], sc, ['selector', y], sc,
* ['arguments', z], sc, ['block', q], sc` where `x` is `include` or
* `extend`, `y` is mixin's identifier (selector), `z` are arguments
* passed to the mixin, `q` is block passed to the mixin and `sc`
* are optional whitespaces
*/
function getInclude2() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an included mixin like `@include nani(foo)`
* @param {Number} i Token's index number
* @returns {Number} Length of the include
*/
function checkInclude3(i) {
var start = i;
var l = void 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'include') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with included mixin like `@include nani(foo)`
* @returns {Array} `['include', ['atkeyword', x], sc, ['selector', y], sc,
* ['arguments', z], sc]` where `x` is `include` or `extend`, `y` is
* mixin's identifier (selector), `z` are arguments passed to the
* mixin and `sc` are optional whitespaces
*/
function getInclude3() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an included mixin with a content block passed
* as an argument (e.g. `@include nani {...}`)
* @param {Number} i Token's index number
* @returns {Number} Length of the mixin
*/
function checkInclude4(i) {
var start = i;
var l = void 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'include') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with an included mixin with a content block passed
* as an argument (e.g. `@include nani {...}`)
* @returns {Array} `['include', x]`
*/
function getInclude4() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkInclude5(i) {
var start = i;
var l = void 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (tokens[start + 1].value !== 'include') return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
/**
* @returns {Array} `['include', x]`
*/
function getInclude5() {
var type = NodeType.IncludeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation());
return newNode(type, content, line, column);
}
/**
* Check if token is part of an interpolated variable (e.g. `#{$nani}`).
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkInterpolation(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.NumberSign || !tokens[i + 1] || tokens[i + 1].type !== TokenType.LeftCurlyBracket) return 0;
i += 2;
while (tokens[i].type !== TokenType.RightCurlyBracket) {
if (l = checkArgument(i)) i += l;else return 0;
}
return tokens[i].type === TokenType.RightCurlyBracket ? i - start + 1 : 0;
}
/**
* Get node with an interpolated variable
* @returns {Array} `['interpolation', x]`
*/
function getInterpolation() {
var type = NodeType.InterpolationType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `#{`:
pos += 2;
while (pos < tokensLength && tokens[pos].type !== TokenType.RightCurlyBracket) {
var body = getArgument();
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
}
var end = getLastPosition(content, line, column, 1);
// Skip `}`:
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check a single keyframe block - `5% {}`
* @param {Number} i
* @returns {Number}
*/
function checkKeyframesBlock(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkKeyframesSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get a single keyframe block - `5% {}`
* @returns {Node}
*/
function getKeyframesBlock() {
var type = NodeType.RulesetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getKeyframesSelectorsGroup(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check all keyframe blocks - `5% {} 100% {}`
* @param {Number} i
* @returns {Number}
*/
function checkKeyframesBlocks(i) {
var start = i;
var l = void 0;
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlock(i)) i += l;
while (tokens[i].type !== TokenType.RightCurlyBracket) {
if (l = checkSC(i)) i += l;else if (l = checkKeyframesBlock(i)) i += l;else if (l = checkAtrule(i)) {
i += l;
if (l = checkSC(i)) i += l;
if (l = checkDeclDelim(i)) i += l;
} else break;
}
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
return i - start;
}
/**
* Get all keyframe blocks - `5% {} 100% {}`
* @returns {Node}
*/
function getKeyframesBlocks() {
var type = NodeType.BlockType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var keyframesBlocksEnd = token.right;
var content = [];
// Skip `{`.
pos++;
while (pos < keyframesBlocksEnd) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkKeyframesBlock(pos)) content.push(getKeyframesBlock());else if (checkAtrule(pos)) {
content.push(getAtrule()); // @content
if (checkSC(pos)) content = content.concat(getSC());
if (checkDeclDelim(pos)) content.push(getDeclDelim());
} else break;
}
var end = getLastPosition(content, line, column, 1);
// Skip `}`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a @keyframes rule.
* @param {Number} i Token's index number
* @return {Number} Length of the @keyframes rule
*/
function checkKeyframesRule(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
var atruleName = joinValues2(i - l, l);
if (atruleName.toLowerCase().indexOf('keyframes') === -1) return 0;
if (l = checkSC(i)) i += l;else return 0;
if (l = checkIdentOrInterpolation(i) || checkPseudoc(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
return i - start;
}
/**
* @return {Node}
*/
function getKeyframesRule() {
var type = NodeType.AtruleType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC());
if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());else if (checkPseudoc(pos)) {
content = content.concat(getPseudoc());
}
content = content.concat(getSC(), getKeyframesBlocks());
return newNode(type, content, line, column);
}
/**
* Check a single keyframe selector - `5%`, `from` etc
* @param {Number} i
* @returns {Number}
*/
function checkKeyframesSelector(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) {
// Valid selectors are only `from` and `to`.
var selector = joinValues2(i, l);
if (selector !== 'from' && selector !== 'to') return 0;
i += l;
tokens[start].keyframesSelectorType = 1;
} else if (l = checkPercentage(i)) {
i += l;
tokens[start].keyframesSelectorType = 2;
} else if (l = checkInterpolation(i)) {
i += l;
tokens[start].keyframesSelectorType = 3;
} else {
return 0;
}
return i - start;
}
/**
* Get a single keyframe selector
* @returns {Node}
*/
function getKeyframesSelector() {
var keyframesSelectorType = NodeType.KeyframesSelectorType;
var selectorType = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (token.keyframesSelectorType === 1) {
content.push(getIdent());
} else if (token.keyframesSelectorType === 2) {
content.push(getPercentage());
} else if (token.keyframesSelectorType === 3) {
content.push(getInterpolation());
}
var keyframesSelector = newNode(keyframesSelectorType, content, line, column);
return newNode(selectorType, [keyframesSelector], line, column);
}
/**
* Check the keyframe's selector groups
* @param {Number} i
* @returns {Number}
*/
function checkKeyframesSelectorsGroup(i) {
var start = i;
var l = void 0;
if (l = checkKeyframesSelector(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
var comma = checkDelim(i + spaceBefore);
if (!comma) break;
var spaceAfter = checkSC(i + spaceBefore + comma);
if (l = checkKeyframesSelector(i + spaceBefore + comma + spaceAfter)) {
i += spaceBefore + comma + spaceAfter + l;
} else break;
}
tokens[start].selectorsGroupEnd = i;
return i - start;
}
/**
* Get the keyframe's selector groups
* @returns {Array} An array of keyframe selectors
*/
function getKeyframesSelectorsGroup() {
var selectorsGroup = [];
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
selectorsGroup.push(getKeyframesSelector());
while (pos < selectorsGroupEnd) {
selectorsGroup = selectorsGroup.concat(getSC(), getDelim(), getSC(), getKeyframesSelector());
}
return selectorsGroup;
}
/**
* Check if token is part of a loop.
* @param {Number} i Token's index number
* @returns {Number} Length of the loop
*/
function checkLoop(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkAtkeyword(i)) i += l;else return 0;
if (['for', 'each', 'while'].indexOf(tokens[start + 1].value) < 0) return 0;
while (i < tokensLength) {
if (l = checkBlock(i)) {
i += l;
break;
} else if (l = checkVariable(i) || checkNumber(i) || checkInterpolation(i) || checkIdent(i) || checkSC(i) || checkOperator(i) || checkCombinator(i) || checkString(i)) i += l;else return 0;
}
return i - start;
}
/**
* Get node with a loop.
* @returns {Array} `['loop', x]`
*/
function getLoop() {
var type = NodeType.LoopType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(getAtkeyword());
while (pos < tokensLength) {
if (checkBlock(pos)) {
content.push(getBlock());
break;
} else if (checkVariable(pos)) content.push(getVariable());else if (checkNumber(pos)) content.push(getNumber());else if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkIdent(pos)) content.push(getIdent());else if (checkOperator(pos)) content.push(getOperator());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkSC(pos)) content = content.concat(getSC());else if (checkString(pos)) content.push(getString());
}
return newNode(type, content, line, column);
}
/**
* Check if token is part of a mixin
* @param {Number} i Token's index number
* @returns {Number} Length of the mixin
*/
function checkMixin(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if ((l = checkAtkeyword(i)) && tokens[i + 1].value === 'mixin') i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkArguments(i)) i += l;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a mixin
* @returns {Array} `['mixin', x]`
*/
function getMixin() {
var type = NodeType.MixinType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getAtkeyword(), getSC());
if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());
content = content.concat(getSC());
if (checkArguments(pos)) content.push(getArguments());
content = content.concat(getSC());
if (checkBlock(pos)) content.push(getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is a namespace sign (`|`)
* @param {Number} i Token's index number
* @returns {Number} `1` if token is `|`, `0` if not
*/
function checkNamespace(i) {
return i < tokensLength && tokens[i].type === TokenType.VerticalLine ? 1 : 0;
}
/**
* Get node with a namespace sign
* @returns {Array} `['namespace']`
*/
function getNamespace() {
var type = NodeType.NamespaceType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '|';
pos++;
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkNmName2(i) {
if (tokens[i].type === TokenType.Identifier) return 1;else if (tokens[i].type !== TokenType.DecimalNumber) return 0;
i++;
return i < tokensLength && tokens[i].type === TokenType.Identifier ? 2 : 1;
}
/**
* @returns {String}
*/
function getNmName2() {
var s = tokens[pos].value;
if (tokens[pos++].type === TokenType.DecimalNumber && pos < tokensLength && tokens[pos].type === TokenType.Identifier) s += tokens[pos++].value;
return s;
}
/**
* Check if token is part of a number
* @param {Number} i Token's index number
* @returns {Number} Length of number
*/
function checkNumber(i) {
if (i >= tokensLength) return 0;
if (tokens[i].number_l) return tokens[i].number_l;
// `10`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && (!tokens[i + 1] || tokens[i + 1] && tokens[i + 1].type !== TokenType.FullStop)) {
tokens[i].number_l = 1;
return 1;
}
// `10.`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && (!tokens[i + 2] || tokens[i + 2].type !== TokenType.DecimalNumber)) {
tokens[i].number_l = 2;
return 2;
}
// `.10`:
if (i < tokensLength && tokens[i].type === TokenType.FullStop && tokens[i + 1].type === TokenType.DecimalNumber) {
tokens[i].number_l = 2;
return 2;
}
// `10.10`:
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && tokens[i + 2] && tokens[i + 2].type === TokenType.DecimalNumber) {
tokens[i].number_l = 3;
return 3;
}
return 0;
}
/**
* Get node with number
* @returns {Array} `['number', x]` where `x` is a number converted
* to string.
*/
function getNumber() {
var type = NodeType.NumberType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var l = tokens[pos].number_l;
var content = '';
for (var j = 0; j < l; j++) {
content += tokens[pos + j].value;
}
pos += l;
return newNode(type, content, line, column);
}
/**
* Check if token is an operator (`/`, `%`, `,`, `:` or `=`).
* @param {Number} i Token's index number
* @returns {Number} `1` if token is an operator, otherwise `0`
*/
function checkOperator(i) {
if (i >= tokensLength) return 0;
switch (tokens[i].type) {
case TokenType.Solidus:
case TokenType.PercentSign:
case TokenType.Comma:
case TokenType.Colon:
case TokenType.EqualsSign:
case TokenType.EqualitySign:
case TokenType.InequalitySign:
case TokenType.LessThanSign:
case TokenType.GreaterThanSign:
case TokenType.Asterisk:
return 1;
}
return 0;
}
/**
* Get node with an operator
* @returns {Array} `['operator', x]` where `x` is an operator converted
* to string.
*/
function getOperator() {
var type = NodeType.OperatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is part of `!optional` word
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkOptional(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].value === 'optional') {
tokens[start].optionalEnd = i;
return i - start + 1;
} else {
return 0;
}
}
/**
* Get node with `!optional` word
*/
function getOptional() {
var type = NodeType.OptionalType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, token.optionalEnd);
pos = token.optionalEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of text inside parentheses, e.g. `(1)`
* @param {Number} i Token's index number
* @return {Number}
*/
function checkParentheses(i) {
if (i >= tokensLength) return 0;
var start = i;
var right = tokens[i].right;
var l = void 0;
// Skip `(`.
if (tokens[i].type === TokenType.LeftParenthesis) i++;else return 0;
if (i < right) {
if (l = checkTsets(i)) i += l;else return 0;
}
// Skip `)`.
i++;
return i - start;
}
/**
* Get node with text inside parentheses, e.g. `(1)`
* @return {Node}
*/
function getParentheses() {
var type = NodeType.ParenthesesType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var right = token.right;
var content = [];
// Skip `(`.
pos++;
if (pos < right) {
content = getTsets();
}
var end = getLastPosition(content, line, column, 1);
// Skip `)`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is a parent selector, e.g. `&`
* @param {number} i Token's index number
* @return {number}
*/
function checkParentSelector(i) {
return i < tokensLength && tokens[i].type === TokenType.Ampersand ? 1 : 0;
}
/**
* Get node with a parent selector
* @return {Node}
*/
function getParentSelector() {
var type = NodeType.ParentSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '&';
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a parent selector extension, e.g. `&--foo-bar`
* @param {number} i Token's index number
* @returns {number} Length of the parent selector extension
*/
function checkParentSelectorExtension(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
while (i < tokensLength) {
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else break;
}
return i - start;
}
/**
* Get parent selector extension node
* @return {Node}
*/
function getParentSelectorExtension() {
var type = NodeType.ParentSelectorExtensionType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
while (pos < tokensLength) {
if (checkIdentOrInterpolation(pos)) {
content = content.concat(getIdentOrInterpolation());
} else if (checkPartialIdent(pos)) {
content.push(getIdent());
} else break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is a parent selector with an extension or not
* @param {number} i Token's index number
* @return {number} Length of the parent selector and extension if applicable
*/
function checkParentSelectorWithExtension(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkParentSelector(i)) i += l;else return 0;
if (l = checkParentSelectorExtension(i)) i += l;
return i - start;
}
/**
* Get parent selector node and extension node if applicable
* @return {Array}
*/
function getParentSelectorWithExtension() {
var content = [getParentSelector()];
if (checkParentSelectorExtension(pos)) content.push(getParentSelectorExtension());
return content;
}
/**
* Check if token is part of a number or an interpolation with a percent sign
* (e.g. `10%`).
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkPercentage(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkNumberOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
// Skip `%`.
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
return i - start;
}
/**
* Get a percentage node that contains either a number or an interpolation
* @returns {Object} The percentage node
*/
function getPercentage() {
var type = NodeType.PercentageType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = getNumberOrInterpolation();
var end = getLastPosition(content, line, column, 1);
// Skip `%`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is a number or an interpolation
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkNumberOrInterpolation(i) {
var start = i;
var l = void 0;
while (i < tokensLength) {
if (l = checkInterpolation(i) || checkNumber(i)) i += l;else break;
}
return i - start;
}
/**
* Get a number and/or interpolation node
* @returns {Array} An array containing a single or multiple nodes
*/
function getNumberOrInterpolation() {
var content = [];
while (pos < tokensLength) {
if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkNumber(pos)) content.push(getNumber());else break;
}
return content;
}
/**
* Check if token is part of a placeholder selector (e.g. `%abc`).
* @param {Number} i Token's index number
* @returns {Number} Length of the selector
*/
function checkPlaceholder(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[start].placeholder_l) return tokens[start].placeholder_l;
// Skip `%`.
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
if (l = checkIdentOrInterpolation(i)) {
i += l;
tokens[start].placeholder_l = i - start;
} else return 0;
return i - start;
}
/**
* Get node with a placeholder selector
* @returns {Array} `['placeholder', ['ident', x]]` where x is a placeholder's
* identifier (without `%`, e.g. `abc`).
*/
function getPlaceholder() {
var type = NodeType.PlaceholderType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `%`.
pos++;
content = content.concat(getIdentOrInterpolation());
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkProgid(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (joinValues2(i, 6) === 'progid:DXImageTransform.Microsoft.') i += 6;else return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type === TokenType.LeftParenthesis) {
tokens[start].progid_end = tokens[i].right;
i = tokens[i].right + 1;
} else return 0;
return i - start;
}
/**
* @returns {Array}
*/
function getProgid() {
var type = NodeType.ProgidType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var progid_end = token.progid_end;
var content = joinValues(pos, progid_end);
pos = progid_end + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is part of a property
* @param {Number} i Token's index number
* @return {Number} Length of the property
*/
function checkProperty(i) {
var start = i;
var l = void 0;
if (l = checkProperty1(i)) tokens[start].propertyType = 1;else if (l = checkProperty2(i)) tokens[start].propertyType = 2;else if (l = checkProperty3(i)) tokens[start].propertyType = 3;
return l;
}
/**
* Get node with a property
* @return {Node}
*/
function getProperty() {
var type = tokens[pos].propertyType;
if (type === 1) return getProperty1();
if (type === 2) return getProperty2();
if (type === 3) return getProperty3();
}
/**
* Check if token is part of a property
* (1) `foo`
* (2) `#{$foo}`
* @param {Number} i Token's index number
* @returns {Number} Length of the property
*/
function checkProperty1(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a property
* @returns {Array}
*/
function getProperty1() {
var type = NodeType.PropertyType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = getIdentOrInterpolation();
return newNode(type, content, line, column);
}
/**
* Check if token is part of a custom property
* (1) `--foo-bar`
* @param {Number} i Token's index number
* @return {Number} Length of the property
*/
function checkProperty2(i) {
return checkCustomProperty(i);
}
/**
* Get node with a custom property
* @return {Node}
*/
function getProperty2() {
return getCustomProperty();
}
/**
* Check if token is part of a property
* (1) `$foo`
* @param {Number} i Token's index number
* @returns {Number} Length of the property
*/
function checkProperty3(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkVariable(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a property
* @returns {Array} `['property', x]`
*/
function getProperty3() {
var type = NodeType.PropertyType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getVariable()];
return newNode(type, content, line, column);
}
/**
* Check if token is part of a custom property
* @param {Number} i Token's index number
* @return {Number} Length of the property
*/
function checkCustomProperty(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.HyphenMinus || tokens[i + 1] && tokens[i + 1].type !== TokenType.HyphenMinus) return 0;
// Skip `--`
i += 2;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a custom property
* @return {Node}
*/
function getCustomProperty() {
var type = NodeType.CustomPropertyType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `--`
pos += 2;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* Check if token is a colon
* @param {Number} i Token's index number
* @returns {Number} `1` if token is a colon, otherwise `0`
*/
function checkPropertyDelim(i) {
return i < tokensLength && tokens[i].type === TokenType.Colon ? 1 : 0;
}
/**
* Get node with a colon
* @returns {Array} `['propertyDelim']`
*/
function getPropertyDelim() {
var type = NodeType.PropertyDelimType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = ':';
// Skip `:`.
pos++;
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkPseudo(i) {
return checkPseudoe(i) || checkPseudoc(i);
}
/**
* @returns {Array}
*/
function getPseudo() {
if (checkPseudoe(pos)) return getPseudoe();
if (checkPseudoc(pos)) return getPseudoc();
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkPseudoe(i) {
var l = void 0;
// Check `::`
if (i >= tokensLength || tokens[i].type !== TokenType.Colon || i >= tokensLength || tokens[i + 1].type !== TokenType.Colon) return 0;
if (l = checkPseudoElement1(i)) tokens[i].pseudoElementType = 1;else if (l = checkPseudoElement2(i)) tokens[i].pseudoElementType = 2;else return 0;
return l;
}
/**
* @returns {Node}
*/
function getPseudoe() {
var childType = tokens[pos].pseudoElementType;
if (childType === 1) return getPseudoElement1();
if (childType === 2) return getPseudoElement2();
}
/**
* (1) `::slotted(selector)`
* (2) `::slotted(selector, selector)`
*/
function checkPseudoElement1(i) {
var start = i;
var l = void 0;
// Skip `::`.
i += 2;
if (i >= tokensLength) return 0;
if (l = checkIdent(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
/**
* (1) `::slotted(selector)`
* (2) `::slotted(selector, selector)`
*/
function getPseudoElement1() {
var type = NodeType.PseudoeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `::`.
pos += 2;
content.push(getIdent());
{
var _type = NodeType.ArgumentsType;
var _token = tokens[pos];
var _line = _token.ln;
var _column = _token.col;
// Skip `(`.
pos++;
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
var end = getLastPosition(selectorContent, _line, _column, 1);
var args = newNode(_type, selectorContent, _line, _column, end);
content.push(args);
// Skip `)`.
pos++;
}
return newNode(type, content, line, column);
}
function checkPseudoElement2(i) {
var start = i;
var l = void 0;
// Skip `::`.
i += 2;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
/**
* @returns {Node}
*/
function getPseudoElement2() {
var type = NodeType.PseudoeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `::`.
pos += 2;
var content = getIdentOrInterpolation();
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkPseudoc(i) {
var l = void 0;
if (i >= tokensLength || tokens[i].type !== TokenType.Colon) return 0;
if (l = checkPseudoClass3(i)) tokens[i].pseudoClassType = 3;else if (l = checkPseudoClass4(i)) tokens[i].pseudoClassType = 4;else if (l = checkPseudoClass5(i)) tokens[i].pseudoClassType = 5;else if (l = checkPseudoClass1(i)) tokens[i].pseudoClassType = 1;else if (l = checkPseudoClass2(i)) tokens[i].pseudoClassType = 2;else if (l = checkPseudoClass6(i)) tokens[i].pseudoClassType = 6;else return 0;
return l;
}
/**
* @returns {Array}
*/
function getPseudoc() {
var childType = tokens[pos].pseudoClassType;
if (childType === 1) return getPseudoClass1();
if (childType === 2) return getPseudoClass2();
if (childType === 3) return getPseudoClass3();
if (childType === 4) return getPseudoClass4();
if (childType === 5) return getPseudoClass5();
if (childType === 6) return getPseudoClass6();
}
/**
* (-) `:not(panda)`
*/
function checkPseudoClass1(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
/**
* (-) `:not(panda)`
*/
function getPseudoClass1() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content = content.concat(getIdentOrInterpolation());
{
var _type2 = NodeType.ArgumentsType;
var _token2 = tokens[pos];
var _line2 = _token2.ln;
var _column2 = _token2.col;
// Skip `(`.
pos++;
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
var end = getLastPosition(selectorContent, _line2, _column2, 1);
var args = newNode(_type2, selectorContent, _line2, _column2, end);
content.push(args);
// Skip `)`.
pos++;
}
return newNode(type, content, line, column);
}
/**
* (1) `:nth-child(odd)`
* (2) `:nth-child(even)`
* (3) `:lang(de-DE)`
*/
function checkPseudoClass2(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass2() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content = content.concat(getIdentOrInterpolation());
var l = tokens[pos].ln;
var c = tokens[pos].col;
// Skip `(`.
pos++;
var value = [].concat(getSC(), getIdentOrInterpolation(), getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(-3n + 2)`
*/
function checkPseudoClass3(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (l = checkNumberOrInterpolation(i)) i += l;
if (i >= tokensLength) return 0;
if (tokens[i].value === 'n') i++;
if (l = checkSC(i)) i += l;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.PlusSign || tokens[i].type === TokenType.HyphenMinus) i++;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkNumberOrInterpolation(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass3() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content = content.concat(getIdentOrInterpolation());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkNumberOrInterpolation(pos)) value = value.concat(getNumberOrInterpolation());
{
var _token3 = tokens[pos];
if (_token3.value === 'n') {
var _l = _token3.ln;
var _c = _token3.col;
var _content = _token3.value;
var ident = newNode(NodeType.IdentType, _content, _l, _c);
value.push(ident);
pos++;
}
}
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
value = value.concat(getSC());
if (checkNumberOrInterpolation(pos)) value = value.concat(getNumberOrInterpolation());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(-3n)`
*/
function checkPseudoClass4(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (l = checkInterpolation(i)) i += l;
if (tokens[i].type === TokenType.DecimalNumber) i++;
if (tokens[i].value === 'n') i++;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass4() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content = content.concat(getIdentOrInterpolation());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkInterpolation(pos)) value.push(getInterpolation());
if (checkNumber(pos)) value.push(getNumber());
if (checkIdent(pos)) value.push(getIdent());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:nth-child(+8)`
*/
function checkPseudoClass5(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
var right = tokens[i].right;
// Skip `(`.
i++;
if (l = checkSC(i)) i += l;
if (l = checkUnary(i)) i += l;
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
if (l = checkSC(i)) i += l;
if (i !== right) return 0;
// Skip `)`.
i++;
return i - start;
}
function getPseudoClass5() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `:`.
pos++;
content = content.concat(getIdentOrInterpolation());
var l = tokens[pos].ln;
var c = tokens[pos].col;
var value = [];
// Skip `(`.
pos++;
value = value.concat(getSC());
if (checkUnary(pos)) value.push(getUnary());
if (checkNumber(pos)) value.push(getNumber());
value = value.concat(getSC());
var end = getLastPosition(value, l, c, 1);
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
content.push(args);
// Skip `)`.
pos++;
return newNode(type, content, line, column);
}
/**
* (-) `:checked`
*/
function checkPseudoClass6(i) {
var start = i;
var l = void 0;
// Skip `:`.
i++;
if (i >= tokensLength) return 0;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
function getPseudoClass6() {
var type = NodeType.PseudocType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `:`.
pos++;
var content = getIdentOrInterpolation();
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkRuleset(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkSelectorsGroup(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkBlock(i)) i += l;else return 0;
return i - start;
}
function getRuleset() {
var type = NodeType.RulesetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [].concat(getSelectorsGroup(), getSC(), getBlock());
return newNode(type, content, line, column);
}
/**
* Check if token is marked as a space (if it's a space or a tab
* or a line break).
* @param {Number} i
* @returns {Number} Number of spaces in a row starting with the given token.
*/
function checkS(i) {
return i < tokensLength && tokens[i].ws ? tokens[i].ws_last - i + 1 : 0;
}
/**
* Get node with spaces
* @returns {Array} `['s', x]` where `x` is a string containing spaces
*/
function getS() {
var type = NodeType.SType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = joinValues(pos, tokens[pos].ws_last);
pos = tokens[pos].ws_last + 1;
return newNode(type, content, line, column);
}
/**
* Check if token is a space or a comment.
* @param {Number} i Token's index number
* @returns {Number} Number of similar (space or comment) tokens
* in a row starting with the given token.
*/
function checkSC(i) {
if (i >= tokensLength) return 0;
var l = void 0;
var lsc = 0;
while (i < tokensLength) {
if (l = checkS(i)) tokens[i].sc_child = 1;else if (l = checkCommentML(i)) tokens[i].sc_child = 2;else if (l = checkCommentSL(i)) tokens[i].sc_child = 3;else break;
i += l;
lsc += l;
}
return lsc || 0;
}
/**
* Get node with spaces and comments
* @returns {Array} Array containing nodes with spaces (if there are any)
* and nodes with comments (if there are any):
* `[['s', x]*, ['comment', y]*]` where `x` is a string of spaces
* and `y` is a comment's text (without `/*` and `* /`).
*/
function getSC() {
var sc = [];
if (pos >= tokensLength) return sc;
while (pos < tokensLength) {
var childType = tokens[pos].sc_child;
if (childType === 1) sc.push(getS());else if (childType === 2) sc.push(getCommentML());else if (childType === 3) sc.push(getCommentSL());else break;
}
return sc;
}
/**
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside a simple
* selector
* @param {number} i Token's index number
* @return {number}
*/
function checkShash(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else return 0;
while (i < tokensLength) {
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else break;
}
tokens[start].shashEnd = i;
return i - start;
}
/**
* Get node with a hexadecimal number (e.g. `#fff`) inside a simple selector
* @returns {Node}
*/
function getShash() {
var type = NodeType.ShashType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = token.shashEnd;
var content = [];
// Skip `#`.
pos++;
while (pos < end) {
if (checkIdentOrInterpolation(pos)) {
content = content.concat(getIdentOrInterpolation());
} else if (checkPartialIdent(pos)) {
content.push(getIdent());
} else break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is part of a string (text wrapped in quotes)
* @param {Number} i Token's index number
* @returns {Number} `1` if token is part of a string, `0` if not
*/
function checkString(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.StringSQ || tokens[i].type === TokenType.StringDQ) {
return 1;
}
return 0;
}
/**
* Get string's node
* @returns {Array} `['string', x]` where `x` is a string (including
* quotes).
*/
function getString() {
var type = NodeType.StringType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Validate stylesheet: it should consist of any number (0 or more) of
* rulesets (sets of rules with selectors), @-rules, whitespaces or
* comments.
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkStylesheet(i) {
var start = i;
var l = void 0;
while (i < tokensLength) {
if (l = checkSC(i)) tokens[i].stylesheet_child = 1;else if (l = checkRuleset(i)) tokens[i].stylesheet_child = 2;else if (l = checkInclude(i)) tokens[i].stylesheet_child = 3;else if (l = checkExtend(i)) tokens[i].stylesheet_child = 4;else if (l = checkMixin(i)) tokens[i].stylesheet_child = 5;else if (l = checkLoop(i)) tokens[i].stylesheet_child = 6;else if (l = checkConditionalStatement(i)) tokens[i].stylesheet_child = 7;else if (l = checkAtrule(i)) tokens[i].stylesheet_child = 8;else if (l = checkDeclaration(i)) tokens[i].stylesheet_child = 9;else if (l = checkDeclDelim(i)) tokens[i].stylesheet_child = 10;else throwError(i);
i += l;
}
return i - start;
}
/**
* @returns {Array} `['stylesheet', x]` where `x` is all stylesheet's
* nodes.
*/
function getStylesheet() {
var type = NodeType.StylesheetType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
while (pos < tokensLength) {
var childType = tokens[pos].stylesheet_child;
if (childType === 1) content = content.concat(getSC());
if (childType === 2) content.push(getRuleset());
if (childType === 3) content.push(getInclude());
if (childType === 4) content.push(getExtend());
if (childType === 5) content.push(getMixin());
if (childType === 6) content.push(getLoop());
if (childType === 7) content.push(getConditionalStatement());
if (childType === 8) content.push(getAtrule());
if (childType === 9) content.push(getDeclaration());
if (childType === 10) content.push(getDeclDelim());
}
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkTset(i) {
var l = void 0;
if (l = checkVhash(i)) tokens[i].tset_child = 1;else if (l = checkOperator(i)) tokens[i].tset_child = 2;else if (l = checkAny(i)) tokens[i].tset_child = 3;else if (l = checkSC(i)) tokens[i].tset_child = 4;
return l;
}
/**
* @returns {Array}
*/
function getTset() {
var childType = tokens[pos].tset_child;
if (childType === 1) return getVhash();
if (childType === 2) return getOperator();
if (childType === 3) return getAny();
if (childType === 4) return getSC();
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkTsets(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
while (l = checkTset(i)) {
i += l;
}
tokens[start].tsets_end = i;
return i - start;
}
/**
* @returns {Array}
*/
function getTsets() {
var content = [];
var t = void 0;
if (pos >= tokensLength) return content;
var end = tokens[pos].tsets_end;
while (pos < end) {
t = getTset();
if (typeof t.content === 'string') content.push(t);else content = content.concat(t);
}
return content;
}
/**
* Check if token is an unary (arithmetical) sign (`+` or `-`)
* @param {Number} i Token's index number
* @returns {Number} `1` if token is an unary sign, `0` if not
*/
function checkUnary(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type === TokenType.HyphenMinus || tokens[i].type === TokenType.PlusSign) {
return 1;
}
return 0;
}
/**
* Get node with an unary (arithmetical) sign (`+` or `-`)
* @returns {Array} `['unary', x]` where `x` is an unary sign
* converted to string.
*/
function getUnary() {
var type = NodeType.OperatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a unicode range (single or multiple <urange> nodes)
* @param {number} i Token's index
* @return {number} Unicode range node's length
*/
function checkUnicodeRange(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkUrange(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
var comma = checkDelim(i + spaceBefore);
if (!comma) break;
var spaceAfter = checkSC(i + spaceBefore + comma);
if (l = checkUrange(i + spaceBefore + comma + spaceAfter)) {
i += spaceBefore + comma + spaceAfter + l;
} else break;
}
return i - start;
}
/**
* Get a unicode range node
* @return {Node}
*/
function getUnicodeRange() {
var type = NodeType.UnicodeRangeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
while (pos < tokensLength) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkDelim(pos)) content.push(getDelim());else if (checkUrange(pos)) content.push(getUrange());else break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is unit
* @param {Number} i Token's index number
* @return {Number}
*/
function checkUnit(i) {
var units = ['em', 'ex', 'ch', 'rem', 'vh', 'vw', 'vmin', 'vmax', 'px', 'mm', 'q', 'cm', 'in', 'pt', 'pc', 'deg', 'grad', 'rad', 'turn', 's', 'ms', 'Hz', 'kHz', 'dpi', 'dpcm', 'dppx'];
return units.indexOf(tokens[i].value) !== -1 ? 1 : 0;
}
/**
* Get unit node of type ident
* @return {Node} An ident node containing the unit value
*/
function getUnit() {
var type = NodeType.IdentType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = token.value;
pos++;
return newNode(type, content, line, column);
}
/**
* Check if token is a u-range (part of a unicode-range)
* (1) `U+416`
* (2) `U+400-4ff`
* (3) `U+4??`
* @param {number} i Token's index
* @return {number} Urange node's length
*/
function checkUrange(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
// Check for unicode prefix (u+ or U+)
if (tokens[i].value === 'U' || tokens[i].value === 'u') i += 1;else return 0;
if (i >= tokensLength) return 0;
if (tokens[i].value === '+') i += 1;else return 0;
while (i < tokensLength) {
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = _checkUnicodeWildcard(i)) i += l;else break;
}
tokens[start].urangeEnd = i - 1;
return i - start;
}
/**
* Get a u-range node (part of a unicode-range)
* @return {Node}
*/
function getUrange() {
var startPos = pos;
var type = NodeType.UrangeType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content = joinValues(startPos, tokens[startPos].urangeEnd);
pos = tokens[startPos].urangeEnd + 1;
return newNode(type, content, line, column);
}
/**
* Check for unicode wildcard characters `?`
* @param {number} i Token's index
* @return {number} Wildcard length
*/
function _checkUnicodeWildcard(i) {
var start = i;
if (i >= tokensLength) return 0;
while (i < tokensLength) {
if (tokens[i].type === TokenType.QuestionMark) i += 1;else break;
}
return i - start;
}
/**
* Check if token is part of URI, e.g. `url('/css/styles.css')`
* @param {number} i Token's index number
* @returns {number} Length of URI
*/
function checkUri(i) {
var start = i;
var l = void 0;
if (i >= tokensLength || tokens[i].value !== 'url') return 0;
// Skip `url`.
i++;
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
// Store the opening parenthesis token as we will reference it's `right`
// property to determine when the parentheses close
var leftParenthesis = tokens[i];
// Skip `(`.
i++;
// Determine the type of URI
while (i < leftParenthesis.right) {
if (l = checkUri1(i)) {
i += l;
tokens[start].uriType = 1; // Raw based URI (without quotes)
} else if (l = checkUri2(i)) {
i += l;
tokens[start].uriType = 2; // Non-raw based URI (with quotes)
} else return 0;
}
// Skip `)`.
i++;
return i - start;
}
/**
* Get specific type of URI node
* @return {Node} Specific type of URI node
*/
function getUri() {
var startPos = pos;
var type = NodeType.UriType;
var token = tokens[startPos];
var line = token.ln;
var column = token.col;
var content = [];
var end = void 0;
var uriType = tokens[startPos].uriType;
// Skip `url` and `(`.
pos += 2;
if (uriType === 1) content = content.concat(getUri1());else if (uriType === 2) content = content.concat(getUri2());else end = getLastPosition(content, line, column, 4);
if (!end) end = getLastPosition(content, line, column, 1);
// Skip `)`.
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token type is valid URI character
* @param {number} i Token's index number
* @return {number} Length of raw node
*/
function checkUriRawCharacters(i) {
var start = i;
var l = void 0;
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else {
switch (tokens[i].type) {
case TokenType.ExclamationMark:
case TokenType.NumberSign:
case TokenType.DollarSign:
case TokenType.PercentSign:
case TokenType.Ampersand:
case TokenType.Asterisk:
case TokenType.PlusSign:
case TokenType.Comma:
case TokenType.HyphenMinus:
case TokenType.FullStop:
case TokenType.Solidus:
case TokenType.Colon:
case TokenType.Semicolon:
case TokenType.LessThanSign:
case TokenType.EqualsSign:
case TokenType.GreaterThanSign:
case TokenType.QuotationMark:
case TokenType.CommercialAt:
case TokenType.LeftSquareBracket:
case TokenType.RightSquareBracket:
case TokenType.CircumflexAccent:
case TokenType.LowLine:
case TokenType.LeftCurlyBracket:
case TokenType.VerticalLine:
case TokenType.RightCurlyBracket:
case TokenType.Tilde:
i += 1;
break;
default:
return 0;
}
}
return i - start;
}
/**
* Check if content of URI can be contained within a raw node
* @param {number} i Token's index number
* @return {number} Length of raw node
*/
function checkUriRaw(i) {
var start = i;
var l = void 0;
while (i < tokensLength) {
if (checkInterpolation(i) || checkVariable(i)) break;else if (l = checkUriRawCharacters(i)) i += l;else break;
}
tokens[start].uri_raw_end = i;
return i - start;
}
/**
* Get a raw node
* @return {Node}
*/
function getUriRaw() {
var startPos = pos;
var type = NodeType.RawType;
var token = tokens[startPos];
var line = token.ln;
var column = token.col;
var content = [];
var l = void 0;
while (pos < tokens[startPos].uri_raw_end) {
if (checkInterpolation(pos) || checkVariable(pos)) break;else if (l = checkUriRawCharacters(pos)) pos += l;else break;
}
content = joinValues(startPos, pos - 1);
return newNode(type, content, line, column);
}
/**
* Check for a raw (without quotes) URI
* (1) http://foo.com/bar.png
* (2) http://foo.com/#{$bar}.png
* (3) #{$foo}/bar.png
* (4) #{$foo}
* @param {number} i Token's index number
* @return {number} Length of URI node
*/
function checkUri1(i) {
var start = i;
var l = void 0;
if (l = checkSC(i)) i += l;
while (i < tokensLength) {
if (l = checkInterpolation(i) || checkUriRaw(i)) i += l;else break;
}
if (l = checkSC(i)) i += l;
// Check that we are at the end of the uri
if (i < tokens[start - 1].right) return 0;
tokens[start].uri_end = i;
return i - start;
}
/**
* Get a raw (without quotes) URI
node
* @return {Array}
*/
function getUri1() {
var startPos = pos;
var content = [];
if (checkSC(pos)) content = content.concat(getSC());
while (pos < tokens[startPos].uri_end) {
if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkUriRaw(pos)) content.push(getUriRaw());else break;
}
if (checkSC(pos)) content = content.concat(getSC());
return content;
}
/**
* Check for a non-raw (with quotes) URI
* (1) 'http://foo.com/bar.png'
* (2) 'http://foo.com/'#{$bar}.png
* (3) #{$foo}'/bar.png'
* @param {number} i Token's index number
* @return {number} Length of URI node
*/
function checkUri2(i) {
var start = i;
var l = void 0;
while (i < tokensLength) {
if (l = checkSC(i)) i += l;else if (l = checkString(i)) i += l;else if (l = checkFunction(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = checkIdentOrInterpolation(i)) i += l;else if (l = checkVariable(i)) i += l;else break;
}
// Check that we are at the end of the uri
if (i < tokens[start - 1].right) return 0;
tokens[start].uri_end = i;
return i - start;
}
/**
* Get a non-raw (with quotes) URI node
* @return {Array}
*/
function getUri2() {
var startPos = pos;
var content = [];
while (pos < tokens[startPos].uri_end) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkUnary(pos)) content.push(getUnary());else if (_checkValue(pos)) content.push(_getValue());else break;
}
return content;
}
/**
* Check if token is part of a value
* @param {Number} i Token's index number
* @returns {Number} Length of the value
*/
function checkValue(i) {
var start = i;
var l = void 0;
var s = void 0;
var _i = void 0;
while (i < tokensLength) {
if (checkDeclDelim(i)) break;
s = checkSC(i);
_i = i + s;
if (l = _checkValue(_i)) i += l + s;
if (!l || checkBlock(i - l)) break;
}
tokens[start].value_end = i;
return i - start;
}
/**
* @returns {Array}
*/
function getValue() {
var type = NodeType.ValueType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var end = tokens[pos].value_end;
var content = [];
var _pos = void 0;
var s = void 0;
while (pos < end) {
s = checkSC(pos);
_pos = pos + s;
if (checkDeclDelim(_pos)) break;
if (!_checkValue(_pos)) break;
if (s) content = content.concat(getSC());
content.push(_getValue());
if (checkBlock(_pos)) break;
}
return newNode(type, content, line, column);
}
/**
* @param {Number} i Token's index number
* @returns {Number}
*/
function _checkValue(i) {
var l = void 0;
if (l = checkInterpolation(i)) tokens[i].value_child = 1;else if (l = checkVariable(i)) tokens[i].value_child = 2;else if (l = checkVhash(i)) tokens[i].value_child = 3;else if (l = checkBlock(i)) tokens[i].value_child = 4;else if (l = checkAtkeyword(i)) tokens[i].value_child = 5;else if (l = checkOperator(i)) tokens[i].value_child = 6;else if (l = checkImportant(i)) tokens[i].value_child = 7;else if (l = checkGlobal(i)) tokens[i].value_child = 8;else if (l = checkDefault(i)) tokens[i].value_child = 9;else if (l = checkProgid(i)) tokens[i].value_child = 10;else if (l = checkAny(i)) tokens[i].value_child = 11;else if (l = checkParentSelector(i)) tokens[i].value_child = 12;
return l;
}
/**
* @returns {Array}
*/
function _getValue() {
var childType = tokens[pos].value_child;
if (childType === 1) return getInterpolation();
if (childType === 2) return getVariable();
if (childType === 3) return getVhash();
if (childType === 4) return getBlock();
if (childType === 5) return getAtkeyword();
if (childType === 6) return getOperator();
if (childType === 7) return getImportant();
if (childType === 8) return getGlobal();
if (childType === 9) return getDefault();
if (childType === 10) return getProgid();
if (childType === 11) return getAny();
if (childType === 12) return getParentSelector();
}
/**
* @param {number} i Token's index number
* @returns {number} Length of the value
*/
function checkSingleValue(i) {
var start = i;
var l = void 0;
var s = void 0;
var _i = void 0;
while (i < tokensLength) {
if (checkDeclDelim(i) || checkDelim(i)) break;
s = checkSC(i);
_i = i + s;
if (l = _checkValue(_i)) i += l + s;
if (!l || checkBlock(i - l)) break;
}
return i - start;
}
/**
* @returns {Array}
*/
function getSingleValue() {
var type = NodeType.ValueType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var _pos = void 0;
var s = void 0;
while (pos < tokensLength) {
s = checkSC(pos);
_pos = pos + s;
if (checkDeclDelim(_pos) || checkDelim(_pos)) break;
if (!_checkValue(_pos)) break;
if (s) content = content.concat(getSC());
content.push(_getValue());
if (checkBlock(_pos)) break;
}
return newNode(type, content, line, column);
}
/**
* Check if token is part of a variable
* @param {Number} i Token's index number
* @returns {Number} Length of the variable
*/
function checkVariable(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
// Skip `$`.
if (tokens[i].type === TokenType.DollarSign) i++;else return 0;
if (l = checkIdent(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a variable
* @returns {Array} `['variable', ['ident', x]]` where `x` is
* a variable name.
*/
function getVariable() {
var type = NodeType.VariableType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `$`.
pos++;
var content = [getIdent()];
return newNode(type, content, line, column);
}
/**
* Check if token is part of a variables list (e.g. `$values...`).
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkVariablesList(i) {
var d = 0; // Number of dots
var l = void 0;
if (i >= tokensLength) return 0;
if (l = checkVariable(i)) i += l;else return 0;
while (i < tokensLength && tokens[i].type === TokenType.FullStop) {
d++;
i++;
}
return d === 3 ? l + d : 0;
}
/**
* Get node with a variables list
* @returns {Array} `['variableslist', ['variable', ['ident', x]]]` where
* `x` is a variable name.
*/
function getVariablesList() {
var type = NodeType.VariablesListType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getVariable()];
var end = getLastPosition(content, line, column, 3);
// Skip `...`.
pos += 3;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
* some value
* @param {Number} i Token's index number
* @returns {Number}
*/
function checkVhash(i) {
var start = i;
var l = void 0;
if (i >= tokensLength) return 0;
// Skip `#`.
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
if (l = checkNmName2(i)) i += l;else return 0;
return i - start;
}
/**
* Get node with a hexadecimal number (e.g. `#fff`) inside some value
* @returns {Array} `['vhash', x]` where `x` is a hexadecimal number
* converted to string (without `#`, e.g. `'fff'`).
*/
function getVhash() {
var type = NodeType.VhashType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
// Skip `#`.
pos++;
var content = getNmName2();
var end = getLastPosition(content, line, column + 1);
return newNode(type, content, line, column, end);
}
function checkSelectorsGroup(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
var selectorCounter = 0;
var delimCounter = 0;
if (l = checkSelector(i)) {
i += l;
selectorCounter++;
} else return 0;
while (i < tokensLength) {
var tempStart = i;
var tempIndex = i;
var tempLength = void 0;
var spaceBefore = checkSC(tempIndex);
if (tempLength = checkDelim(tempIndex + spaceBefore)) {
tempIndex += spaceBefore + tempLength;
delimCounter++;
if (tempLength = checkSC(tempIndex)) tempIndex += tempLength;
if (tempLength = checkSelector(tempIndex)) {
tempIndex += tempLength;
selectorCounter++;
}
} else break;
i += tempIndex - tempStart;
}
tokens[start].selectorsGroupEnd = i;
tokens[start].selectorsGroupSelectorCount = selectorCounter;
tokens[start].selectorsGroupDelimCount = delimCounter;
return i - start;
}
function getSelectorsGroup() {
var selectorsGroup = [];
var selectorCounter = 0;
var delimCounter = 0;
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
var selectorCount = tokens[pos].selectorsGroupSelectorCount;
var delimCount = tokens[pos].selectorsGroupDelimCount;
selectorsGroup.push(getSelector());
selectorCounter++;
while (pos < selectorsGroupEnd) {
if (delimCounter < delimCount) {
selectorsGroup = selectorsGroup.concat(getSC());
selectorsGroup = selectorsGroup.concat(getDelim());
delimCounter++;
selectorsGroup = selectorsGroup.concat(getSC());
if (selectorCounter < selectorCount) {
selectorsGroup = selectorsGroup.concat(getSelector());
selectorCounter++;
}
}
}
return selectorsGroup;
}
function checkSelector(i) {
var l = void 0;
if (l = checkSelector1(i)) tokens[i].selectorType = 1;else if (l = checkSelector2(i)) tokens[i].selectorType = 2;
return l;
}
function getSelector() {
var selectorType = tokens[pos].selectorType;
if (selectorType === 1) return getSelector1();else return getSelector2();
}
/**
* Checks for selector which starts with a compound selector.
*/
function checkSelector1(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkCompoundSelector(i)) i += l;else return 0;
while (i < tokensLength) {
var space = checkSC(i);
var comma = checkCombinator(i + space);
if (!space && !comma) break;
if (comma) {
i += space + comma;
space = checkSC(i);
}
if (l = checkCompoundSelector(i + space)) i += space + l;else break;
}
tokens[start].selectorEnd = i;
return i - start;
}
function getSelector1() {
var type = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var selectorEnd = token.selectorEnd;
var content = getCompoundSelector();
while (pos < selectorEnd) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
}
return newNode(type, content, line, column);
}
/**
* Checks for a selector that starts with a combinator.
*/
function checkSelector2(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkCombinator(i)) i += l;else return 0;
while (i < tokensLength) {
var spaceBefore = checkSC(i);
if (l = checkCompoundSelector(i + spaceBefore)) i += spaceBefore + l;else break;
var spaceAfter = checkSC(i);
var comma = checkCombinator(i + spaceAfter);
if (!spaceAfter && !comma) break;
if (comma) {
i += spaceAfter + comma;
}
}
tokens[start].selectorEnd = i;
return i - start;
}
function getSelector2() {
var type = NodeType.SelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var selectorEnd = token.selectorEnd;
var content = [getCombinator()];
while (pos < selectorEnd) {
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
}
return newNode(type, content, line, column);
}
function checkCompoundSelector(i) {
var l = void 0;
if (l = checkCompoundSelector1(i)) {
tokens[i].compoundSelectorType = 1;
} else if (l = checkCompoundSelector2(i)) {
tokens[i].compoundSelectorType = 2;
}
return l;
}
function getCompoundSelector() {
var type = tokens[pos].compoundSelectorType;
if (type === 1) return getCompoundSelector1();
if (type === 2) return getCompoundSelector2();
}
/**
* Check for compound selectors that start with either a type selector,
* placeholder or parent selector with extension
* (1) `foo.bar`
* (2) `foo[attr=val]`
* (3) `foo:first-of-type`
* (4) `foo%bar`
* @param {number} i Token's index
* @return {number} Compound selector's length
*/
function checkCompoundSelector1(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkUniversalSelector(i) || checkTypeSelector(i) || checkPlaceholder(i) || checkParentSelectorWithExtension(i)) i += l;else return 0;
while (i < tokensLength) {
var _l2 = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i) || checkPlaceholder(i) || checkInterpolation(i);
if (_l2) i += _l2;else break;
}
tokens[start].compoundSelectorEnd = i;
return i - start;
}
/**
* @return {Array} An array of nodes that make up the compound selector
*/
function getCompoundSelector1() {
var sequence = [];
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
if (checkUniversalSelector(pos)) sequence.push(getUniversalSelector());else if (checkTypeSelector(pos)) sequence.push(getTypeSelector());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkParentSelectorWithExtension(pos)) sequence = sequence.concat(getParentSelectorWithExtension());
while (pos < compoundSelectorEnd) {
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkInterpolation(pos)) sequence.push(getInterpolation());else break;
}
return sequence;
}
/**
* Check for all other compound selectors
* (1) `.foo.bar`
* (2) `.foo[attr=val]`
* (3) `.foo:first-of-type`
* (4) `.foo%bar`
* (5) `.foo#{$bar}`
* @param {number} i Token's index
* @return {number} Compound selector's length
*/
function checkCompoundSelector2(i) {
if (i >= tokensLength) return 0;
var start = i;
while (i < tokensLength) {
var l = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i) || checkPlaceholder(i) || checkInterpolation(i);
if (l) i += l;else break;
}
tokens[start].compoundSelectorEnd = i;
return i - start;
}
/**
* @return {Array} An array of nodes that make up the compound selector
*/
function getCompoundSelector2() {
var sequence = [];
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
while (pos < compoundSelectorEnd) {
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkInterpolation(pos)) sequence.push(getInterpolation());else break;
}
return sequence;
}
function checkUniversalSelector(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (tokens[i].type === TokenType.Asterisk) i++;else return 0;
return i - start;
}
function getUniversalSelector() {
var type = NodeType.UniversalSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
var end = void 0;
if (checkNamePrefix(pos)) {
content.push(getNamePrefix());
end = getLastPosition(content, line, column, 1);
}
pos++;
return newNode(type, content, line, column, end);
}
/**
* Check if token is part of a type selector
* @param {number} i Token's index
* @return {number} Type selector's length
*/
function checkTypeSelector(i) {
if (i >= tokensLength) return 0;
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
/**
* Get type selector node
* @return {Node}
*/
function getTypeSelector() {
var type = NodeType.TypeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkNamePrefix(pos)) content.push(getNamePrefix());
content = content.concat(getIdentOrInterpolation());
return newNode(type, content, line, column);
}
function checkAttributeSelector(i) {
var l = void 0;
if (l = checkAttributeSelector1(i)) tokens[i].attributeSelectorType = 1;else if (l = checkAttributeSelector2(i)) tokens[i].attributeSelectorType = 2;
return l;
}
function getAttributeSelector() {
var type = tokens[pos].attributeSelectorType;
if (type === 1) return getAttributeSelector1();else return getAttributeSelector2();
}
/**
* (1) `[panda=nani]`
* (2) `[panda='nani']`
* (3) `[panda='nani' i]`
*
*/
function checkAttributeSelector1(i) {
var start = i;
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeName(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeMatch(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeValue(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeFlags(i)) {
i += l;
if (l = checkSC(i)) i += l;
}
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
return i - start;
}
function getAttributeSelector1() {
var type = NodeType.AttributeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `[`.
pos++;
content = content.concat(getSC(), getAttributeName(), getSC(), getAttributeMatch(), getSC(), getAttributeValue(), getSC());
if (checkAttributeFlags(pos)) {
content.push(getAttributeFlags());
content = content.concat(getSC());
}
// Skip `]`.
pos++;
var end = getLastPosition(content, line, column, 1);
return newNode(type, content, line, column, end);
}
/**
* (1) `[panda]`
*/
function checkAttributeSelector2(i) {
var start = i;
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
var l = void 0;
if (l = checkSC(i)) i += l;
if (l = checkAttributeName(i)) i += l;else return 0;
if (l = checkSC(i)) i += l;
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
return i - start;
}
function getAttributeSelector2() {
var type = NodeType.AttributeSelectorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
// Skip `[`.
pos++;
content = content.concat(getSC(), getAttributeName(), getSC());
// Skip `]`.
pos++;
var end = getLastPosition(content, line, column, 1);
return newNode(type, content, line, column, end);
}
function checkAttributeName(i) {
var start = i;
var l = void 0;
if (l = checkNamePrefix(i)) i += l;
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
return i - start;
}
function getAttributeName() {
var type = NodeType.AttributeNameType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkNamePrefix(pos)) content.push(getNamePrefix());
content = content.concat(getIdentOrInterpolation());
return newNode(type, content, line, column);
}
function checkAttributeMatch(i) {
var l = void 0;
if (l = checkAttributeMatch1(i)) tokens[i].attributeMatchType = 1;else if (l = checkAttributeMatch2(i)) tokens[i].attributeMatchType = 2;
return l;
}
function getAttributeMatch() {
var type = tokens[pos].attributeMatchType;
if (type === 1) return getAttributeMatch1();else return getAttributeMatch2();
}
function checkAttributeMatch1(i) {
var start = i;
var type = tokens[i].type;
if (type === TokenType.Tilde || type === TokenType.VerticalLine || type === TokenType.CircumflexAccent || type === TokenType.DollarSign || type === TokenType.Asterisk) i++;else return 0;
if (tokens[i].type === TokenType.EqualsSign) i++;else return 0;
return i - start;
}
function getAttributeMatch1() {
var type = NodeType.AttributeMatchType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = tokens[pos].value + tokens[pos + 1].value;
pos += 2;
return newNode(type, content, line, column);
}
function checkAttributeMatch2(i) {
if (tokens[i].type === TokenType.EqualsSign) return 1;else return 0;
}
function getAttributeMatch2() {
var type = NodeType.AttributeMatchType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '=';
pos++;
return newNode(type, content, line, column);
}
function checkAttributeValue(i) {
return checkString(i) || checkIdentOrInterpolation(i);
}
function getAttributeValue() {
var type = NodeType.AttributeValueType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (checkString(pos)) content.push(getString());else content = content.concat(getIdentOrInterpolation());
return newNode(type, content, line, column);
}
function checkAttributeFlags(i) {
return checkIdentOrInterpolation(i);
}
function getAttributeFlags() {
var type = NodeType.AttributeFlagsType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = getIdentOrInterpolation();
return newNode(type, content, line, column);
}
function checkNamePrefix(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (l = checkNamePrefix1(i)) tokens[i].namePrefixType = 1;else if (l = checkNamePrefix2(i)) tokens[i].namePrefixType = 2;
return l;
}
function getNamePrefix() {
var type = tokens[pos].namePrefixType;
if (type === 1) return getNamePrefix1();else return getNamePrefix2();
}
/**
* (1) `panda|`
* (2) `panda<comment>|`
*/
function checkNamePrefix1(i) {
var start = i;
var l = void 0;
if (l = checkNamespacePrefix(i)) i += l;else return 0;
if (l = checkCommentML(i)) i += l;
if (l = checkNamespaceSeparator(i)) i += l;else return 0;
return i - start;
}
function getNamePrefix1() {
var type = NodeType.NamePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
content.push(getNamespacePrefix());
if (checkCommentML(pos)) content.push(getCommentML());
content.push(getNamespaceSeparator());
return newNode(type, content, line, column);
}
/**
* (1) `|`
*/
function checkNamePrefix2(i) {
return checkNamespaceSeparator(i);
}
function getNamePrefix2() {
var type = NodeType.NamePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [getNamespaceSeparator()];
return newNode(type, content, line, column);
}
/**
* (1) `*`
* (2) `panda`
*/
function checkNamespacePrefix(i) {
if (i >= tokensLength) return 0;
var l = void 0;
if (tokens[i].type === TokenType.Asterisk) return 1;else if (l = checkIdentOrInterpolation(i)) return l;else return 0;
}
function getNamespacePrefix() {
var type = NodeType.NamespacePrefixType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = [];
if (token.type === TokenType.Asterisk) {
var asteriskNode = newNode(NodeType.IdentType, '*', line, column);
content.push(asteriskNode);
pos++;
} else if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());
return newNode(type, content, line, column);
}
/**
* (1) `|`
*/
function checkNamespaceSeparator(i) {
if (i >= tokensLength) return 0;
if (tokens[i].type !== TokenType.VerticalLine) return 0;
// Return false if `|=` - [attr|=value]
if (tokens[i + 1] && tokens[i + 1].type === TokenType.EqualsSign) return 0;
return 1;
}
function getNamespaceSeparator() {
var type = NodeType.NamespaceSeparatorType;
var token = tokens[pos];
var line = token.ln;
var column = token.col;
var content = '|';
pos++;
return newNode(type, content, line, column);
}
module.exports = function (_tokens, context) {
tokens = _tokens;
tokensLength = tokens.length;
pos = 0;
return contexts[context]();
};
/***/ }),
/* 28 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
module.exports = function (css, tabSize) {
var TokenType = __webpack_require__(13);
var tokens = [];
var urlMode = false;
var c = void 0; // Current character
var cn = void 0; // Next character
var pos = 0;
var tn = 0;
var ln = 1;
var col = 1;
var Punctuation = {
' ': TokenType.Space,
'\n': TokenType.Newline,
'\r': TokenType.Newline,
'\t': TokenType.Tab,
'!': TokenType.ExclamationMark,
'"': TokenType.QuotationMark,
'#': TokenType.NumberSign,
'$': TokenType.DollarSign,
'%': TokenType.PercentSign,
'&': TokenType.Ampersand,
'\'': TokenType.Apostrophe,
'(': TokenType.LeftParenthesis,
')': TokenType.RightParenthesis,
'*': TokenType.Asterisk,
'+': TokenType.PlusSign,
',': TokenType.Comma,
'-': TokenType.HyphenMinus,
'.': TokenType.FullStop,
'/': TokenType.Solidus,
':': TokenType.Colon,
';': TokenType.Semicolon,
'<': TokenType.LessThanSign,
'=': TokenType.EqualsSign,
'==': TokenType.EqualitySign,
'!=': TokenType.InequalitySign,
'>': TokenType.GreaterThanSign,
'?': TokenType.QuestionMark,
'@': TokenType.CommercialAt,
'[': TokenType.LeftSquareBracket,
']': TokenType.RightSquareBracket,
'^': TokenType.CircumflexAccent,
'_': TokenType.LowLine,
'{': TokenType.LeftCurlyBracket,
'|': TokenType.VerticalLine,
'}': TokenType.RightCurlyBracket,
'~': TokenType.Tilde,
'`': TokenType.Backtick
};
/**
* Add a token to the token list
* @param {string} type
* @param {string} value
*/
function pushToken(type, value, column) {
tokens.push({
tn: tn++,
ln: ln,
col: column,
type: type,
value: value
});
}
/**
* Check if a character is a decimal digit
* @param {string} c Character
* @returns {boolean}
*/
function isDecimalDigit(c) {
return '0123456789'.indexOf(c) >= 0;
}
/**
* Parse spaces
* @param {string} css Unparsed part of CSS string
*/
function parseSpaces(css) {
var start = pos;
// Read the string until we meet a non-space character:
for (; pos < css.length; pos++) {
if (css.charAt(pos) !== ' ') break;
}
// Add a substring containing only spaces to tokens:
pushToken(TokenType.Space, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Parse a string within quotes
* @param {string} css Unparsed part of CSS string
* @param {string} q Quote (either `'` or `"`)
*/
function parseString(css, q) {
var start = pos;
// Read the string until we meet a matching quote:
for (pos++; pos < css.length; pos++) {
// Skip escaped quotes:
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) === q) break;
}
// Add the string (including quotes) to tokens:
var type = q === '"' ? TokenType.StringDQ : TokenType.StringSQ;
pushToken(type, css.substring(start, pos + 1), col);
col += pos - start;
}
/**
* Parse numbers
* @param {string} css Unparsed part of CSS string
*/
function parseDecimalNumber(css) {
var start = pos;
// Read the string until we meet a character that's not a digit:
for (; pos < css.length; pos++) {
if (!isDecimalDigit(css.charAt(pos))) break;
}
// Add the number to tokens:
pushToken(TokenType.DecimalNumber, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Parse identifier
* @param {string} css Unparsed part of CSS string
*/
function parseIdentifier(css) {
var start = pos;
// Skip all opening slashes:
while (css.charAt(pos) === '/') {
pos++;
} // Read the string until we meet a punctuation mark:
for (; pos < css.length; pos++) {
// Skip all '\':
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) in Punctuation) break;
}
var ident = css.substring(start, pos--);
// Enter url mode if parsed substring is `url`:
if (!urlMode && ident === 'url' && css.charAt(pos + 1) === '(') {
urlMode = true;
}
// Add identifier to tokens:
pushToken(TokenType.Identifier, ident, col);
col += pos - start;
}
/**
* Parse equality sign
*/
function parseEquality() {
pushToken(TokenType.EqualitySign, '==', col);
pos++;
col++;
}
/**
* Parse inequality sign
*/
function parseInequality() {
pushToken(TokenType.InequalitySign, '!=', col);
pos++;
col++;
}
/**
* Parse a multiline comment
* @param {string} css Unparsed part of CSS string
*/
function parseMLComment(css) {
var start = pos;
// Read the string until we meet `*/`.
// Since we already know first 2 characters (`/*`), start reading
// from `pos + 2`:
for (pos += 2; pos < css.length; pos++) {
if (css.charAt(pos) === '*' && css.charAt(pos + 1) === '/') {
pos++;
break;
}
}
// Add full comment (including `/*` and `*/`) to the list of tokens:
var comment = css.substring(start, pos + 1);
pushToken(TokenType.CommentML, comment, col);
var newlines = comment.split('\n');
if (newlines.length > 1) {
ln += newlines.length - 1;
col = newlines[newlines.length - 1].length;
} else {
col += pos - start;
}
}
/**
* Parse a single line comment
* @param {string} css Unparsed part of CSS string
*/
function parseSLComment(css) {
var start = pos;
// Read the string until we meet line break.
// Since we already know first 2 characters (`//`), start reading
// from `pos + 2`:
for (pos += 2; pos < css.length; pos++) {
if (css.charAt(pos) === '\n' || css.charAt(pos) === '\r') {
break;
}
}
// Add comment (including `//` and line break) to the list of tokens:
pushToken(TokenType.CommentSL, css.substring(start, pos--), col);
col += pos - start;
}
/**
* Convert a CSS string to a list of tokens
* @param {string} css CSS string
* @returns {Array} List of tokens
* @private
*/
function getTokens(css) {
// Parse string, character by character:
for (pos = 0; pos < css.length; col++, pos++) {
c = css.charAt(pos);
cn = css.charAt(pos + 1);
// If we meet `/*`, it's a start of a multiline comment.
// Parse following characters as a multiline comment:
if (c === '/' && cn === '*') {
parseMLComment(css);
}
// If we meet `//` and it is not a part of url:
else if (!urlMode && c === '/' && cn === '/') {
// If we're currently inside a block, treat `//` as a start
// of identifier. Else treat `//` as a start of a single-line
// comment:
parseSLComment(css);
}
// If current character is a double or single quote, it's a start
// of a string:
else if (c === '"' || c === "'") {
parseString(css, c);
}
// If current character is a space:
else if (c === ' ') {
parseSpaces(css);
}
// If current character is `=`, it must be combined with next `=`
else if (c === '=' && cn === '=') {
parseEquality(css);
}
// If we meet `!=`, this must be inequality
else if (c === '!' && cn === '=') {
parseInequality(css);
}
// If current character is a punctuation mark:
else if (c in Punctuation) {
// Check for CRLF here or just LF
if (c === '\r' && cn === '\n' || c === '\n') {
// If \r we know the next character is \n due to statement above
// so we push a CRLF token type to the token list and importantly
// skip the next character so as not to double count newlines or
// columns etc
if (c === '\r') {
pushToken(TokenType.Newline, '\r\n', col);
pos++; // If CRLF skip the next character and push crlf token
} else if (c === '\n') {
// If just a LF newline and not part of CRLF newline we can just
// push punctuation as usual
pushToken(Punctuation[c], c, col);
}
ln++; // Go to next line
col = 0; // Reset the column count
} else if (c !== '\r' && c !== '\n') {
// Handle all other punctuation and add to list of tokens
pushToken(Punctuation[c], c, col);
} // Go to next line
if (c === ')') urlMode = false; // Exit url mode
else if (c === '\t' && tabSize > 1) col += tabSize - 1;
}
// If current character is a decimal digit:
else if (isDecimalDigit(c)) {
parseDecimalNumber(css);
}
// If current character is anything else:
else {
parseIdentifier(css);
}
}
return tokens;
}
return getTokens(css);
};
/***/ }),
/* 29 */
/***/ (function(module, exports, __webpack_require__) {
'use strict';
var Node = __webpack_require__(1);
var NodeTypes = __webpack_require__(15);
module.exports = function () {
return new Node({
type: NodeTypes.StylesheetType,
content: [],
start: [0, 0],
end: [0, 0]
});
};
/***/ })
/******/ ])
});
;