Actualizacion maquina principal
[dotfiles/.git] / .config / coc / extensions / node_modules / coc-prettier / node_modules / gonzales-pe / lib / gonzales.js
diff --git a/.config/coc/extensions/node_modules/coc-prettier/node_modules/gonzales-pe/lib/gonzales.js b/.config/coc/extensions/node_modules/coc-prettier/node_modules/gonzales-pe/lib/gonzales.js
new file mode 100644 (file)
index 0000000..b8dea4e
--- /dev/null
@@ -0,0 +1,22435 @@
+(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.3.0","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.2.5"},"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 = checkIdentOrInterpolation(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 = getIdentOrInterpolation();
+
+         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 = checkIdentOrInterpolation(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 = getIdentOrInterpolation();
+
+         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]
+         });
+       };
+
+/***/ })
+/******/ ])
+});
+;
\ No newline at end of file