22435 lines
548 KiB
JavaScript
22435 lines
548 KiB
JavaScript
(function webpackUniversalModuleDefinition(root, factory) {
|
|
if(typeof exports === 'object' && typeof module === 'object')
|
|
module.exports = factory();
|
|
else if(typeof define === 'function' && define.amd)
|
|
define([], factory);
|
|
else if(typeof exports === 'object')
|
|
exports["gonzales"] = factory();
|
|
else
|
|
root["gonzales"] = factory();
|
|
})(this, function() {
|
|
return /******/ (function(modules) { // webpackBootstrap
|
|
/******/ // The module cache
|
|
/******/ var installedModules = {};
|
|
|
|
/******/ // The require function
|
|
/******/ function __webpack_require__(moduleId) {
|
|
|
|
/******/ // Check if module is in cache
|
|
/******/ if(installedModules[moduleId])
|
|
/******/ return installedModules[moduleId].exports;
|
|
|
|
/******/ // Create a new module (and put it into the cache)
|
|
/******/ var module = installedModules[moduleId] = {
|
|
/******/ exports: {},
|
|
/******/ id: moduleId,
|
|
/******/ loaded: false
|
|
/******/ };
|
|
|
|
/******/ // Execute the module function
|
|
/******/ modules[moduleId].call(module.exports, module, module.exports, __webpack_require__);
|
|
|
|
/******/ // Flag the module as loaded
|
|
/******/ module.loaded = true;
|
|
|
|
/******/ // Return the exports of the module
|
|
/******/ return module.exports;
|
|
/******/ }
|
|
|
|
|
|
/******/ // expose the modules object (__webpack_modules__)
|
|
/******/ __webpack_require__.m = modules;
|
|
|
|
/******/ // expose the module cache
|
|
/******/ __webpack_require__.c = installedModules;
|
|
|
|
/******/ // __webpack_public_path__
|
|
/******/ __webpack_require__.p = "";
|
|
|
|
/******/ // Load entry module and return exports
|
|
/******/ return __webpack_require__(0);
|
|
/******/ })
|
|
/************************************************************************/
|
|
/******/ ([
|
|
/* 0 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var Node = __webpack_require__(1);
|
|
var parse = __webpack_require__(7);
|
|
|
|
module.exports = {
|
|
createNode: function createNode(options) {
|
|
return new Node(options);
|
|
},
|
|
parse: parse
|
|
};
|
|
|
|
/***/ }),
|
|
/* 1 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
/**
|
|
* @param {string} type
|
|
* @param {array|string} content
|
|
* @param {number} line
|
|
* @param {number} column
|
|
* @constructor
|
|
*/
|
|
|
|
var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
|
|
|
|
function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
|
|
|
|
var Node = function () {
|
|
function Node(options) {
|
|
_classCallCheck(this, Node);
|
|
|
|
this.type = options.type;
|
|
this.content = options.content;
|
|
this.syntax = options.syntax;
|
|
|
|
if (options.start) this.start = options.start;
|
|
if (options.end) this.end = options.end;
|
|
}
|
|
|
|
/**
|
|
* @param {String} type Node type
|
|
* @return {Boolean} Whether there is a child node of given type
|
|
*/
|
|
|
|
|
|
Node.prototype.contains = function contains(type) {
|
|
if (!Array.isArray(this.content)) {
|
|
return false;
|
|
}
|
|
|
|
return this.content.some(function (node) {
|
|
return node.type === type;
|
|
});
|
|
};
|
|
|
|
/**
|
|
* @param {String} type Node type
|
|
* @param {Function} callback Function to call for every found node
|
|
*/
|
|
|
|
|
|
Node.prototype.eachFor = function eachFor(type, callback) {
|
|
if (!Array.isArray(this.content)) return;
|
|
|
|
if (typeof type !== 'string') {
|
|
callback = type;
|
|
type = null;
|
|
}
|
|
|
|
var l = this.content.length;
|
|
var breakLoop;
|
|
|
|
for (var i = l; i--;) {
|
|
if (breakLoop === null) break;
|
|
|
|
if (!type || this.content[i] && this.content[i].type === type) breakLoop = callback(this.content[i], i, this);
|
|
}
|
|
|
|
if (breakLoop === null) return null;
|
|
};
|
|
|
|
/**
|
|
* @param {String} type
|
|
* @return {?Node} First child node or `null` if nothing's been found.
|
|
*/
|
|
|
|
|
|
Node.prototype.first = function first(type) {
|
|
if (!Array.isArray(this.content)) return null;
|
|
|
|
if (!type) return this.content[0];
|
|
|
|
var i = 0;
|
|
var l = this.content.length;
|
|
|
|
for (; i < l; i++) {
|
|
if (this.content[i].type === type) return this.content[i];
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
/**
|
|
* @param {String} type Node type
|
|
* @param {Function} callback Function to call for every found node
|
|
*/
|
|
|
|
|
|
Node.prototype.forEach = function forEach(type, callback) {
|
|
if (!Array.isArray(this.content)) return;
|
|
|
|
if (typeof type !== 'string') {
|
|
callback = type;
|
|
type = null;
|
|
}
|
|
|
|
var i = 0;
|
|
var l = this.content.length;
|
|
var breakLoop;
|
|
|
|
for (; i < l; i++) {
|
|
if (breakLoop === null) break;
|
|
|
|
if (!type || this.content[i] && this.content[i].type === type) breakLoop = callback(this.content[i], i, this);
|
|
}
|
|
|
|
if (breakLoop === null) return null;
|
|
};
|
|
|
|
/**
|
|
* @param {Number} index
|
|
* @return {?Node}
|
|
*/
|
|
|
|
|
|
Node.prototype.get = function get(index) {
|
|
if (!Array.isArray(this.content)) return null;
|
|
|
|
var node = this.content[index];
|
|
return node ? node : null;
|
|
};
|
|
|
|
/**
|
|
* @param {Number} index
|
|
* @param {Node} node
|
|
*/
|
|
|
|
|
|
Node.prototype.insert = function insert(index, node) {
|
|
if (!Array.isArray(this.content)) return;
|
|
|
|
this.content.splice(index, 0, node);
|
|
};
|
|
|
|
/**
|
|
* @param {String} type
|
|
* @return {Boolean} Whether the node is of given type
|
|
*/
|
|
|
|
|
|
Node.prototype.is = function is(type) {
|
|
return this.type === type;
|
|
};
|
|
|
|
/**
|
|
* @param {String} type
|
|
* @return {?Node} Last child node or `null` if nothing's been found.
|
|
*/
|
|
|
|
|
|
Node.prototype.last = function last(type) {
|
|
if (!Array.isArray(this.content)) return null;
|
|
|
|
var i = this.content.length;
|
|
if (!type) return this.content[i - 1];
|
|
|
|
for (; i--;) {
|
|
if (this.content[i].type === type) return this.content[i];
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
/**
|
|
* Number of child nodes.
|
|
* @type {number}
|
|
*/
|
|
|
|
|
|
/**
|
|
* @param {Number} index
|
|
* @return {Node}
|
|
*/
|
|
Node.prototype.removeChild = function removeChild(index) {
|
|
if (!Array.isArray(this.content)) return;
|
|
|
|
var removedChild = this.content.splice(index, 1);
|
|
|
|
return removedChild;
|
|
};
|
|
|
|
Node.prototype.toJson = function toJson() {
|
|
return JSON.stringify(this, false, 2);
|
|
};
|
|
|
|
Node.prototype.toString = function toString() {
|
|
var stringify = void 0;
|
|
|
|
try {
|
|
stringify = __webpack_require__(2)("./" + this.syntax + '/stringify');
|
|
} catch (e) {
|
|
var message = 'Syntax "' + this.syntax + '" is not supported yet, sorry';
|
|
return console.error(message);
|
|
}
|
|
|
|
return stringify(this);
|
|
};
|
|
|
|
/**
|
|
* @param {Function} callback
|
|
*/
|
|
|
|
|
|
Node.prototype.traverse = function traverse(callback, index) {
|
|
var level = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : 0;
|
|
var parent = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : null;
|
|
|
|
var breakLoop;
|
|
var x;
|
|
|
|
level++;
|
|
|
|
callback(this, index, parent, level);
|
|
|
|
if (!Array.isArray(this.content)) return;
|
|
|
|
for (var i = 0, l = this.content.length; i < l; i++) {
|
|
breakLoop = this.content[i].traverse(callback, i, level, this);
|
|
if (breakLoop === null) break;
|
|
|
|
// If some nodes were removed or added:
|
|
if (x = this.content.length - l) {
|
|
l += x;
|
|
i += x;
|
|
}
|
|
}
|
|
|
|
if (breakLoop === null) return null;
|
|
};
|
|
|
|
Node.prototype.traverseByType = function traverseByType(type, callback) {
|
|
this.traverse(function (node) {
|
|
if (node.type === type) callback.apply(node, arguments);
|
|
});
|
|
};
|
|
|
|
Node.prototype.traverseByTypes = function traverseByTypes(types, callback) {
|
|
this.traverse(function (node) {
|
|
if (types.indexOf(node.type) !== -1) callback.apply(node, arguments);
|
|
});
|
|
};
|
|
|
|
_createClass(Node, [{
|
|
key: 'length',
|
|
get: function get() {
|
|
if (!Array.isArray(this.content)) return 0;
|
|
return this.content.length;
|
|
}
|
|
}]);
|
|
|
|
return Node;
|
|
}();
|
|
|
|
module.exports = Node;
|
|
|
|
/***/ }),
|
|
/* 2 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
var map = {
|
|
"./css/stringify": 3,
|
|
"./less/stringify": 4,
|
|
"./sass/stringify": 5,
|
|
"./scss/stringify": 6
|
|
};
|
|
function webpackContext(req) {
|
|
return __webpack_require__(webpackContextResolve(req));
|
|
};
|
|
function webpackContextResolve(req) {
|
|
return map[req] || (function() { throw new Error("Cannot find module '" + req + "'.") }());
|
|
};
|
|
webpackContext.keys = function webpackContextKeys() {
|
|
return Object.keys(map);
|
|
};
|
|
webpackContext.resolve = webpackContextResolve;
|
|
module.exports = webpackContext;
|
|
webpackContext.id = 2;
|
|
|
|
|
|
/***/ }),
|
|
/* 3 */
|
|
/***/ (function(module, exports) {
|
|
|
|
'use strict';
|
|
|
|
module.exports = function stringify(tree) {
|
|
// TODO: Better error message
|
|
if (!tree) throw new Error('We need tree to translate');
|
|
|
|
function _t(tree) {
|
|
var type = tree.type;
|
|
if (_unique[type]) return _unique[type](tree);
|
|
if (typeof tree.content === 'string') return tree.content;
|
|
if (Array.isArray(tree.content)) return _composite(tree.content);
|
|
return '';
|
|
}
|
|
|
|
function _composite(t, i) {
|
|
if (!t) return '';
|
|
|
|
var s = '';
|
|
i = i || 0;
|
|
for (; i < t.length; i++) {
|
|
s += _t(t[i]);
|
|
}return s;
|
|
}
|
|
|
|
var _unique = {
|
|
'arguments': function _arguments(t) {
|
|
return '(' + _composite(t.content) + ')';
|
|
},
|
|
'atkeyword': function atkeyword(t) {
|
|
return '@' + _composite(t.content);
|
|
},
|
|
'attributeSelector': function attributeSelector(t) {
|
|
return '[' + _composite(t.content) + ']';
|
|
},
|
|
'block': function block(t) {
|
|
return '{' + _composite(t.content) + '}';
|
|
},
|
|
'brackets': function brackets(t) {
|
|
return '[' + _composite(t.content) + ']';
|
|
},
|
|
'class': function _class(t) {
|
|
return '.' + _composite(t.content);
|
|
},
|
|
'color': function color(t) {
|
|
return '#' + t.content;
|
|
},
|
|
'customProperty': function customProperty(t) {
|
|
return '--' + t.content;
|
|
},
|
|
'expression': function expression(t) {
|
|
return 'expression(' + t.content + ')';
|
|
},
|
|
'id': function id(t) {
|
|
return '#' + _composite(t.content);
|
|
},
|
|
'multilineComment': function multilineComment(t) {
|
|
return '/*' + t.content + '*/';
|
|
},
|
|
'nthSelector': function nthSelector(t) {
|
|
return ':' + _t(t.content[0]) + '(' + _composite(t.content.slice(1)) + ')';
|
|
},
|
|
'parentheses': function parentheses(t) {
|
|
return '(' + _composite(t.content) + ')';
|
|
},
|
|
'percentage': function percentage(t) {
|
|
return _composite(t.content) + '%';
|
|
},
|
|
'pseudoClass': function pseudoClass(t) {
|
|
return ':' + _composite(t.content);
|
|
},
|
|
'pseudoElement': function pseudoElement(t) {
|
|
return '::' + _composite(t.content);
|
|
},
|
|
'universalSelector': function universalSelector(t) {
|
|
return _composite(t.content) + '*';
|
|
},
|
|
'uri': function uri(t) {
|
|
return 'url(' + _composite(t.content) + ')';
|
|
}
|
|
};
|
|
|
|
return _t(tree);
|
|
};
|
|
|
|
/***/ }),
|
|
/* 4 */
|
|
/***/ (function(module, exports) {
|
|
|
|
'use strict';
|
|
|
|
module.exports = function stringify(tree) {
|
|
// TODO: Better error message
|
|
if (!tree) throw new Error('We need tree to translate');
|
|
|
|
function _t(tree) {
|
|
var type = tree.type;
|
|
if (_unique[type]) return _unique[type](tree);
|
|
if (typeof tree.content === 'string') return tree.content;
|
|
if (Array.isArray(tree.content)) return _composite(tree.content);
|
|
return '';
|
|
}
|
|
|
|
function _composite(t, i) {
|
|
if (!t) return '';
|
|
|
|
var s = '';
|
|
i = i || 0;
|
|
for (; i < t.length; i++) {
|
|
s += _t(t[i]);
|
|
}return s;
|
|
}
|
|
|
|
var _unique = {
|
|
'arguments': function _arguments(t) {
|
|
return '(' + _composite(t.content) + ')';
|
|
},
|
|
'atkeyword': function atkeyword(t) {
|
|
return '@' + _composite(t.content);
|
|
},
|
|
'attributeSelector': function attributeSelector(t) {
|
|
return '[' + _composite(t.content) + ']';
|
|
},
|
|
'block': function block(t) {
|
|
return '{' + _composite(t.content) + '}';
|
|
},
|
|
'brackets': function brackets(t) {
|
|
return '[' + _composite(t.content) + ']';
|
|
},
|
|
'class': function _class(t) {
|
|
return '.' + _composite(t.content);
|
|
},
|
|
'color': function color(t) {
|
|
return '#' + t.content;
|
|
},
|
|
'escapedString': function escapedString(t) {
|
|
return '~' + t.content;
|
|
},
|
|
'expression': function expression(t) {
|
|
return 'expression(' + t.content + ')';
|
|
},
|
|
'id': function id(t) {
|
|
return '#' + _composite(t.content);
|
|
},
|
|
'interpolatedVariable': function interpolatedVariable(t) {
|
|
return '@{' + _composite(t.content) + '}';
|
|
},
|
|
'multilineComment': function multilineComment(t) {
|
|
return '/*' + t.content + '*/';
|
|
},
|
|
'nthSelector': function nthSelector(t) {
|
|
return ':' + _t(t.content[0]) + '(' + _composite(t.content.slice(1)) + ')';
|
|
},
|
|
'parentheses': function parentheses(t) {
|
|
return '(' + _composite(t.content) + ')';
|
|
},
|
|
'percentage': function percentage(t) {
|
|
return _composite(t.content) + '%';
|
|
},
|
|
'pseudoClass': function pseudoClass(t) {
|
|
return ':' + _composite(t.content);
|
|
},
|
|
'pseudoElement': function pseudoElement(t) {
|
|
return '::' + _composite(t.content);
|
|
},
|
|
'singlelineComment': function singlelineComment(t) {
|
|
return '/' + '/' + t.content;
|
|
},
|
|
'universalSelector': function universalSelector(t) {
|
|
return _composite(t.content) + '*';
|
|
},
|
|
'uri': function uri(t) {
|
|
return 'url(' + _composite(t.content) + ')';
|
|
},
|
|
'variable': function variable(t) {
|
|
return '@' + _composite(t.content);
|
|
},
|
|
'variablesList': function variablesList(t) {
|
|
return _composite(t.content) + '...';
|
|
}
|
|
};
|
|
|
|
return _t(tree);
|
|
};
|
|
|
|
/***/ }),
|
|
/* 5 */
|
|
/***/ (function(module, exports) {
|
|
|
|
'use strict';
|
|
|
|
module.exports = function stringify(tree) {
|
|
// TODO: Better error message
|
|
if (!tree) throw new Error('We need tree to translate');
|
|
|
|
function _t(tree) {
|
|
var type = tree.type;
|
|
if (_unique[type]) return _unique[type](tree);
|
|
if (typeof tree.content === 'string') return tree.content;
|
|
if (Array.isArray(tree.content)) return _composite(tree.content);
|
|
return '';
|
|
}
|
|
|
|
function _composite(t, i) {
|
|
if (!t) return '';
|
|
|
|
var s = '';
|
|
i = i || 0;
|
|
for (; i < t.length; i++) {
|
|
s += _t(t[i]);
|
|
}return s;
|
|
}
|
|
|
|
var _unique = {
|
|
'arguments': function _arguments(t) {
|
|
return '(' + _composite(t.content) + ')';
|
|
},
|
|
'atkeyword': function atkeyword(t) {
|
|
return '@' + _composite(t.content);
|
|
},
|
|
'attributeSelector': function attributeSelector(t) {
|
|
return '[' + _composite(t.content) + ']';
|
|
},
|
|
'block': function block(t) {
|
|
return _composite(t.content);
|
|
},
|
|
'brackets': function brackets(t) {
|
|
return '[' + _composite(t.content) + ']';
|
|
},
|
|
'class': function _class(t) {
|
|
return '.' + _composite(t.content);
|
|
},
|
|
'color': function color(t) {
|
|
return '#' + t.content;
|
|
},
|
|
'customProperty': function customProperty(t) {
|
|
return '--' + t.content;
|
|
},
|
|
'expression': function expression(t) {
|
|
return 'expression(' + t.content + ')';
|
|
},
|
|
'functionsList': function functionsList(t) {
|
|
return _composite(t.content) + '...';
|
|
},
|
|
'id': function id(t) {
|
|
return '#' + _composite(t.content);
|
|
},
|
|
'interpolation': function interpolation(t) {
|
|
return '#{' + _composite(t.content) + '}';
|
|
},
|
|
'multilineComment': function multilineComment(t) {
|
|
var lines = t.content.split('\n');
|
|
var close = '';
|
|
|
|
if (lines.length > 1) {
|
|
var lastLine = lines[lines.length - 1];
|
|
if (lastLine.length < t.end.column) {
|
|
close = '*/';
|
|
}
|
|
} else if (t.content.length + 4 === t.end.column - t.start.column + 1) {
|
|
close = '*/';
|
|
}
|
|
|
|
return '/*' + t.content + close;
|
|
},
|
|
'nthSelector': function nthSelector(t) {
|
|
return ':' + _t(t.content[0]) + '(' + _composite(t.content.slice(1)) + ')';
|
|
},
|
|
'parentheses': function parentheses(t) {
|
|
return '(' + _composite(t.content) + ')';
|
|
},
|
|
'percentage': function percentage(t) {
|
|
return _composite(t.content) + '%';
|
|
},
|
|
'placeholder': function placeholder(t) {
|
|
return '%' + _composite(t.content);
|
|
},
|
|
'pseudoClass': function pseudoClass(t) {
|
|
return ':' + _composite(t.content);
|
|
},
|
|
'pseudoElement': function pseudoElement(t) {
|
|
return '::' + _composite(t.content);
|
|
},
|
|
'singlelineComment': function singlelineComment(t) {
|
|
return '/' + '/' + t.content;
|
|
},
|
|
'universalSelector': function universalSelector(t) {
|
|
return _composite(t.content) + '*';
|
|
},
|
|
'uri': function uri(t) {
|
|
return 'url(' + _composite(t.content) + ')';
|
|
},
|
|
'variable': function variable(t) {
|
|
return '$' + _composite(t.content);
|
|
},
|
|
'variablesList': function variablesList(t) {
|
|
return _composite(t.content) + '...';
|
|
}
|
|
};
|
|
|
|
return _t(tree);
|
|
};
|
|
|
|
/***/ }),
|
|
/* 6 */
|
|
/***/ (function(module, exports) {
|
|
|
|
'use strict';
|
|
|
|
module.exports = function stringify(tree) {
|
|
// TODO: Better error message
|
|
if (!tree) throw new Error('We need tree to translate');
|
|
|
|
function _t(tree) {
|
|
var type = tree.type;
|
|
if (_unique[type]) return _unique[type](tree);
|
|
if (typeof tree.content === 'string') return tree.content;
|
|
if (Array.isArray(tree.content)) return _composite(tree.content);
|
|
return '';
|
|
}
|
|
|
|
function _composite(t, i) {
|
|
if (!t) return '';
|
|
|
|
var s = '';
|
|
i = i || 0;
|
|
for (; i < t.length; i++) {
|
|
s += _t(t[i]);
|
|
}return s;
|
|
}
|
|
|
|
var _unique = {
|
|
'arguments': function _arguments(t) {
|
|
return '(' + _composite(t.content) + ')';
|
|
},
|
|
'atkeyword': function atkeyword(t) {
|
|
return '@' + _composite(t.content);
|
|
},
|
|
'attributeSelector': function attributeSelector(t) {
|
|
return '[' + _composite(t.content) + ']';
|
|
},
|
|
'block': function block(t) {
|
|
return '{' + _composite(t.content) + '}';
|
|
},
|
|
'brackets': function brackets(t) {
|
|
return '[' + _composite(t.content) + ']';
|
|
},
|
|
'class': function _class(t) {
|
|
return '.' + _composite(t.content);
|
|
},
|
|
'color': function color(t) {
|
|
return '#' + t.content;
|
|
},
|
|
'customProperty': function customProperty(t) {
|
|
return '--' + t.content;
|
|
},
|
|
'expression': function expression(t) {
|
|
return 'expression(' + t.content + ')';
|
|
},
|
|
'functionsList': function functionsList(t) {
|
|
return _composite(t.content) + '...';
|
|
},
|
|
'id': function id(t) {
|
|
return '#' + _composite(t.content);
|
|
},
|
|
'interpolation': function interpolation(t) {
|
|
return '#{' + _composite(t.content) + '}';
|
|
},
|
|
'multilineComment': function multilineComment(t) {
|
|
return '/*' + t.content + '*/';
|
|
},
|
|
'nthSelector': function nthSelector(t) {
|
|
return ':' + _t(t.content[0]) + '(' + _composite(t.content.slice(1)) + ')';
|
|
},
|
|
'parentheses': function parentheses(t) {
|
|
return '(' + _composite(t.content) + ')';
|
|
},
|
|
'percentage': function percentage(t) {
|
|
return _composite(t.content) + '%';
|
|
},
|
|
'placeholder': function placeholder(t) {
|
|
return '%' + _composite(t.content);
|
|
},
|
|
'pseudoClass': function pseudoClass(t) {
|
|
return ':' + _composite(t.content);
|
|
},
|
|
'pseudoElement': function pseudoElement(t) {
|
|
return '::' + _composite(t.content);
|
|
},
|
|
'singlelineComment': function singlelineComment(t) {
|
|
return '/' + '/' + t.content;
|
|
},
|
|
'universalSelector': function universalSelector(t) {
|
|
return _composite(t.content) + '*';
|
|
},
|
|
'uri': function uri(t) {
|
|
return 'url(' + _composite(t.content) + ')';
|
|
},
|
|
'variable': function variable(t) {
|
|
return '$' + _composite(t.content);
|
|
},
|
|
'variablesList': function variablesList(t) {
|
|
return _composite(t.content) + '...';
|
|
}
|
|
};
|
|
|
|
return _t(tree);
|
|
};
|
|
|
|
/***/ }),
|
|
/* 7 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var ParsingError = __webpack_require__(8);
|
|
var syntaxes = __webpack_require__(10);
|
|
|
|
var isInteger = Number.isInteger || function (value) {
|
|
return typeof value === 'number' && Math.floor(value) === value;
|
|
};
|
|
|
|
/**
|
|
* @param {String} css
|
|
* @param {Object} options
|
|
* @return {Object} AST
|
|
*/
|
|
function parser(css, options) {
|
|
if (typeof css !== 'string') throw new Error('Please, pass a string to parse');else if (!css) return __webpack_require__(29)();
|
|
|
|
var syntax = options && options.syntax || 'css';
|
|
var context = options && options.context || 'stylesheet';
|
|
var tabSize = options && options.tabSize;
|
|
if (!isInteger(tabSize) || tabSize < 1) tabSize = 1;
|
|
|
|
var syntaxParser = syntaxes[syntax];
|
|
|
|
if (!syntaxParser) {
|
|
var message = 'Syntax "' + syntax + '" is not supported yet, sorry';
|
|
return console.error(message);
|
|
}
|
|
|
|
var getTokens = syntaxParser.tokenizer;
|
|
var mark = syntaxParser.mark;
|
|
var parse = syntaxParser.parse;
|
|
|
|
var tokens = getTokens(css, tabSize);
|
|
mark(tokens);
|
|
|
|
var ast;
|
|
try {
|
|
ast = parse(tokens, context);
|
|
} catch (e) {
|
|
if (!e.syntax) throw e;
|
|
throw new ParsingError(e, css);
|
|
}
|
|
|
|
return ast;
|
|
}
|
|
|
|
module.exports = parser;
|
|
|
|
/***/ }),
|
|
/* 8 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var parserPackage = __webpack_require__(9);
|
|
|
|
/**
|
|
* @param {Error} e
|
|
* @param {String} css
|
|
*/
|
|
function ParsingError(e, css) {
|
|
this.line = e.line;
|
|
this.syntax = e.syntax;
|
|
this.css_ = css;
|
|
}
|
|
|
|
ParsingError.prototype = {
|
|
/**
|
|
* @type {String}
|
|
* @private
|
|
*/
|
|
customMessage_: '',
|
|
|
|
/**
|
|
* @type {Number}
|
|
*/
|
|
line: null,
|
|
|
|
/**
|
|
* @type {String}
|
|
*/
|
|
name: 'Parsing error',
|
|
|
|
/**
|
|
* @type {String}
|
|
*/
|
|
syntax: null,
|
|
|
|
/**
|
|
* @type {String}
|
|
*/
|
|
version: parserPackage.version,
|
|
|
|
/**
|
|
* @type {String}
|
|
*/
|
|
get context() {
|
|
var LINES_AROUND = 2;
|
|
|
|
var result = [];
|
|
var currentLineNumber = this.line;
|
|
var start = currentLineNumber - 1 - LINES_AROUND;
|
|
var end = currentLineNumber + LINES_AROUND;
|
|
var lines = this.css_.split(/\r\n|\r|\n/);
|
|
|
|
for (var i = start; i < end; i++) {
|
|
var line = lines[i];
|
|
if (!line) continue;
|
|
var ln = i + 1;
|
|
var mark = ln === currentLineNumber ? '*' : ' ';
|
|
result.push(ln + mark + '| ' + line);
|
|
}
|
|
|
|
return result.join('\n');
|
|
},
|
|
|
|
/**
|
|
* @type {String}
|
|
*/
|
|
get message() {
|
|
if (this.customMessage_) {
|
|
return this.customMessage_;
|
|
} else {
|
|
var message = 'Please check validity of the block';
|
|
if (typeof this.line === 'number') message += ' starting from line #' + this.line;
|
|
return message;
|
|
}
|
|
},
|
|
|
|
set message(message) {
|
|
this.customMessage_ = message;
|
|
},
|
|
|
|
/**
|
|
* @return {String}
|
|
*/
|
|
toString: function toString() {
|
|
return [this.name + ': ' + this.message, '', this.context, '', 'Syntax: ' + this.syntax, 'Gonzales PE version: ' + this.version].join('\n');
|
|
}
|
|
};
|
|
|
|
module.exports = ParsingError;
|
|
|
|
/***/ }),
|
|
/* 9 */
|
|
/***/ (function(module, exports) {
|
|
|
|
module.exports = {"name":"gonzales-pe","description":"Gonzales Preprocessor Edition (fast CSS parser)","version":"4.2.4","homepage":"http://github.com/tonyganch/gonzales-pe","bugs":"http://github.com/tonyganch/gonzales-pe/issues","license":"MIT","author":{"name":"Tony Ganch","email":"tonyganch+github@gmail.com","url":"http://tonyganch.com"},"main":"./lib/gonzales","repository":{"type":"git","url":"http://github.com/tonyganch/gonzales-pe.git"},"scripts":{"autofix-tests":"bash ./scripts/build.sh && bash ./scripts/autofix-tests.sh","build":"bash ./scripts/build.sh","init":"bash ./scripts/init.sh","lint":"bash ./scripts/lint.sh","log":"bash ./scripts/log.sh","prepublishOnly":"bash ./scripts/build.sh","test":"bash ./scripts/test.sh","watch":"bash ./scripts/watch.sh"},"bin":{"gonzales":"./bin/gonzales.js"},"dependencies":{"minimist":"1.1.x"},"devDependencies":{"babel-core":"^6.18.2","babel-loader":"^6.2.7","babel-plugin-add-module-exports":"^0.2.1","babel-preset-es2015":"^6.18.0","coffee-script":"~1.7.1","eslint":"^3.0.0","jscs":"2.1.0","jshint":"2.10.2","json-loader":"^0.5.3","mocha":"2.2.x","webpack":"^1.12.2","webpack-closure-compiler":"^2.0.2"},"engines":{"node":">=0.6.0"},"files":["MIT-LICENSE.txt","bin/gonzales.js","lib/gonzales.js"]}
|
|
|
|
/***/ }),
|
|
/* 10 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
module.exports = {
|
|
css: __webpack_require__(11),
|
|
less: __webpack_require__(17),
|
|
sass: __webpack_require__(21),
|
|
scss: __webpack_require__(25)
|
|
};
|
|
|
|
/***/ }),
|
|
/* 11 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
exports.__esModule = true;
|
|
exports.default = {
|
|
mark: __webpack_require__(12),
|
|
parse: __webpack_require__(14),
|
|
stringify: __webpack_require__(3),
|
|
tokenizer: __webpack_require__(16)
|
|
};
|
|
module.exports = exports['default'];
|
|
|
|
/***/ }),
|
|
/* 12 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
/**
|
|
* Mark whitespaces and comments
|
|
* @param {Array} tokens
|
|
*/
|
|
function markSpacesAndComments(tokens) {
|
|
var tokensLength = tokens.length;
|
|
var spaces = [-1, -1];
|
|
var type; // Current token's type
|
|
|
|
// For every token in the token list, mark spaces and line breaks
|
|
// as spaces (set both `ws` and `sc` flags). Mark multiline comments
|
|
// with `sc` flag.
|
|
// If there are several spaces or tabs or line breaks or multiline
|
|
// comments in a row, group them: take the last one's index number
|
|
// and save it to the first token in the group as a reference:
|
|
// e.g., `ws_last = 7` for a group of whitespaces or `sc_last = 9`
|
|
// for a group of whitespaces and comments.
|
|
for (var i = 0; i < tokensLength; i++) {
|
|
type = tokens[i].type;
|
|
|
|
if (type === TokenType.Space || type === TokenType.Tab || type === TokenType.Newline) {
|
|
markSpace(tokens, i, spaces);
|
|
} else if (type === TokenType.CommentML) {
|
|
markComment(tokens, i, spaces);
|
|
} else {
|
|
markEndOfSpacesAndComments(tokens, i, spaces);
|
|
}
|
|
}
|
|
|
|
markEndOfSpacesAndComments(tokens, i, spaces);
|
|
}
|
|
|
|
function markSpace(tokens, i, spaces) {
|
|
var token = tokens[i];
|
|
token.ws = true;
|
|
token.sc = true;
|
|
|
|
if (spaces[0] === -1) spaces[0] = i;
|
|
if (spaces[1] === -1) spaces[1] = i;
|
|
}
|
|
|
|
function markComment(tokens, i, spaces) {
|
|
var ws = spaces[0];
|
|
tokens[i].sc = true;
|
|
|
|
if (ws !== -1) {
|
|
tokens[ws].ws_last = i - 1;
|
|
spaces[0] = -1;
|
|
}
|
|
}
|
|
|
|
function markEndOfSpacesAndComments(tokens, i, spaces) {
|
|
var ws = spaces[0];
|
|
var sc = spaces[1];
|
|
if (ws !== -1) {
|
|
tokens[ws].ws_last = i - 1;
|
|
spaces[0] = -1;
|
|
}
|
|
if (sc !== -1) {
|
|
tokens[sc].sc_last = i - 1;
|
|
spaces[1] = -1;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Pair brackets
|
|
* @param {Array} tokens
|
|
*/
|
|
function markBrackets(tokens) {
|
|
var tokensLength = tokens.length;
|
|
var ps = []; // Parentheses
|
|
var sbs = []; // Square brackets
|
|
var cbs = []; // Curly brackets
|
|
var t = void 0; // Current token
|
|
|
|
// For every token in the token list, if we meet an opening (left)
|
|
// bracket, push its index number to a corresponding array.
|
|
// If we then meet a closing (right) bracket, look at the corresponding
|
|
// array. If there are any elements (records about previously met
|
|
// left brackets), take a token of the last left bracket (take
|
|
// the last index number from the array and find a token with
|
|
// this index number) and save right bracket's index as a reference:
|
|
for (var i = 0; i < tokensLength; i++) {
|
|
t = tokens[i];
|
|
var type = t.type;
|
|
|
|
if (type === TokenType.LeftParenthesis) {
|
|
ps.push(i);
|
|
} else if (type === TokenType.RightParenthesis) {
|
|
if (ps.length) {
|
|
t.left = ps.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
} else if (type === TokenType.LeftSquareBracket) {
|
|
sbs.push(i);
|
|
} else if (type === TokenType.RightSquareBracket) {
|
|
if (sbs.length) {
|
|
t.left = sbs.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
} else if (type === TokenType.LeftCurlyBracket) {
|
|
cbs.push(i);
|
|
} else if (type === TokenType.RightCurlyBracket) {
|
|
if (cbs.length) {
|
|
t.left = cbs.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {Array} tokens
|
|
*/
|
|
function markTokens(tokens) {
|
|
// Mark paired brackets:
|
|
markBrackets(tokens);
|
|
// Mark whitespaces and comments:
|
|
markSpacesAndComments(tokens);
|
|
}
|
|
|
|
module.exports = markTokens;
|
|
|
|
/***/ }),
|
|
/* 13 */
|
|
/***/ (function(module, exports) {
|
|
|
|
// jscs:disable
|
|
|
|
'use strict';
|
|
|
|
module.exports = {
|
|
StringSQ: 'StringSQ',
|
|
StringDQ: 'StringDQ',
|
|
CommentML: 'CommentML',
|
|
CommentSL: 'CommentSL',
|
|
|
|
Newline: 'Newline',
|
|
Space: 'Space',
|
|
Tab: 'Tab',
|
|
|
|
ExclamationMark: 'ExclamationMark', // !
|
|
QuotationMark: 'QuotationMark', // "
|
|
NumberSign: 'NumberSign', // #
|
|
DollarSign: 'DollarSign', // $
|
|
PercentSign: 'PercentSign', // %
|
|
Ampersand: 'Ampersand', // &
|
|
Apostrophe: 'Apostrophe', // '
|
|
LeftParenthesis: 'LeftParenthesis', // (
|
|
RightParenthesis: 'RightParenthesis', // )
|
|
Asterisk: 'Asterisk', // *
|
|
PlusSign: 'PlusSign', // +
|
|
Comma: 'Comma', // ,
|
|
HyphenMinus: 'HyphenMinus', // -
|
|
FullStop: 'FullStop', // .
|
|
Solidus: 'Solidus', // /
|
|
Colon: 'Colon', // :
|
|
Semicolon: 'Semicolon', // ;
|
|
LessThanSign: 'LessThanSign', // <
|
|
EqualsSign: 'EqualsSign', // =
|
|
EqualitySign: 'EqualitySign', // ==
|
|
InequalitySign: 'InequalitySign', // !=
|
|
GreaterThanSign: 'GreaterThanSign', // >
|
|
QuestionMark: 'QuestionMark', // ?
|
|
CommercialAt: 'CommercialAt', // @
|
|
LeftSquareBracket: 'LeftSquareBracket', // [
|
|
ReverseSolidus: 'ReverseSolidus', // \
|
|
RightSquareBracket: 'RightSquareBracket', // ]
|
|
CircumflexAccent: 'CircumflexAccent', // ^
|
|
LowLine: 'LowLine', // _
|
|
LeftCurlyBracket: 'LeftCurlyBracket', // {
|
|
VerticalLine: 'VerticalLine', // |
|
|
RightCurlyBracket: 'RightCurlyBracket', // }
|
|
Tilde: 'Tilde', // ~
|
|
|
|
Identifier: 'Identifier',
|
|
DecimalNumber: 'DecimalNumber'
|
|
};
|
|
|
|
/***/ }),
|
|
/* 14 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var Node = __webpack_require__(1);
|
|
var NodeType = __webpack_require__(15);
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
/**
|
|
* @type {Array}
|
|
*/
|
|
var tokens = void 0;
|
|
|
|
/**
|
|
* @type {Number}
|
|
*/
|
|
var tokensLength = void 0;
|
|
|
|
/**
|
|
* @type {Number}
|
|
*/
|
|
var pos = void 0;
|
|
|
|
var contexts = {
|
|
'atkeyword': function atkeyword() {
|
|
return checkAtkeyword(pos) && getAtkeyword();
|
|
},
|
|
'atrule': function atrule() {
|
|
return checkAtrule(pos) && getAtrule();
|
|
},
|
|
'attributeSelector': function attributeSelector() {
|
|
return checkAttributeSelector(pos) && getAttributeSelector();
|
|
},
|
|
'block': function block() {
|
|
return checkBlock(pos) && getBlock();
|
|
},
|
|
'brackets': function brackets() {
|
|
return checkBrackets(pos) && getBrackets();
|
|
},
|
|
'class': function _class() {
|
|
return checkClass(pos) && getClass();
|
|
},
|
|
'combinator': function combinator() {
|
|
return checkCombinator(pos) && getCombinator();
|
|
},
|
|
'commentML': function commentML() {
|
|
return checkCommentML(pos) && getCommentML();
|
|
},
|
|
'declaration': function declaration() {
|
|
return checkDeclaration(pos) && getDeclaration();
|
|
},
|
|
'declDelim': function declDelim() {
|
|
return checkDeclDelim(pos) && getDeclDelim();
|
|
},
|
|
'delim': function delim() {
|
|
return checkDelim(pos) && getDelim();
|
|
},
|
|
'dimension': function dimension() {
|
|
return checkDimension(pos) && getDimension();
|
|
},
|
|
'expression': function expression() {
|
|
return checkExpression(pos) && getExpression();
|
|
},
|
|
'function': function _function() {
|
|
return checkFunction(pos) && getFunction();
|
|
},
|
|
'ident': function ident() {
|
|
return checkIdent(pos) && getIdent();
|
|
},
|
|
'important': function important() {
|
|
return checkImportant(pos) && getImportant();
|
|
},
|
|
'namespace': function namespace() {
|
|
return checkNamespace(pos) && getNamespace();
|
|
},
|
|
'number': function number() {
|
|
return checkNumber(pos) && getNumber();
|
|
},
|
|
'operator': function operator() {
|
|
return checkOperator(pos) && getOperator();
|
|
},
|
|
'parentheses': function parentheses() {
|
|
return checkParentheses(pos) && getParentheses();
|
|
},
|
|
'percentage': function percentage() {
|
|
return checkPercentage(pos) && getPercentage();
|
|
},
|
|
'progid': function progid() {
|
|
return checkProgid(pos) && getProgid();
|
|
},
|
|
'property': function property() {
|
|
return checkProperty(pos) && getProperty();
|
|
},
|
|
'propertyDelim': function propertyDelim() {
|
|
return checkPropertyDelim(pos) && getPropertyDelim();
|
|
},
|
|
'pseudoc': function pseudoc() {
|
|
return checkPseudoc(pos) && getPseudoc();
|
|
},
|
|
'pseudoe': function pseudoe() {
|
|
return checkPseudoe(pos) && getPseudoe();
|
|
},
|
|
'ruleset': function ruleset() {
|
|
return checkRuleset(pos) && getRuleset();
|
|
},
|
|
's': function s() {
|
|
return checkS(pos) && getS();
|
|
},
|
|
'selector': function selector() {
|
|
return checkSelector(pos) && getSelector();
|
|
},
|
|
'shash': function shash() {
|
|
return checkShash(pos) && getShash();
|
|
},
|
|
'string': function string() {
|
|
return checkString(pos) && getString();
|
|
},
|
|
'stylesheet': function stylesheet() {
|
|
return checkStylesheet(pos) && getStylesheet();
|
|
},
|
|
'unary': function unary() {
|
|
return checkUnary(pos) && getUnary();
|
|
},
|
|
'unicodeRange': function unicodeRange() {
|
|
return checkUnicodeRange(pos) && getUnicodeRange();
|
|
},
|
|
'universalSelector': function universalSelector() {
|
|
return checkUniversalSelector(pos) && getUniversalSelector();
|
|
},
|
|
'urange': function urange() {
|
|
return checkUrange(pos) && getUrange();
|
|
},
|
|
'uri': function uri() {
|
|
return checkUri(pos) && getUri();
|
|
},
|
|
'value': function value() {
|
|
return checkValue(pos) && getValue();
|
|
},
|
|
'vhash': function vhash() {
|
|
return checkVhash(pos) && getVhash();
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Stop parsing and display error
|
|
* @param {Number=} i Token's index number
|
|
*/
|
|
function throwError(i) {
|
|
var ln = tokens[i].ln;
|
|
|
|
throw { line: ln, syntax: 'css' };
|
|
}
|
|
|
|
/**
|
|
* @param {Object} exclude
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkExcluding(exclude, i) {
|
|
var start = i;
|
|
|
|
while (i < tokensLength) {
|
|
if (exclude[tokens[i++].type]) break;
|
|
}
|
|
|
|
return i - start - 2;
|
|
}
|
|
|
|
/**
|
|
* @param {Number} start
|
|
* @param {Number} finish
|
|
* @return {String}
|
|
*/
|
|
function joinValues(start, finish) {
|
|
var s = '';
|
|
|
|
for (var i = start; i < finish + 1; i++) {
|
|
s += tokens[i].value;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* @param {Number} start
|
|
* @param {Number} num
|
|
* @return {String}
|
|
*/
|
|
function joinValues2(start, num) {
|
|
if (start + num - 1 >= tokensLength) return;
|
|
|
|
var s = '';
|
|
|
|
for (var i = 0; i < num; i++) {
|
|
s += tokens[start + i].value;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
function getLastPosition(content, line, column, colOffset) {
|
|
return typeof content === 'string' ? getLastPositionForString(content, line, column, colOffset) : getLastPositionForArray(content, line, column, colOffset);
|
|
}
|
|
|
|
function getLastPositionForString(content, line, column, colOffset) {
|
|
var position = [];
|
|
|
|
if (!content) {
|
|
position = [line, column];
|
|
if (colOffset) position[1] += colOffset - 1;
|
|
return position;
|
|
}
|
|
|
|
var lastLinebreak = content.lastIndexOf('\n');
|
|
var endsWithLinebreak = lastLinebreak === content.length - 1;
|
|
var splitContent = content.split('\n');
|
|
var linebreaksCount = splitContent.length - 1;
|
|
var prevLinebreak = linebreaksCount === 0 || linebreaksCount === 1 ? -1 : content.length - splitContent[linebreaksCount - 1].length - 2;
|
|
|
|
// Line:
|
|
var offset = endsWithLinebreak ? linebreaksCount - 1 : linebreaksCount;
|
|
position[0] = line + offset;
|
|
|
|
// Column:
|
|
if (endsWithLinebreak) {
|
|
offset = prevLinebreak !== -1 ? content.length - prevLinebreak : content.length - 1;
|
|
} else {
|
|
offset = linebreaksCount !== 0 ? content.length - lastLinebreak - column - 1 : content.length - 1;
|
|
}
|
|
position[1] = column + offset;
|
|
|
|
if (!colOffset) return position;
|
|
|
|
if (endsWithLinebreak) {
|
|
position[0]++;
|
|
position[1] = colOffset;
|
|
} else {
|
|
position[1] += colOffset;
|
|
}
|
|
|
|
return position;
|
|
}
|
|
|
|
function getLastPositionForArray(content, line, column, colOffset) {
|
|
var position = void 0;
|
|
|
|
if (content.length === 0) {
|
|
position = [line, column];
|
|
} else {
|
|
var c = content[content.length - 1];
|
|
if (c.hasOwnProperty('end')) {
|
|
position = [c.end.line, c.end.column];
|
|
} else {
|
|
position = getLastPosition(c.content, line, column);
|
|
}
|
|
}
|
|
|
|
if (!colOffset) return position;
|
|
|
|
if (tokens[pos - 1] && tokens[pos - 1].type !== 'Newline') {
|
|
position[1] += colOffset;
|
|
} else {
|
|
position[0]++;
|
|
position[1] = 1;
|
|
}
|
|
|
|
return position;
|
|
}
|
|
|
|
function newNode(type, content, line, column, end) {
|
|
if (!end) end = getLastPosition(content, line, column);
|
|
return new Node({
|
|
type: type,
|
|
content: content,
|
|
start: {
|
|
line: line,
|
|
column: column
|
|
},
|
|
end: {
|
|
line: end[0],
|
|
column: end[1]
|
|
},
|
|
syntax: 'css'
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkAny(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkBrackets(i)) tokens[i].any_child = 1;else if (l = checkParentheses(i)) tokens[i].any_child = 2;else if (l = checkString(i)) tokens[i].any_child = 3;else if (l = checkPercentage(i)) tokens[i].any_child = 4;else if (l = checkDimension(i)) tokens[i].any_child = 5;else if (l = checkUnicodeRange(i)) tokens[i].any_child = 13;else if (l = checkNumber(i)) tokens[i].any_child = 6;else if (l = checkUri(i)) tokens[i].any_child = 7;else if (l = checkExpression(i)) tokens[i].any_child = 8;else if (l = checkFunction(i)) tokens[i].any_child = 9;else if (l = checkIdent(i)) tokens[i].any_child = 10;else if (l = checkClass(i)) tokens[i].any_child = 11;else if (l = checkUnary(i)) tokens[i].any_child = 12;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getAny() {
|
|
var childType = tokens[pos].any_child;
|
|
|
|
if (childType === 1) return getBrackets();
|
|
if (childType === 2) return getParentheses();
|
|
if (childType === 3) return getString();
|
|
if (childType === 4) return getPercentage();
|
|
if (childType === 5) return getDimension();
|
|
if (childType === 13) return getUnicodeRange();
|
|
if (childType === 6) return getNumber();
|
|
if (childType === 7) return getUri();
|
|
if (childType === 8) return getExpression();
|
|
if (childType === 9) return getFunction();
|
|
if (childType === 10) return getIdent();
|
|
if (childType === 11) return getClass();
|
|
if (childType === 12) return getUnary();
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getArguments() {
|
|
var type = NodeType.ArgumentsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var body = void 0;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
while (pos < tokensLength && tokens[pos].type !== TokenType.RightParenthesis) {
|
|
if (checkDeclaration(pos)) content.push(getDeclaration());else if (checkArgument(pos)) {
|
|
body = getArgument();
|
|
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
|
|
} else if (checkClass(pos)) content.push(getClass());else throwError(pos);
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkArgument(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkVhash(i)) tokens[i].argument_child = 1;else if (l = checkCustomProperty(i)) tokens[i].argument_child = 2;else if (l = checkAny(i)) tokens[i].argument_child = 3;else if (l = checkSC(i)) tokens[i].argument_child = 4;else if (l = checkOperator(i)) tokens[i].argument_child = 5;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getArgument() {
|
|
var childType = tokens[pos].argument_child;
|
|
|
|
if (childType === 1) return getVhash();
|
|
if (childType === 2) return getCustomProperty();
|
|
if (childType === 3) return getAny();
|
|
if (childType === 4) return getSC();
|
|
if (childType === 5) return getOperator();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an @-word (e.g. `@import`, `@include`)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkAtkeyword(i) {
|
|
var l = void 0;
|
|
|
|
// Check that token is `@`:
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.CommercialAt) return 0;
|
|
|
|
return (l = checkIdent(i)) ? l + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with @-word
|
|
* @return {Node}
|
|
*/
|
|
function getAtkeyword() {
|
|
var type = NodeType.AtkeywordType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `@`.
|
|
pos++;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a part of an @-rule
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of @-rule
|
|
*/
|
|
function checkAtrule(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// If token already has a record of being part of an @-rule,
|
|
// return the @-rule's length:
|
|
if (tokens[i].atrule_l !== undefined) return tokens[i].atrule_l;
|
|
|
|
// If token is part of an @-rule, save the rule's type to token.
|
|
// @keyframes:
|
|
if (l = checkKeyframesRule(i)) tokens[i].atrule_type = 4;
|
|
// @-rule with ruleset:
|
|
else if (l = checkAtruler(i)) tokens[i].atrule_type = 1;
|
|
// Block @-rule:
|
|
else if (l = checkAtruleb(i)) tokens[i].atrule_type = 2;
|
|
// Single-line @-rule:
|
|
else if (l = checkAtrules(i)) tokens[i].atrule_type = 3;else return 0;
|
|
|
|
// If token is part of an @-rule, save the rule's length to token:
|
|
tokens[i].atrule_l = l;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* Get node with @-rule
|
|
* @return {Node}
|
|
*/
|
|
function getAtrule() {
|
|
var childType = tokens[pos].atrule_type;
|
|
|
|
if (childType === 1) return getAtruler(); // @-rule with ruleset
|
|
if (childType === 2) return getAtruleb(); // Block @-rule
|
|
if (childType === 3) return getAtrules(); // Single-line @-rule
|
|
if (childType === 4) return getKeyframesRule();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a block @-rule
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the @-rule
|
|
*/
|
|
function checkAtruleb(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a block @-rule
|
|
* @return {Node}
|
|
*/
|
|
function getAtruleb() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an @-rule with ruleset
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the @-rule
|
|
*/
|
|
function checkAtruler(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
|
|
|
|
if (l = checkAtrulers(i)) i += l;
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with an @-rule with ruleset
|
|
* @return {Node}
|
|
*/
|
|
function getAtruler() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets(), getAtrulers());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkAtrulers(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkSC(i)) tokens[i].atrulers_child = 1;else if (l = checkAtrule(i)) tokens[i].atrulers_child = 2;else if (l = checkRuleset(i)) tokens[i].atrulers_child = 3;else break;
|
|
i += l;
|
|
}
|
|
|
|
if (i < tokensLength) tokens[i].atrulers_end = 1;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getAtrulers() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `{`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC());
|
|
|
|
while (pos < tokensLength && !tokens[pos].atrulers_end) {
|
|
var childType = tokens[pos].atrulers_child;
|
|
if (childType === 1) content = content.concat(getSC());else if (childType === 2) content.push(getAtrule());else if (childType === 3) content.push(getRuleset());else break;
|
|
}
|
|
|
|
content = content.concat(getSC());
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `}`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkAtrules(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getAtrules() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a block (e.g. `{...}`).
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the block
|
|
*/
|
|
function checkBlock(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket ? tokens[i].right - i + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a block
|
|
* @return {Node}
|
|
*/
|
|
function getBlock() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = tokens[pos].right;
|
|
var content = [];
|
|
|
|
// Skip `{`.
|
|
pos++;
|
|
|
|
while (pos < end) {
|
|
if (checkBlockdecl(pos)) content = content.concat(getBlockdecl());else throwError(pos);
|
|
}
|
|
|
|
var end_ = getLastPosition(content, line, column, 1);
|
|
pos = end + 1;
|
|
|
|
return newNode(type, content, line, column, end_);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a declaration (property-value pair)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the declaration
|
|
*/
|
|
function checkBlockdecl(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkBlockdecl1(i)) tokens[i].bd_type = 1;else if (l = checkBlockdecl2(i)) tokens[i].bd_type = 2;else if (l = checkBlockdecl3(i)) tokens[i].bd_type = 3;else if (l = checkBlockdecl4(i)) tokens[i].bd_type = 4;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getBlockdecl() {
|
|
var childType = tokens[pos].bd_type;
|
|
|
|
if (childType === 1) return getBlockdecl1();
|
|
if (childType === 2) return getBlockdecl2();
|
|
if (childType === 3) return getBlockdecl3();
|
|
if (childType === 4) return getBlockdecl4();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkBlockdecl1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkDeclaration(i)) tokens[i].bd_kind = 1;else if (l = checkAtrule(i)) tokens[i].bd_kind = 2;else return 0;
|
|
|
|
i += l;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i < tokensLength && (l = checkDeclDelim(i))) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getBlockdecl1() {
|
|
var sc = getSC();
|
|
var content = void 0;
|
|
|
|
switch (tokens[pos].bd_kind) {
|
|
case 1:
|
|
content = getDeclaration();
|
|
break;
|
|
case 2:
|
|
content = getAtrule();
|
|
break;
|
|
}
|
|
|
|
return sc.concat(content, getSC(), getDeclDelim(), getSC());
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkBlockdecl2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkDeclaration(i)) tokens[i].bd_kind = 1;else if (l = checkAtrule(i)) tokens[i].bd_kind = 2;else return 0;
|
|
|
|
i += l;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getBlockdecl2() {
|
|
var sc = getSC();
|
|
var content = void 0;
|
|
|
|
switch (tokens[pos].bd_kind) {
|
|
case 1:
|
|
content = getDeclaration();
|
|
break;
|
|
case 2:
|
|
content = getAtrule();
|
|
break;
|
|
}
|
|
|
|
return sc.concat(content, getSC());
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkBlockdecl3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkDeclDelim(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getBlockdecl3() {
|
|
return [].concat(getSC(), getDeclDelim(), getSC());
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkBlockdecl4(i) {
|
|
return checkSC(i);
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getBlockdecl4() {
|
|
return getSC();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of text inside square brackets, e.g. `[1]`
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkBrackets(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
|
|
// Skip `[`.
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
if (i < tokens[start].right) {
|
|
var l = checkTsets(i);
|
|
if (l) i += l;else return 0;
|
|
}
|
|
|
|
// Skip `]`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with text inside square brackets, e.g. `[1]`
|
|
* @return {Node}
|
|
*/
|
|
function getBrackets() {
|
|
var type = NodeType.BracketsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var right = token.right;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
if (pos < right) {
|
|
content = getTsets();
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a class selector (e.g. `.abc`)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the class selector
|
|
*/
|
|
function checkClass(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].class_l) return tokens[i].class_l;
|
|
|
|
// Skip `.`.
|
|
if (tokens[i].type === TokenType.FullStop) i++;else return 0;
|
|
|
|
if (l = checkIdent(i)) {
|
|
tokens[start].class_l = l + 1;
|
|
i += l;
|
|
} else return 0;
|
|
|
|
tokens[start].classEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a class selector
|
|
* @return {Node}
|
|
*/
|
|
function getClass() {
|
|
var type = NodeType.ClassType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `.`
|
|
pos++;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkCombinator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkCombinator1(i)) tokens[i].combinatorType = 1;else if (l = checkCombinator2(i)) tokens[i].combinatorType = 2;else if (l = checkCombinator3(i)) tokens[i].combinatorType = 3;else if (l = checkCombinator4(i)) tokens[i].combinatorType = 4;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getCombinator() {
|
|
var type = tokens[pos].combinatorType;
|
|
if (type === 1) return getCombinator1();
|
|
if (type === 2) return getCombinator2();
|
|
if (type === 3) return getCombinator3();
|
|
if (type === 4) return getCombinator4();
|
|
}
|
|
|
|
/**
|
|
* (1) `>>>`
|
|
*
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkCombinator1(i) {
|
|
if (i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign) return 3;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator1() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '>>>';
|
|
|
|
// Skip combinator
|
|
pos += 3;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `||`
|
|
* (2) `>>`
|
|
*
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkCombinator2(i) {
|
|
if (i + 1 >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.VerticalLine && tokens[i + 1].type === TokenType.VerticalLine) return 2;
|
|
|
|
if (tokens[i].type === TokenType.GreaterThanSign && tokens[i + 1].type === TokenType.GreaterThanSign) return 2;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator2() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '' + token.value + tokens[pos + 1].value;
|
|
|
|
// Skip combinator
|
|
pos += 2;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `>`
|
|
* (2) `+`
|
|
* (3) `~`
|
|
*
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkCombinator3(i) {
|
|
var type = tokens[i].type;
|
|
if (type === TokenType.PlusSign || type === TokenType.GreaterThanSign || type === TokenType.Tilde) return 1;else return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator3() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
// Skip combinator
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `/panda/`
|
|
*/
|
|
function checkCombinator4(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator4() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `/`.
|
|
pos++;
|
|
|
|
var ident = getIdent();
|
|
|
|
// Skip `/`.
|
|
pos++;
|
|
|
|
var content = '/' + ident.content + '/';
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a multiline comment.
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} `1` if token is a multiline comment, otherwise `0`
|
|
*/
|
|
function checkCommentML(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.CommentML ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a multiline comment
|
|
* @return {Node}
|
|
*/
|
|
function getCommentML() {
|
|
var type = NodeType.CommentMLType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos].value.substring(2);
|
|
var l = content.length;
|
|
|
|
if (content.charAt(l - 2) === '*' && content.charAt(l - 1) === '/') content = content.substring(0, l - 2);
|
|
|
|
var end = getLastPosition(content, line, column, 2);
|
|
if (end[0] === line) end[1] += 2;
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a declaration (property-value pair)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the declaration
|
|
*/
|
|
function checkDeclaration(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkProperty(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkPropertyDelim(i)) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkValue(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a declaration
|
|
* @return {Node}
|
|
*/
|
|
function getDeclaration() {
|
|
var type = NodeType.DeclarationType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getProperty(), getSC(), getPropertyDelim(), getSC(), getValue());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a semicolon
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} `1` if token is a semicolon, otherwise `0`
|
|
*/
|
|
function checkDeclDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Semicolon ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a semicolon
|
|
* @return {Node}
|
|
*/
|
|
function getDeclDelim() {
|
|
var type = NodeType.DeclDelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ';';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a comma
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} `1` if token is a comma, otherwise `0`
|
|
*/
|
|
function checkDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Comma ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a comma
|
|
* @return {Node}
|
|
*/
|
|
function getDelim() {
|
|
var type = NodeType.DelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ',';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number with dimension unit (e.g. `10px`)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkDimension(i) {
|
|
var ln = checkNumber(i);
|
|
var li = void 0;
|
|
|
|
if (i >= tokensLength || !ln || i + ln >= tokensLength) return 0;
|
|
|
|
return (li = checkUnit(i + ln)) ? ln + li : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node of a number with dimension unit
|
|
* @return {Node}
|
|
*/
|
|
function getDimension() {
|
|
var type = NodeType.DimensionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getNumber(), getUnit()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkExpression(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength || tokens[i++].value !== 'expression' || i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) {
|
|
return 0;
|
|
}
|
|
|
|
return tokens[i].right - start + 1;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getExpression() {
|
|
var type = NodeType.ExpressionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
pos++;
|
|
|
|
var content = joinValues(pos + 1, tokens[pos].right - 1);
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
if (end[0] === line) end[1] += 11;
|
|
pos = tokens[pos].right + 1;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkFunction(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i < tokensLength && tokens[i].type === TokenType.LeftParenthesis ? tokens[i].right - start + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getFunction() {
|
|
var type = NodeType.FunctionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getIdent(), getArguments());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an identifierю
|
|
* Grammar from CSS spec:
|
|
* h [0-9a-f]
|
|
* nonascii [\240-\377]
|
|
* unicode \\{h}{1,6}(\r\n|[ \t\r\n\f])?
|
|
* escape {unicode}|\\[^\r\n\f0-9a-f]
|
|
* nmstart [_a-z]|{nonascii}|{escape}
|
|
* nmchar [_a-z0-9-]|{nonascii}|{escape}
|
|
* ident -?{nmstart}{nmchar}*
|
|
*
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the identifier
|
|
*/
|
|
function checkIdent(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.HyphenMinus) i++;
|
|
|
|
if (tokens[i].type === TokenType.LowLine || tokens[i].type === TokenType.Identifier) i++;else return 0;
|
|
|
|
for (; i < tokensLength; i++) {
|
|
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
|
|
}
|
|
|
|
tokens[start].ident_last = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with an identifier
|
|
* @return {Node}
|
|
*/
|
|
function getIdent() {
|
|
var type = NodeType.IdentType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, tokens[pos].ident_last);
|
|
|
|
pos = tokens[pos].ident_last + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of `!important` word
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkImportant(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].value === 'important') {
|
|
tokens[start].importantEnd = i;
|
|
return i - start + 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get node with `!important` word
|
|
* @return {Node}
|
|
*/
|
|
function getImportant() {
|
|
var type = NodeType.ImportantType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, token.importantEnd);
|
|
|
|
pos = token.importantEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check a single keyframe block - `5% {}`
|
|
* @param {Number} i
|
|
* @returns {Number}
|
|
*/
|
|
function checkKeyframesBlock(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkKeyframesSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a single keyframe block - `5% {}`
|
|
* @returns {Node}
|
|
*/
|
|
function getKeyframesBlock() {
|
|
var type = NodeType.RulesetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getKeyframesSelectorsGroup(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check all keyframe blocks - `5% {} 100% {}`
|
|
* @param {Number} i
|
|
* @returns {Number}
|
|
*/
|
|
function checkKeyframesBlocks(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlock(i)) i += l;else return 0;
|
|
|
|
while (tokens[i].type !== TokenType.RightCurlyBracket) {
|
|
if (l = checkSC(i)) i += l;else if (l = checkKeyframesBlock(i)) i += l;else break;
|
|
}
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get all keyframe blocks - `5% {} 100% {}`
|
|
* @returns {Node}
|
|
*/
|
|
function getKeyframesBlocks() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var keyframesBlocksEnd = token.right;
|
|
var content = [];
|
|
|
|
// Skip `{`.
|
|
pos++;
|
|
|
|
while (pos < keyframesBlocksEnd) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkKeyframesBlock(pos)) content.push(getKeyframesBlock());
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `}`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a @keyframes rule.
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the @keyframes rule
|
|
*/
|
|
function checkKeyframesRule(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
var atruleName = joinValues2(i - l, l);
|
|
if (atruleName.toLowerCase().indexOf('keyframes') === -1) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getKeyframesRule() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdent(), getSC(), getKeyframesBlocks());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check a single keyframe selector - `5%`, `from` etc
|
|
* @param {Number} i
|
|
* @returns {Number}
|
|
*/
|
|
function checkKeyframesSelector(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) {
|
|
// Valid selectors are only `from` and `to`.
|
|
var selector = joinValues2(i, l);
|
|
if (selector !== 'from' && selector !== 'to') return 0;
|
|
|
|
i += l;
|
|
tokens[start].keyframesSelectorType = 1;
|
|
} else if (l = checkPercentage(i)) {
|
|
i += l;
|
|
tokens[start].keyframesSelectorType = 2;
|
|
} else {
|
|
return 0;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a single keyframe selector
|
|
* @returns {Node}
|
|
*/
|
|
function getKeyframesSelector() {
|
|
var keyframesSelectorType = NodeType.KeyframesSelectorType;
|
|
var selectorType = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (token.keyframesSelectorType === 1) {
|
|
content.push(getIdent());
|
|
} else {
|
|
content.push(getPercentage());
|
|
}
|
|
|
|
var keyframesSelector = newNode(keyframesSelectorType, content, line, column);
|
|
|
|
return newNode(selectorType, [keyframesSelector], line, column);
|
|
}
|
|
|
|
/**
|
|
* Check the keyframe's selector groups
|
|
* @param {Number} i
|
|
* @returns {Number}
|
|
*/
|
|
function checkKeyframesSelectorsGroup(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkKeyframesSelector(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
var comma = checkDelim(i + spaceBefore);
|
|
if (!comma) break;
|
|
|
|
var spaceAfter = checkSC(i + spaceBefore + comma);
|
|
if (l = checkKeyframesSelector(i + spaceBefore + comma + spaceAfter)) {
|
|
i += spaceBefore + comma + spaceAfter + l;
|
|
} else break;
|
|
}
|
|
|
|
tokens[start].selectorsGroupEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get the keyframe's selector groups
|
|
* @returns {Array} An array of keyframe selectors
|
|
*/
|
|
function getKeyframesSelectorsGroup() {
|
|
var selectorsGroup = [];
|
|
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
|
|
|
|
selectorsGroup.push(getKeyframesSelector());
|
|
|
|
while (pos < selectorsGroupEnd) {
|
|
selectorsGroup = selectorsGroup.concat(getSC(), getDelim(), getSC(), getKeyframesSelector());
|
|
}
|
|
|
|
return selectorsGroup;
|
|
}
|
|
|
|
/**
|
|
* Check if token is a namespace sign (`|`)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} `1` if token is `|`, `0` if not
|
|
*/
|
|
function checkNamespace(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.VerticalLine ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a namespace sign
|
|
* @return {Node}
|
|
*/
|
|
function getNamespace() {
|
|
var type = NodeType.NamespaceType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '|';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkNmName2(i) {
|
|
if (tokens[i].type === TokenType.Identifier) return 1;else if (tokens[i].type !== TokenType.DecimalNumber) return 0;
|
|
|
|
i++;
|
|
|
|
return i < tokensLength && tokens[i].type === TokenType.Identifier ? 2 : 1;
|
|
}
|
|
|
|
/**
|
|
* @return {String}
|
|
*/
|
|
function getNmName2() {
|
|
var s = tokens[pos].value;
|
|
|
|
if (tokens[pos++].type === TokenType.DecimalNumber && pos < tokensLength && tokens[pos].type === TokenType.Identifier) s += tokens[pos++].value;
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of number
|
|
*/
|
|
function checkNumber(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].number_l) return tokens[i].number_l;
|
|
|
|
// `10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && (!tokens[i + 1] || tokens[i + 1] && tokens[i + 1].type !== TokenType.FullStop)) {
|
|
tokens[i].number_l = 1;
|
|
return 1;
|
|
}
|
|
|
|
// `10.`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && (!tokens[i + 2] || tokens[i + 2].type !== TokenType.DecimalNumber)) {
|
|
tokens[i].number_l = 2;
|
|
return 2;
|
|
}
|
|
|
|
// `.10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.FullStop && tokens[i + 1].type === TokenType.DecimalNumber) {
|
|
tokens[i].number_l = 2;
|
|
return 2;
|
|
}
|
|
|
|
// `10.10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && tokens[i + 2] && tokens[i + 2].type === TokenType.DecimalNumber) {
|
|
tokens[i].number_l = 3;
|
|
return 3;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with number
|
|
* @return {Node}
|
|
*/
|
|
function getNumber() {
|
|
var type = NodeType.NumberType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var l = tokens[pos].number_l;
|
|
var content = '';
|
|
|
|
for (var j = 0; j < l; j++) {
|
|
content += tokens[pos + j].value;
|
|
}
|
|
|
|
pos += l;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is an operator (`/`, `,`, `:`, `=`, `*`).
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} `1` if token is an operator, otherwise `0`
|
|
*/
|
|
function checkOperator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
switch (tokens[i].type) {
|
|
case TokenType.Solidus:
|
|
case TokenType.Comma:
|
|
case TokenType.Colon:
|
|
case TokenType.EqualsSign:
|
|
case TokenType.Asterisk:
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an operator
|
|
* @return {Node}
|
|
*/
|
|
function getOperator() {
|
|
var type = NodeType.OperatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of text inside parentheses, e.g. `(1)`
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkParentheses(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
if (tokens[i].type === TokenType.LeftParenthesis) i++;else return 0;
|
|
|
|
if (i < right) {
|
|
var l = checkTsets(i);
|
|
if (l) i += l;else return 0;
|
|
}
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with text inside parentheses, e.g. `(1)`
|
|
* @return {Node}
|
|
*/
|
|
function getParentheses() {
|
|
var type = NodeType.ParenthesesType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var right = token.right;
|
|
var content = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
if (pos < right) {
|
|
content = getTsets();
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number with percent sign (e.g. `10%`)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkPercentage(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkNumber(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Skip `%`.
|
|
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node of number with percent sign
|
|
* @return {Node}
|
|
*/
|
|
function getPercentage() {
|
|
var type = NodeType.PercentageType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getNumber()];
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `%`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkProgid(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (joinValues2(i, 6) === 'progid:DXImageTransform.Microsoft.') i += 6;else return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.LeftParenthesis) {
|
|
tokens[start].progid_end = tokens[i].right;
|
|
i = tokens[i].right + 1;
|
|
} else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getProgid() {
|
|
var type = NodeType.ProgidType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var progid_end = token.progid_end;
|
|
var content = joinValues(pos, progid_end);
|
|
|
|
pos = progid_end + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a property
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the property
|
|
*/
|
|
function checkProperty(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkProperty1(i)) tokens[start].propertyType = 1;else if (l = checkProperty2(i)) tokens[start].propertyType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* Get node with a property
|
|
* @return {Node}
|
|
*/
|
|
function getProperty() {
|
|
var type = tokens[pos].propertyType;
|
|
|
|
if (type === 1) return getProperty1();
|
|
if (type === 2) return getProperty2();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a property
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the property
|
|
*/
|
|
function checkProperty1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a property
|
|
* @return {Node}
|
|
*/
|
|
function getProperty1() {
|
|
var type = NodeType.PropertyType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a custom property
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the property
|
|
*/
|
|
function checkProperty2(i) {
|
|
return checkCustomProperty(i);
|
|
}
|
|
|
|
/**
|
|
* Get node with a custom property
|
|
* @return {Node}
|
|
*/
|
|
function getProperty2() {
|
|
return getCustomProperty();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a custom property
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the property
|
|
*/
|
|
function checkCustomProperty(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type !== TokenType.HyphenMinus || tokens[i + 1] && tokens[i + 1].type !== TokenType.HyphenMinus) return 0;
|
|
|
|
// Skip `--`
|
|
i += 2;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a custom property
|
|
* @return {Node}
|
|
*/
|
|
function getCustomProperty() {
|
|
var type = NodeType.CustomPropertyType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `--`
|
|
pos += 2;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a colon
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} `1` if token is a colon, otherwise `0`
|
|
*/
|
|
function checkPropertyDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Colon ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a colon
|
|
* @return {Node}
|
|
*/
|
|
function getPropertyDelim() {
|
|
var type = NodeType.PropertyDelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ':';
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkPseudo(i) {
|
|
return checkPseudoe(i) || checkPseudoc(i);
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getPseudo() {
|
|
if (checkPseudoe(pos)) return getPseudoe();
|
|
if (checkPseudoc(pos)) return getPseudoc();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkPseudoe(i) {
|
|
var l = void 0;
|
|
|
|
// Check `::`
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.Colon || i >= tokensLength || tokens[i + 1].type !== TokenType.Colon) return 0;
|
|
|
|
if (l = checkPseudoElement1(i)) tokens[i].pseudoElementType = 1;else if (l = checkPseudoElement2(i)) tokens[i].pseudoElementType = 2;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getPseudoe() {
|
|
var childType = tokens[pos].pseudoElementType;
|
|
if (childType === 1) return getPseudoElement1();
|
|
if (childType === 2) return getPseudoElement2();
|
|
}
|
|
|
|
/**
|
|
* (1) `::slotted(selector)`
|
|
* (2) `::slotted(selector, selector)`
|
|
*/
|
|
function checkPseudoElement1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `::`.
|
|
i += 2;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* (1) `::slotted(selector)`
|
|
* (2) `::slotted(selector, selector)`
|
|
*/
|
|
function getPseudoElement1() {
|
|
var type = NodeType.PseudoeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `::`.
|
|
pos += 2;
|
|
|
|
content.push(getIdent());
|
|
|
|
{
|
|
var _type = NodeType.ArgumentsType;
|
|
var _token = tokens[pos];
|
|
var _line = _token.ln;
|
|
var _column = _token.col;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
|
|
|
|
var end = getLastPosition(selectorContent, _line, _column, 1);
|
|
var args = newNode(_type, selectorContent, _line, _column, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkPseudoElement2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `::`.
|
|
i += 2;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getPseudoElement2() {
|
|
var type = NodeType.PseudoeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `::`.
|
|
pos += 2;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkPseudoc(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.Colon) return 0;
|
|
|
|
if (l = checkPseudoClass1(i)) tokens[i].pseudoClassType = 1;else if (l = checkPseudoClass2(i)) tokens[i].pseudoClassType = 2;else if (l = checkPseudoClass3(i)) tokens[i].pseudoClassType = 3;else if (l = checkPseudoClass4(i)) tokens[i].pseudoClassType = 4;else if (l = checkPseudoClass5(i)) tokens[i].pseudoClassType = 5;else if (l = checkPseudoClass6(i)) tokens[i].pseudoClassType = 6;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getPseudoc() {
|
|
var childType = tokens[pos].pseudoClassType;
|
|
if (childType === 1) return getPseudoClass1();
|
|
if (childType === 2) return getPseudoClass2();
|
|
if (childType === 3) return getPseudoClass3();
|
|
if (childType === 4) return getPseudoClass4();
|
|
if (childType === 5) return getPseudoClass5();
|
|
if (childType === 6) return getPseudoClass6();
|
|
}
|
|
|
|
/**
|
|
* (1) `:panda(selector)`
|
|
* (2) `:panda(selector, selector)`
|
|
*/
|
|
function checkPseudoClass1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* (-) `:not(panda)`
|
|
*/
|
|
function getPseudoClass1() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content.push(getIdent());
|
|
|
|
{
|
|
var _type2 = NodeType.ArgumentsType;
|
|
var _token2 = tokens[pos];
|
|
var _line2 = _token2.ln;
|
|
var _column2 = _token2.col;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
|
|
|
|
var end = getLastPosition(selectorContent, _line2, _column2, 1);
|
|
var args = newNode(_type2, selectorContent, _line2, _column2, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `:nth-child(odd)`
|
|
* (2) `:nth-child(even)`
|
|
* (3) `:lang(de-DE)`
|
|
*/
|
|
function checkPseudoClass2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass2() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content.push(getIdent());
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [].concat(getSC(), getIdent(), getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(-3n + 2)`
|
|
*/
|
|
function checkPseudoClass3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].value === 'n') i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.PlusSign || tokens[i].type === TokenType.HyphenMinus) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass3() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content.push(getIdent());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
|
|
{
|
|
var _l = tokens[pos].ln;
|
|
var _c = tokens[pos].col;
|
|
var _content = tokens[pos].value;
|
|
var ident = newNode(NodeType.IdentType, _content, _l, _c);
|
|
value.push(ident);
|
|
pos++;
|
|
}
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(-3n)`
|
|
*/
|
|
function checkPseudoClass4(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;
|
|
|
|
if (tokens[i].value === 'n') i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass4() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content.push(getIdent());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
if (checkIdent(pos)) value.push(getIdent());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(+8)`
|
|
*/
|
|
function checkPseudoClass5(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass5() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content.push(getIdent());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:checked`
|
|
*/
|
|
function checkPseudoClass6(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass6() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkRuleset(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getRuleset() {
|
|
var type = NodeType.RulesetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getSelectorsGroup(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is marked as a space (if it's a space or a tab
|
|
* or a line break).
|
|
* @param {Number} i
|
|
* @return {Number} Number of spaces in a row starting with the given token.
|
|
*/
|
|
function checkS(i) {
|
|
return i < tokensLength && tokens[i].ws ? tokens[i].ws_last - i + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with spaces
|
|
* @return {Node}
|
|
*/
|
|
function getS() {
|
|
var type = NodeType.SType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, tokens[pos].ws_last);
|
|
|
|
pos = tokens[pos].ws_last + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a space or a comment.
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Number of similar (space or comment) tokens
|
|
* in a row starting with the given token.
|
|
*/
|
|
function checkSC(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
var lsc = 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkS(i)) tokens[i].sc_child = 1;else if (l = checkCommentML(i)) tokens[i].sc_child = 2;else break;
|
|
|
|
i += l;
|
|
lsc += l;
|
|
}
|
|
|
|
return lsc || 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with spaces and comments
|
|
* @return {Array}
|
|
*/
|
|
function getSC() {
|
|
var sc = [];
|
|
|
|
if (pos >= tokensLength) return sc;
|
|
|
|
while (pos < tokensLength) {
|
|
var childType = tokens[pos].sc_child;
|
|
|
|
if (childType === 1) sc.push(getS());else if (childType === 2) sc.push(getCommentML());else break;
|
|
}
|
|
|
|
return sc;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
|
|
* a simple selector
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkShash(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a hexadecimal number (e.g. `#fff`) inside a simple
|
|
* selector
|
|
* @return {Node}
|
|
*/
|
|
function getShash() {
|
|
var type = NodeType.ShashType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `#`.
|
|
pos++;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a string (text wrapped in quotes)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} `1` if token is part of a string, `0` if not
|
|
*/
|
|
function checkString(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.StringSQ || tokens[i].type === TokenType.StringDQ) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get string's node
|
|
* @return {Array} `['string', x]` where `x` is a string (including
|
|
* quotes).
|
|
*/
|
|
function getString() {
|
|
var type = NodeType.StringType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Validate stylesheet: it should consist of any number (0 or more) of
|
|
* rulesets (sets of rules with selectors), @-rules, whitespaces or
|
|
* comments.
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkStylesheet(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkSC(i)) tokens[i].stylesheet_child = 1;else if (l = checkRuleset(i)) tokens[i].stylesheet_child = 2;else if (l = checkAtrule(i)) tokens[i].stylesheet_child = 3;else if (l = checkDeclDelim(i)) tokens[i].stylesheet_child = 4;else throwError(i);
|
|
|
|
i += l;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array} `['stylesheet', x]` where `x` is all stylesheet's
|
|
* nodes.
|
|
*/
|
|
function getStylesheet() {
|
|
var type = NodeType.StylesheetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
var childType = tokens[pos].stylesheet_child;
|
|
|
|
if (childType === 1) content = content.concat(getSC());
|
|
if (childType === 2) content.push(getRuleset());
|
|
if (childType === 3) content.push(getAtrule());
|
|
if (childType === 4) content.push(getDeclDelim());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkTset(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkVhash(i)) tokens[i].tset_child = 1;else if (l = checkAny(i)) tokens[i].tset_child = 2;else if (l = checkSC(i)) tokens[i].tset_child = 3;else if (l = checkOperator(i)) tokens[i].tset_child = 4;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getTset() {
|
|
var childType = tokens[pos].tset_child;
|
|
|
|
if (childType === 1) return getVhash();
|
|
if (childType === 2) return getAny();
|
|
if (childType === 3) return getSC();
|
|
if (childType === 4) return getOperator();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkTsets(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (l = checkTset(i)) {
|
|
i += l;
|
|
}
|
|
|
|
tokens[start].tsets_end = i;
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getTsets() {
|
|
var content = [];
|
|
var t = void 0;
|
|
|
|
if (pos >= tokensLength) return content;
|
|
|
|
var end = tokens[pos].tsets_end;
|
|
while (pos < end) {
|
|
t = getTset();
|
|
if (typeof t.content === 'string') content.push(t);else content = content.concat(t);
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is an unary (arithmetical) sign (`+` or `-`)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} `1` if token is an unary sign, `0` if not
|
|
*/
|
|
function checkUnary(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.HyphenMinus || tokens[i].type === TokenType.PlusSign) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an unary (arithmetical) sign (`+` or `-`)
|
|
* @return {Array} `['unary', x]` where `x` is an unary sign
|
|
* converted to string.
|
|
*/
|
|
function getUnary() {
|
|
var type = NodeType.OperatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a unicode range (single or multiple <urange> nodes)
|
|
* @param {number} i Token's index
|
|
* @return {number} Unicode range node's length
|
|
*/
|
|
function checkUnicodeRange(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkUrange(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
var comma = checkDelim(i + spaceBefore);
|
|
if (!comma) break;
|
|
|
|
var spaceAfter = checkSC(i + spaceBefore + comma);
|
|
if (l = checkUrange(i + spaceBefore + comma + spaceAfter)) {
|
|
i += spaceBefore + comma + spaceAfter + l;
|
|
} else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a unicode range node
|
|
* @return {Node}
|
|
*/
|
|
function getUnicodeRange() {
|
|
var type = NodeType.UnicodeRangeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkDelim(pos)) content.push(getDelim());else if (checkUrange(pos)) content.push(getUrange());else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is unit
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkUnit(i) {
|
|
var units = ['em', 'ex', 'ch', 'rem', 'vh', 'vw', 'vmin', 'vmax', 'px', 'mm', 'q', 'cm', 'in', 'pt', 'pc', 'deg', 'grad', 'rad', 'turn', 's', 'ms', 'Hz', 'kHz', 'dpi', 'dpcm', 'dppx'];
|
|
|
|
return units.indexOf(tokens[i].value) !== -1 ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get unit node of type ident
|
|
* @return {Node} An ident node containing the unit value
|
|
*/
|
|
function getUnit() {
|
|
var type = NodeType.IdentType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a u-range (part of a unicode-range)
|
|
* (1) `U+416`
|
|
* (2) `U+400-4ff`
|
|
* (3) `U+4??`
|
|
* @param {number} i Token's index
|
|
* @return {number} Urange node's length
|
|
*/
|
|
function checkUrange(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Check for unicode prefix (u+ or U+)
|
|
if (tokens[i].value === 'U' || tokens[i].value === 'u') i += 1;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].value === '+') i += 1;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = _checkUnicodeWildcard(i)) i += l;else break;
|
|
}
|
|
|
|
tokens[start].urangeEnd = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a u-range node (part of a unicode-range)
|
|
* @return {Node}
|
|
*/
|
|
function getUrange() {
|
|
var startPos = pos;
|
|
var type = NodeType.UrangeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content = joinValues(startPos, tokens[startPos].urangeEnd);
|
|
pos = tokens[startPos].urangeEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check for unicode wildcard characters `?`
|
|
* @param {number} i Token's index
|
|
* @return {number} Wildcard length
|
|
*/
|
|
function _checkUnicodeWildcard(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (tokens[i].type === TokenType.QuestionMark) i += 1;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of URI (e.g. `url('/css/styles.css')`)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of URI
|
|
*/
|
|
function checkUri(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength || tokens[i].value !== 'url') return 0;
|
|
|
|
// Skip `url`.
|
|
i++;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
return tokens[i].right - start + 1;
|
|
}
|
|
|
|
/**
|
|
* Get node with URI
|
|
* @return {Array} `['uri', x]` where `x` is URI's nodes (without `url`
|
|
* and braces, e.g. `['string', ''/css/styles.css'']`).
|
|
*/
|
|
function getUri() {
|
|
var startPos = pos;
|
|
var uriExcluding = {};
|
|
var uri = void 0;
|
|
var l = void 0;
|
|
var raw = void 0;
|
|
|
|
var rawContent = void 0;
|
|
var t = void 0;
|
|
|
|
pos += 2;
|
|
|
|
uriExcluding[TokenType.Space] = 1;
|
|
uriExcluding[TokenType.Tab] = 1;
|
|
uriExcluding[TokenType.Newline] = 1;
|
|
uriExcluding[TokenType.LeftParenthesis] = 1;
|
|
uriExcluding[TokenType.RightParenthesis] = 1;
|
|
|
|
if (checkUri1(pos)) {
|
|
uri = [].concat(getSC(), getString(), getSC());
|
|
} else {
|
|
uri = checkSC(pos) ? getSC() : [];
|
|
l = checkExcluding(uriExcluding, pos);
|
|
rawContent = joinValues(pos, pos + l);
|
|
t = tokens[pos];
|
|
raw = newNode(NodeType.RawType, rawContent, t.ln, t.col);
|
|
|
|
uri.push(raw);
|
|
|
|
pos += l + 1;
|
|
|
|
if (checkSC(pos)) uri = uri.concat(getSC());
|
|
}
|
|
|
|
t = tokens[startPos];
|
|
var line = t.ln;
|
|
var column = t.col;
|
|
var end = getLastPosition(uri, line, column, 1);
|
|
pos++;
|
|
|
|
return newNode(NodeType.UriType, uri, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkUri1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type !== TokenType.StringDQ && tokens[i].type !== TokenType.StringSQ) return 0;
|
|
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a value
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the value
|
|
*/
|
|
function checkValue(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
var s = void 0;
|
|
var _i = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
s = checkSC(i);
|
|
_i = i + s;
|
|
|
|
if (l = _checkValue(_i)) i += l + s;else break;
|
|
}
|
|
|
|
tokens[start].value_end = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getValue() {
|
|
var type = NodeType.ValueType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = tokens[pos].value_end;
|
|
var content = [];
|
|
|
|
while (pos < end) {
|
|
if (tokens[pos].value_child) content.push(_getValue());else content = content.concat(getSC());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function _checkValue(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkProgid(i)) tokens[i].value_child = 1;else if (l = checkVhash(i)) tokens[i].value_child = 2;else if (l = checkAny(i)) tokens[i].value_child = 3;else if (l = checkOperator(i)) tokens[i].value_child = 4;else if (l = checkImportant(i)) tokens[i].value_child = 5;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function _getValue() {
|
|
var childType = tokens[pos].value_child;
|
|
if (childType === 1) return getProgid();
|
|
if (childType === 2) return getVhash();
|
|
if (childType === 3) return getAny();
|
|
if (childType === 4) return getOperator();
|
|
if (childType === 5) return getImportant();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
|
|
* some value
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkVhash(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Skip `#`.
|
|
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
|
|
|
|
if (l = checkNmName2(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a hexadecimal number (e.g. `#fff`) inside some value
|
|
* @return {Array} `['vhash', x]` where `x` is a hexadecimal number
|
|
* converted to string (without `#`, e.g. `'fff'`).
|
|
*/
|
|
function getVhash() {
|
|
var type = NodeType.VhashType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `#`.
|
|
pos++;
|
|
|
|
var content = getNmName2();
|
|
var end = getLastPosition(content, line, column + 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkSelectorsGroup(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
var selectorCounter = 0;
|
|
var delimCounter = 0;
|
|
|
|
if (l = checkSelector(i)) {
|
|
i += l;
|
|
selectorCounter++;
|
|
} else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var tempStart = i;
|
|
var tempIndex = i;
|
|
var tempLength = void 0;
|
|
|
|
var spaceBefore = checkSC(tempIndex);
|
|
|
|
if (tempLength = checkDelim(tempIndex + spaceBefore)) {
|
|
tempIndex += spaceBefore + tempLength;
|
|
delimCounter++;
|
|
|
|
if (tempLength = checkSC(tempIndex)) tempIndex += tempLength;
|
|
if (tempLength = checkSelector(tempIndex)) {
|
|
tempIndex += tempLength;
|
|
selectorCounter++;
|
|
}
|
|
} else break;
|
|
|
|
i += tempIndex - tempStart;
|
|
}
|
|
|
|
tokens[start].selectorsGroupEnd = i;
|
|
tokens[start].selectorsGroupSelectorCount = selectorCounter;
|
|
tokens[start].selectorsGroupDelimCount = delimCounter;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getSelectorsGroup() {
|
|
var selectorsGroup = [];
|
|
var selectorCounter = 0;
|
|
var delimCounter = 0;
|
|
|
|
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
|
|
var selectorCount = tokens[pos].selectorsGroupSelectorCount;
|
|
var delimCount = tokens[pos].selectorsGroupDelimCount;
|
|
|
|
selectorsGroup.push(getSelector());
|
|
selectorCounter++;
|
|
|
|
while (pos < selectorsGroupEnd) {
|
|
if (delimCounter < delimCount) {
|
|
selectorsGroup = selectorsGroup.concat(getSC());
|
|
selectorsGroup = selectorsGroup.concat(getDelim());
|
|
delimCounter++;
|
|
|
|
selectorsGroup = selectorsGroup.concat(getSC());
|
|
|
|
if (selectorCounter < selectorCount) {
|
|
selectorsGroup = selectorsGroup.concat(getSelector());
|
|
selectorCounter++;
|
|
}
|
|
}
|
|
}
|
|
|
|
return selectorsGroup;
|
|
}
|
|
|
|
function checkSelector(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkCompoundSelector(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
var comma = checkCombinator(i + spaceBefore);
|
|
if (!spaceBefore && !comma) break;
|
|
|
|
var spaceAfter = checkSC(i + spaceBefore + comma);
|
|
if (l = checkCompoundSelector(i + spaceBefore + comma + spaceAfter)) {
|
|
i += spaceBefore + comma + spaceAfter + l;
|
|
} else break;
|
|
}
|
|
|
|
tokens[start].selectorEnd = i;
|
|
return i - start;
|
|
}
|
|
|
|
function getSelector() {
|
|
var type = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var selectorEnd = token.selectorEnd;
|
|
var content = getCompoundSelector();
|
|
|
|
while (pos < selectorEnd) {
|
|
content = content.concat(getSC());
|
|
if (checkCombinator(pos)) content.push(getCombinator());
|
|
content = content.concat(getSC(), getCompoundSelector());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkCompoundSelector(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkCompoundSelector1(i)) {
|
|
tokens[i].compoundSelectorType = 1;
|
|
} else if (l = checkCompoundSelector2(i)) {
|
|
tokens[i].compoundSelectorType = 2;
|
|
}
|
|
|
|
return l;
|
|
}
|
|
|
|
function getCompoundSelector() {
|
|
var type = tokens[pos].compoundSelectorType;
|
|
if (type === 1) return getCompoundSelector1();
|
|
if (type === 2) return getCompoundSelector2();
|
|
}
|
|
|
|
function checkCompoundSelector1(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkUniversalSelector(i) || checkTypeSelector(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var _l2 = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i);
|
|
if (_l2) i += _l2;else break;
|
|
}
|
|
|
|
tokens[start].compoundSelectorEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getCompoundSelector1() {
|
|
var sequence = [];
|
|
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
|
|
|
|
if (checkUniversalSelector(pos)) sequence.push(getUniversalSelector());else sequence.push(getTypeSelector());
|
|
|
|
while (pos < compoundSelectorEnd) {
|
|
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());
|
|
}
|
|
|
|
return sequence;
|
|
}
|
|
|
|
function checkCompoundSelector2(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
|
|
while (i < tokensLength) {
|
|
var l = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i);
|
|
|
|
if (l) i += l;else break;
|
|
}
|
|
|
|
tokens[start].compoundSelectorEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getCompoundSelector2() {
|
|
var sequence = [];
|
|
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
|
|
|
|
while (pos < compoundSelectorEnd) {
|
|
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());else break;
|
|
}
|
|
|
|
return sequence;
|
|
}
|
|
|
|
function checkUniversalSelector(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.Asterisk) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getUniversalSelector() {
|
|
var type = NodeType.UniversalSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var end = void 0;
|
|
|
|
if (checkNamePrefix(pos)) {
|
|
content.push(getNamePrefix());
|
|
end = getLastPosition(content, line, column, 1);
|
|
}
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkTypeSelector(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getTypeSelector() {
|
|
var type = NodeType.TypeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkNamePrefix(pos)) content.push(getNamePrefix());
|
|
|
|
content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeSelector(i) {
|
|
var l = void 0;
|
|
if (l = checkAttributeSelector1(i)) tokens[i].attributeSelectorType = 1;else if (l = checkAttributeSelector2(i)) tokens[i].attributeSelectorType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getAttributeSelector() {
|
|
var type = tokens[pos].attributeSelectorType;
|
|
if (type === 1) return getAttributeSelector1();else return getAttributeSelector2();
|
|
}
|
|
|
|
/**
|
|
* (1) `[panda=nani]`
|
|
* (2) `[panda='nani']`
|
|
* (3) `[panda='nani' i]`
|
|
*
|
|
*/
|
|
function checkAttributeSelector1(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeName(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeMatch(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeValue(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeFlags(i)) {
|
|
i += l;
|
|
if (l = checkSC(i)) i += l;
|
|
}
|
|
|
|
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeSelector1() {
|
|
var type = NodeType.AttributeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC(), getAttributeName(), getSC(), getAttributeMatch(), getSC(), getAttributeValue(), getSC());
|
|
|
|
if (checkAttributeFlags(pos)) {
|
|
content.push(getAttributeFlags());
|
|
content = content.concat(getSC());
|
|
}
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* (1) `[panda]`
|
|
*/
|
|
function checkAttributeSelector2(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeName(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeSelector2() {
|
|
var type = NodeType.AttributeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC(), getAttributeName(), getSC());
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkAttributeName(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeName() {
|
|
var type = NodeType.AttributeNameType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkNamePrefix(pos)) content.push(getNamePrefix());
|
|
content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeMatch(i) {
|
|
var l = void 0;
|
|
if (l = checkAttributeMatch1(i)) tokens[i].attributeMatchType = 1;else if (l = checkAttributeMatch2(i)) tokens[i].attributeMatchType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getAttributeMatch() {
|
|
var type = tokens[pos].attributeMatchType;
|
|
if (type === 1) return getAttributeMatch1();else return getAttributeMatch2();
|
|
}
|
|
|
|
function checkAttributeMatch1(i) {
|
|
var start = i;
|
|
|
|
var type = tokens[i].type;
|
|
if (type === TokenType.Tilde || type === TokenType.VerticalLine || type === TokenType.CircumflexAccent || type === TokenType.DollarSign || type === TokenType.Asterisk) i++;else return 0;
|
|
|
|
if (tokens[i].type === TokenType.EqualsSign) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeMatch1() {
|
|
var type = NodeType.AttributeMatchType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos].value + tokens[pos + 1].value;
|
|
pos += 2;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeMatch2(i) {
|
|
if (tokens[i].type === TokenType.EqualsSign) return 1;else return 0;
|
|
}
|
|
|
|
function getAttributeMatch2() {
|
|
var type = NodeType.AttributeMatchType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '=';
|
|
|
|
pos++;
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeValue(i) {
|
|
return checkString(i) || checkIdent(i);
|
|
}
|
|
|
|
function getAttributeValue() {
|
|
var type = NodeType.AttributeValueType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkString(pos)) content.push(getString());else content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeFlags(i) {
|
|
return checkIdent(i);
|
|
}
|
|
|
|
function getAttributeFlags() {
|
|
var type = NodeType.AttributeFlagsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkNamePrefix(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkNamePrefix1(i)) tokens[i].namePrefixType = 1;else if (l = checkNamePrefix2(i)) tokens[i].namePrefixType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getNamePrefix() {
|
|
var type = tokens[pos].namePrefixType;
|
|
if (type === 1) return getNamePrefix1();else return getNamePrefix2();
|
|
}
|
|
|
|
/**
|
|
* (1) `panda|`
|
|
* (2) `panda<comment>|`
|
|
*/
|
|
function checkNamePrefix1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamespacePrefix(i)) i += l;else return 0;
|
|
|
|
if (l = checkCommentML(i)) i += l;
|
|
|
|
if (l = checkNamespaceSeparator(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getNamePrefix1() {
|
|
var type = NodeType.NamePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(getNamespacePrefix());
|
|
|
|
if (checkCommentML(pos)) content.push(getCommentML());
|
|
|
|
content.push(getNamespaceSeparator());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `|`
|
|
*/
|
|
function checkNamePrefix2(i) {
|
|
return checkNamespaceSeparator(i);
|
|
}
|
|
|
|
function getNamePrefix2() {
|
|
var type = NodeType.NamePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getNamespaceSeparator()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `*`
|
|
* (2) `panda`
|
|
*/
|
|
function checkNamespacePrefix(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
|
|
if (tokens[i].type === TokenType.Asterisk) return 1;else if (l = checkIdent(i)) return l;else return 0;
|
|
}
|
|
|
|
function getNamespacePrefix() {
|
|
var type = NodeType.NamespacePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (token.type === TokenType.Asterisk) {
|
|
var asteriskNode = newNode(NodeType.IdentType, '*', line, column);
|
|
content.push(asteriskNode);
|
|
pos++;
|
|
} else if (checkIdent(pos)) content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `|`
|
|
*/
|
|
function checkNamespaceSeparator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type !== TokenType.VerticalLine) return 0;
|
|
|
|
// Return false if `|=` - [attr|=value]
|
|
if (tokens[i + 1] && tokens[i + 1].type === TokenType.EqualsSign) return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
function getNamespaceSeparator() {
|
|
var type = NodeType.NamespaceSeparatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '|';
|
|
|
|
pos++;
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
module.exports = function (_tokens, context) {
|
|
tokens = _tokens;
|
|
tokensLength = tokens.length;
|
|
pos = 0;
|
|
|
|
return contexts[context]();
|
|
};
|
|
|
|
/***/ }),
|
|
/* 15 */
|
|
/***/ (function(module, exports) {
|
|
|
|
'use strict';
|
|
|
|
module.exports = {
|
|
ArgumentsType: 'arguments',
|
|
AtkeywordType: 'atkeyword',
|
|
AtruleType: 'atrule',
|
|
AttributeSelectorType: 'attributeSelector',
|
|
AttributeNameType: 'attributeName',
|
|
AttributeFlagsType: 'attributeFlags',
|
|
AttributeMatchType: 'attributeMatch',
|
|
AttributeValueType: 'attributeValue',
|
|
BlockType: 'block',
|
|
BracketsType: 'brackets',
|
|
ClassType: 'class',
|
|
CombinatorType: 'combinator',
|
|
CommentMLType: 'multilineComment',
|
|
CommentSLType: 'singlelineComment',
|
|
ConditionType: 'condition',
|
|
ConditionalStatementType: 'conditionalStatement',
|
|
CustomPropertyType: 'customProperty',
|
|
DeclarationType: 'declaration',
|
|
DeclDelimType: 'declarationDelimiter',
|
|
DefaultType: 'default',
|
|
DelimType: 'delimiter',
|
|
DimensionType: 'dimension',
|
|
EscapedStringType: 'escapedString',
|
|
ExtendType: 'extend',
|
|
ExpressionType: 'expression',
|
|
FunctionType: 'function',
|
|
FunctionsListType: 'functionsList',
|
|
GlobalType: 'global',
|
|
IdentType: 'ident',
|
|
ImportantType: 'important',
|
|
IncludeType: 'include',
|
|
InterpolationType: 'interpolation',
|
|
InterpolatedVariableType: 'interpolatedVariable',
|
|
KeyframesSelectorType: 'keyframesSelector',
|
|
LoopType: 'loop',
|
|
MixinType: 'mixin',
|
|
NamePrefixType: 'namePrefix',
|
|
NamespacePrefixType: 'namespacePrefix',
|
|
NamespaceSeparatorType: 'namespaceSeparator',
|
|
NumberType: 'number',
|
|
OperatorType: 'operator',
|
|
OptionalType: 'optional',
|
|
ParenthesesType: 'parentheses',
|
|
ParentSelectorType: 'parentSelector',
|
|
ParentSelectorExtensionType: 'parentSelectorExtension',
|
|
PercentageType: 'percentage',
|
|
PlaceholderType: 'placeholder',
|
|
ProgidType: 'progid',
|
|
PropertyType: 'property',
|
|
PropertyDelimType: 'propertyDelimiter',
|
|
PseudocType: 'pseudoClass',
|
|
PseudoeType: 'pseudoElement',
|
|
RawType: 'raw',
|
|
RulesetType: 'ruleset',
|
|
SType: 'space',
|
|
SelectorType: 'selector',
|
|
ShashType: 'id',
|
|
StringType: 'string',
|
|
StylesheetType: 'stylesheet',
|
|
TypeSelectorType: 'typeSelector',
|
|
UnicodeRangeType: 'unicodeRange',
|
|
UniversalSelectorType: 'universalSelector',
|
|
UriType: 'uri',
|
|
UrangeType: 'urange',
|
|
ValueType: 'value',
|
|
VariableType: 'variable',
|
|
VariablesListType: 'variablesList',
|
|
VhashType: 'color'
|
|
};
|
|
|
|
/***/ }),
|
|
/* 16 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
module.exports = function (css, tabSize) {
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
var tokens = [];
|
|
var urlMode = false;
|
|
var blockMode = 0;
|
|
var pos = 0;
|
|
var tn = 0;
|
|
var ln = 1;
|
|
var col = 1;
|
|
var cssLength = 0;
|
|
|
|
var Punctuation = {
|
|
' ': TokenType.Space,
|
|
'\n': TokenType.Newline,
|
|
'\r': TokenType.Newline,
|
|
'\t': TokenType.Tab,
|
|
'!': TokenType.ExclamationMark,
|
|
'"': TokenType.QuotationMark,
|
|
'#': TokenType.NumberSign,
|
|
'$': TokenType.DollarSign,
|
|
'%': TokenType.PercentSign,
|
|
'&': TokenType.Ampersand,
|
|
'\'': TokenType.Apostrophe,
|
|
'(': TokenType.LeftParenthesis,
|
|
')': TokenType.RightParenthesis,
|
|
'*': TokenType.Asterisk,
|
|
'+': TokenType.PlusSign,
|
|
',': TokenType.Comma,
|
|
'-': TokenType.HyphenMinus,
|
|
'.': TokenType.FullStop,
|
|
'/': TokenType.Solidus,
|
|
':': TokenType.Colon,
|
|
';': TokenType.Semicolon,
|
|
'<': TokenType.LessThanSign,
|
|
'=': TokenType.EqualsSign,
|
|
'>': TokenType.GreaterThanSign,
|
|
'?': TokenType.QuestionMark,
|
|
'@': TokenType.CommercialAt,
|
|
'[': TokenType.LeftSquareBracket,
|
|
']': TokenType.RightSquareBracket,
|
|
'^': TokenType.CircumflexAccent,
|
|
'_': TokenType.LowLine,
|
|
'{': TokenType.LeftCurlyBracket,
|
|
'|': TokenType.VerticalLine,
|
|
'}': TokenType.RightCurlyBracket,
|
|
'~': TokenType.Tilde
|
|
};
|
|
|
|
/**
|
|
* Add a token to the token list
|
|
* @param {string} type
|
|
* @param {string} value
|
|
*/
|
|
function pushToken(type, value, column) {
|
|
tokens.push({
|
|
tn: tn++,
|
|
ln: ln,
|
|
col: column,
|
|
type: type,
|
|
value: value
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Check if a character is a decimal digit
|
|
* @param {string} c Character
|
|
* @returns {boolean}
|
|
*/
|
|
function isDecimalDigit(c) {
|
|
return '0123456789'.indexOf(c) >= 0;
|
|
}
|
|
|
|
/**
|
|
* Parse spaces
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseSpaces(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a non-space character:
|
|
for (; pos < cssLength; pos++) {
|
|
if (css.charAt(pos) !== ' ') break;
|
|
}
|
|
|
|
// Add a substring containing only spaces to tokens:
|
|
pushToken(TokenType.Space, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse a string within quotes
|
|
* @param {string} css Unparsed part of CSS string
|
|
* @param {string} q Quote (either `'` or `"`)
|
|
*/
|
|
function parseString(css, q) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a matching quote:
|
|
for (pos++; pos < cssLength; pos++) {
|
|
// Skip escaped quotes:
|
|
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) === q) break;
|
|
}
|
|
|
|
// Add the string (including quotes) to tokens:
|
|
pushToken(q === '"' ? TokenType.StringDQ : TokenType.StringSQ, css.substring(start, pos + 1), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse numbers
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseDecimalNumber(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a character that's not a digit:
|
|
for (; pos < cssLength; pos++) {
|
|
if (!isDecimalDigit(css.charAt(pos))) break;
|
|
}
|
|
|
|
// Add the number to tokens:
|
|
pushToken(TokenType.DecimalNumber, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse identifier
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseIdentifier(css) {
|
|
var start = pos;
|
|
|
|
// Skip all opening slashes:
|
|
while (css.charAt(pos) === '/') {
|
|
pos++;
|
|
} // Read the string until we meet a punctuation mark:
|
|
for (; pos < cssLength; pos++) {
|
|
// Skip all '\':
|
|
if (css.charAt(pos) === '\\') pos++;else if (Punctuation[css.charAt(pos)]) break;
|
|
}
|
|
|
|
var ident = css.substring(start, pos--);
|
|
|
|
// Enter url mode if parsed substring is `url`:
|
|
urlMode = urlMode || ident === 'url';
|
|
|
|
// Add identifier to tokens:
|
|
pushToken(TokenType.Identifier, ident, col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse a multiline comment
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseMLComment(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet `*/`.
|
|
// Since we already know first 2 characters (`/*`), start reading
|
|
// from `pos + 2`:
|
|
for (pos = pos + 2; pos < cssLength; pos++) {
|
|
if (css.charAt(pos) === '*' && css.charAt(pos + 1) === '/') {
|
|
pos++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Add full comment (including `/*` and `*/`) to the list of tokens:
|
|
var comment = css.substring(start, pos + 1);
|
|
pushToken(TokenType.CommentML, comment, col);
|
|
|
|
var newlines = comment.split('\n');
|
|
if (newlines.length > 1) {
|
|
ln += newlines.length - 1;
|
|
col = newlines[newlines.length - 1].length;
|
|
} else {
|
|
col += pos - start;
|
|
}
|
|
}
|
|
|
|
function parseSLComment(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet line break.
|
|
// Since we already know first 2 characters (`//`), start reading
|
|
// from `pos + 2`:
|
|
for (pos += 2; pos < cssLength; pos++) {
|
|
if (css.charAt(pos) === '\n' || css.charAt(pos) === '\r') {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Add comment (including `//` and line break) to the list of tokens:
|
|
pushToken(TokenType.CommentSL, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Convert a CSS string to a list of tokens
|
|
* @param {string} css CSS string
|
|
* @returns {Array} List of tokens
|
|
* @private
|
|
*/
|
|
function getTokens(css) {
|
|
var c; // Current character
|
|
var cn; // Next character
|
|
|
|
cssLength = css.length;
|
|
|
|
// Parse string, character by character:
|
|
for (pos = 0; pos < cssLength; col++, pos++) {
|
|
c = css.charAt(pos);
|
|
cn = css.charAt(pos + 1);
|
|
|
|
// If we meet `/*`, it's a start of a multiline comment.
|
|
// Parse following characters as a multiline comment:
|
|
if (c === '/' && cn === '*') {
|
|
parseMLComment(css);
|
|
}
|
|
|
|
// If we meet `//` and it is not a part of url:
|
|
else if (!urlMode && c === '/' && cn === '/') {
|
|
// If we're currently inside a block, treat `//` as a start
|
|
// of identifier. Else treat `//` as a start of a single-line
|
|
// comment:
|
|
if (blockMode > 0) parseIdentifier(css);else parseSLComment(css);
|
|
}
|
|
|
|
// If current character is a double or single quote, it's a start
|
|
// of a string:
|
|
else if (c === '"' || c === "'") {
|
|
parseString(css, c);
|
|
}
|
|
|
|
// If current character is a space:
|
|
else if (c === ' ') {
|
|
parseSpaces(css);
|
|
}
|
|
|
|
// If current character is a punctuation mark:
|
|
else if (Punctuation[c]) {
|
|
// Add it to the list of tokens:
|
|
pushToken(Punctuation[c], c, col);
|
|
if (c === '\n' || c === '\r') {
|
|
ln++;
|
|
col = 0;
|
|
} // Go to next line
|
|
else if (c === ')') urlMode = false; // Exit url mode
|
|
else if (c === '{') blockMode++; // Enter a block
|
|
else if (c === '}') blockMode--; // Exit a block
|
|
else if (c === '\t' && tabSize > 1) col += tabSize - 1;
|
|
}
|
|
|
|
// If current character is a decimal digit:
|
|
else if (isDecimalDigit(c)) {
|
|
parseDecimalNumber(css);
|
|
}
|
|
|
|
// If current character is anything else:
|
|
else {
|
|
parseIdentifier(css);
|
|
}
|
|
}
|
|
|
|
return tokens;
|
|
}
|
|
|
|
return getTokens(css);
|
|
};
|
|
|
|
/***/ }),
|
|
/* 17 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
exports.__esModule = true;
|
|
exports.default = {
|
|
mark: __webpack_require__(18),
|
|
parse: __webpack_require__(19),
|
|
stringify: __webpack_require__(4),
|
|
tokenizer: __webpack_require__(20)
|
|
};
|
|
module.exports = exports['default'];
|
|
|
|
/***/ }),
|
|
/* 18 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
module.exports = function () {
|
|
/**
|
|
* Mark whitespaces and comments
|
|
*/
|
|
function markSC(tokens) {
|
|
var tokensLength = tokens.length;
|
|
var ws = -1; // Flag for whitespaces
|
|
var sc = -1; // Flag for whitespaces and comments
|
|
var t = void 0; // Current token
|
|
|
|
// For every token in the token list, mark spaces and line breaks
|
|
// as spaces (set both `ws` and `sc` flags). Mark multiline comments
|
|
// with `sc` flag.
|
|
// If there are several spaces or tabs or line breaks or multiline
|
|
// comments in a row, group them: take the last one's index number
|
|
// and save it to the first token in the group as a reference:
|
|
// e.g., `ws_last = 7` for a group of whitespaces or `sc_last = 9`
|
|
// for a group of whitespaces and comments.
|
|
for (var i = 0; i < tokensLength; i++) {
|
|
t = tokens[i];
|
|
switch (t.type) {
|
|
case TokenType.Space:
|
|
case TokenType.Tab:
|
|
case TokenType.Newline:
|
|
t.ws = true;
|
|
t.sc = true;
|
|
|
|
if (ws === -1) ws = i;
|
|
if (sc === -1) sc = i;
|
|
|
|
break;
|
|
case TokenType.CommentML:
|
|
case TokenType.CommentSL:
|
|
if (ws !== -1) {
|
|
tokens[ws].ws_last = i - 1;
|
|
ws = -1;
|
|
}
|
|
|
|
t.sc = true;
|
|
|
|
break;
|
|
default:
|
|
if (ws !== -1) {
|
|
tokens[ws].ws_last = i - 1;
|
|
ws = -1;
|
|
}
|
|
|
|
if (sc !== -1) {
|
|
tokens[sc].sc_last = i - 1;
|
|
sc = -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ws !== -1) tokens[ws].ws_last = i - 1;
|
|
if (sc !== -1) tokens[sc].sc_last = i - 1;
|
|
}
|
|
|
|
/**
|
|
* Pair brackets
|
|
*/
|
|
function markBrackets(tokens) {
|
|
var tokensLength = tokens.length;
|
|
var ps = []; // Parentheses
|
|
var sbs = []; // Square brackets
|
|
var cbs = []; // Curly brackets
|
|
var t = void 0; // Current token
|
|
|
|
// For every token in the token list, if we meet an opening (left)
|
|
// bracket, push its index number to a corresponding array.
|
|
// If we then meet a closing (right) bracket, look at the corresponding
|
|
// array. If there are any elements (records about previously met
|
|
// left brackets), take a token of the last left bracket (take
|
|
// the last index number from the array and find a token with
|
|
// this index number) and save right bracket's index as a reference:
|
|
for (var i = 0; i < tokensLength; i++) {
|
|
t = tokens[i];
|
|
switch (t.type) {
|
|
case TokenType.LeftParenthesis:
|
|
ps.push(i);
|
|
break;
|
|
case TokenType.RightParenthesis:
|
|
if (ps.length) {
|
|
t.left = ps.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
break;
|
|
case TokenType.LeftSquareBracket:
|
|
sbs.push(i);
|
|
break;
|
|
case TokenType.RightSquareBracket:
|
|
if (sbs.length) {
|
|
t.left = sbs.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
break;
|
|
case TokenType.LeftCurlyBracket:
|
|
cbs.push(i);
|
|
break;
|
|
case TokenType.RightCurlyBracket:
|
|
if (cbs.length) {
|
|
t.left = cbs.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return function (tokens) {
|
|
markBrackets(tokens);
|
|
markSC(tokens);
|
|
};
|
|
}();
|
|
|
|
/***/ }),
|
|
/* 19 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var Node = __webpack_require__(1);
|
|
var NodeType = __webpack_require__(15);
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
var tokens = void 0;
|
|
var tokensLength = void 0;
|
|
var pos = void 0;
|
|
|
|
var contexts = {
|
|
'arguments': function _arguments() {
|
|
return checkArguments(pos) && getArguments();
|
|
},
|
|
'atkeyword': function atkeyword() {
|
|
return checkAtkeyword(pos) && getAtkeyword();
|
|
},
|
|
'atrule': function atrule() {
|
|
return checkAtrule(pos) && getAtrule();
|
|
},
|
|
'attributeSelector': function attributeSelector() {
|
|
return checkAttributeSelector(pos) && getAttributeSelector();
|
|
},
|
|
'block': function block() {
|
|
return checkBlock(pos) && getBlock();
|
|
},
|
|
'brackets': function brackets() {
|
|
return checkBrackets(pos) && getBrackets();
|
|
},
|
|
'class': function _class() {
|
|
return checkClass(pos) && getClass();
|
|
},
|
|
'combinator': function combinator() {
|
|
return checkCombinator(pos) && getCombinator();
|
|
},
|
|
'commentML': function commentML() {
|
|
return checkCommentML(pos) && getCommentML();
|
|
},
|
|
'commentSL': function commentSL() {
|
|
return checkCommentSL(pos) && getCommentSL();
|
|
},
|
|
'condition': function condition() {
|
|
return checkCondition(pos) && getCondition();
|
|
},
|
|
'declaration': function declaration() {
|
|
return checkDeclaration(pos) && getDeclaration();
|
|
},
|
|
'declDelim': function declDelim() {
|
|
return checkDeclDelim(pos) && getDeclDelim();
|
|
},
|
|
'delim': function delim() {
|
|
return checkDelim(pos) && getDelim();
|
|
},
|
|
'dimension': function dimension() {
|
|
return checkDimension(pos) && getDimension();
|
|
},
|
|
'escapedString': function escapedString() {
|
|
return checkEscapedString(pos) && getEscapedString();
|
|
},
|
|
'expression': function expression() {
|
|
return checkExpression(pos) && getExpression();
|
|
},
|
|
'extend': function extend() {
|
|
return checkExtend(pos) && getExtend();
|
|
},
|
|
'function': function _function() {
|
|
return checkFunction(pos) && getFunction();
|
|
},
|
|
'ident': function ident() {
|
|
return checkIdent(pos) && getIdent();
|
|
},
|
|
'important': function important() {
|
|
return checkImportant(pos) && getImportant();
|
|
},
|
|
'include': function include() {
|
|
return checkInclude(pos) && getInclude();
|
|
},
|
|
'interpolatedVariable': function interpolatedVariable() {
|
|
return checkInterpolatedVariable(pos) && getInterpolatedVariable();
|
|
},
|
|
'mixin': function mixin() {
|
|
return checkMixin(pos) && getMixin();
|
|
},
|
|
'namespace': function namespace() {
|
|
return checkNamespace(pos) && getNamespace();
|
|
},
|
|
'number': function number() {
|
|
return checkNumber(pos) && getNumber();
|
|
},
|
|
'operator': function operator() {
|
|
return checkOperator(pos) && getOperator();
|
|
},
|
|
'parentheses': function parentheses() {
|
|
return checkParentheses(pos) && getParentheses();
|
|
},
|
|
'parentselector': function parentselector() {
|
|
return checkParentSelector(pos) && getParentSelector();
|
|
},
|
|
'percentage': function percentage() {
|
|
return checkPercentage(pos) && getPercentage();
|
|
},
|
|
'progid': function progid() {
|
|
return checkProgid(pos) && getProgid();
|
|
},
|
|
'property': function property() {
|
|
return checkProperty(pos) && getProperty();
|
|
},
|
|
'propertyDelim': function propertyDelim() {
|
|
return checkPropertyDelim(pos) && getPropertyDelim();
|
|
},
|
|
'pseudoc': function pseudoc() {
|
|
return checkPseudoc(pos) && getPseudoc();
|
|
},
|
|
'pseudoe': function pseudoe() {
|
|
return checkPseudoe(pos) && getPseudoe();
|
|
},
|
|
'ruleset': function ruleset() {
|
|
return checkRuleset(pos) && getRuleset();
|
|
},
|
|
's': function s() {
|
|
return checkS(pos) && getS();
|
|
},
|
|
'selector': function selector() {
|
|
return checkSelector(pos) && getSelector();
|
|
},
|
|
'shash': function shash() {
|
|
return checkShash(pos) && getShash();
|
|
},
|
|
'string': function string() {
|
|
return checkString(pos) && getString();
|
|
},
|
|
'stylesheet': function stylesheet() {
|
|
return checkStylesheet(pos) && getStylesheet();
|
|
},
|
|
'unary': function unary() {
|
|
return checkUnary(pos) && getUnary();
|
|
},
|
|
'unicodeRange': function unicodeRange() {
|
|
return checkUnicodeRange(pos) && getUnicodeRange();
|
|
},
|
|
'universalSelector': function universalSelector() {
|
|
return checkUniversalSelector(pos) && getUniversalSelector();
|
|
},
|
|
'urange': function urange() {
|
|
return checkUrange(pos) && getUrange();
|
|
},
|
|
'uri': function uri() {
|
|
return checkUri(pos) && getUri();
|
|
},
|
|
'value': function value() {
|
|
return checkValue(pos) && getValue();
|
|
},
|
|
'variable': function variable() {
|
|
return checkVariable(pos) && getVariable();
|
|
},
|
|
'variableslist': function variableslist() {
|
|
return checkVariablesList(pos) && getVariablesList();
|
|
},
|
|
'vhash': function vhash() {
|
|
return checkVhash(pos) && getVhash();
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Stop parsing and display error
|
|
* @param {Number=} i Token's index number
|
|
*/
|
|
function throwError(i) {
|
|
var ln = tokens[i].ln;
|
|
|
|
throw { line: ln, syntax: 'less' };
|
|
}
|
|
|
|
/**
|
|
* @param {Object} exclude
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkExcluding(exclude, i) {
|
|
var start = i;
|
|
|
|
while (i < tokensLength) {
|
|
if (exclude[tokens[i++].type]) break;
|
|
}
|
|
|
|
return i - start - 2;
|
|
}
|
|
|
|
/**
|
|
* @param {Number} start
|
|
* @param {Number} finish
|
|
* @returns {String}
|
|
*/
|
|
function joinValues(start, finish) {
|
|
var s = '';
|
|
|
|
for (var i = start; i < finish + 1; i++) {
|
|
s += tokens[i].value;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* @param {Number} start
|
|
* @param {Number} num
|
|
* @returns {String}
|
|
*/
|
|
function joinValues2(start, num) {
|
|
if (start + num - 1 >= tokensLength) return;
|
|
|
|
var s = '';
|
|
|
|
for (var i = 0; i < num; i++) {
|
|
s += tokens[start + i].value;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
function getLastPosition(content, line, column, colOffset) {
|
|
return typeof content === 'string' ? getLastPositionForString(content, line, column, colOffset) : getLastPositionForArray(content, line, column, colOffset);
|
|
}
|
|
|
|
function getLastPositionForString(content, line, column, colOffset) {
|
|
var position = [];
|
|
|
|
if (!content) {
|
|
position = [line, column];
|
|
if (colOffset) position[1] += colOffset - 1;
|
|
return position;
|
|
}
|
|
|
|
var lastLinebreak = content.lastIndexOf('\n');
|
|
var endsWithLinebreak = lastLinebreak === content.length - 1;
|
|
var splitContent = content.split('\n');
|
|
var linebreaksCount = splitContent.length - 1;
|
|
var prevLinebreak = linebreaksCount === 0 || linebreaksCount === 1 ? -1 : content.length - splitContent[linebreaksCount - 1].length - 2;
|
|
|
|
// Line:
|
|
var offset = endsWithLinebreak ? linebreaksCount - 1 : linebreaksCount;
|
|
position[0] = line + offset;
|
|
|
|
// Column:
|
|
if (endsWithLinebreak) {
|
|
offset = prevLinebreak !== -1 ? content.length - prevLinebreak : content.length - 1;
|
|
} else {
|
|
offset = linebreaksCount !== 0 ? content.length - lastLinebreak - column - 1 : content.length - 1;
|
|
}
|
|
position[1] = column + offset;
|
|
|
|
if (!colOffset) return position;
|
|
|
|
if (endsWithLinebreak) {
|
|
position[0]++;
|
|
position[1] = colOffset;
|
|
} else {
|
|
position[1] += colOffset;
|
|
}
|
|
|
|
return position;
|
|
}
|
|
|
|
function getLastPositionForArray(content, line, column, colOffset) {
|
|
var position = void 0;
|
|
|
|
if (content.length === 0) {
|
|
position = [line, column];
|
|
} else {
|
|
var c = content[content.length - 1];
|
|
if (c.hasOwnProperty('end')) {
|
|
position = [c.end.line, c.end.column];
|
|
} else {
|
|
position = getLastPosition(c.content, line, column);
|
|
}
|
|
}
|
|
|
|
if (!colOffset) return position;
|
|
|
|
if (tokens[pos - 1] && tokens[pos - 1].type !== 'Newline') {
|
|
position[1] += colOffset;
|
|
} else {
|
|
position[0]++;
|
|
position[1] = 1;
|
|
}
|
|
|
|
return position;
|
|
}
|
|
|
|
function newNode(type, content, line, column, end) {
|
|
if (!end) end = getLastPosition(content, line, column);
|
|
return new Node({
|
|
type: type,
|
|
content: content,
|
|
start: {
|
|
line: line,
|
|
column: column
|
|
},
|
|
end: {
|
|
line: end[0],
|
|
column: end[1]
|
|
},
|
|
syntax: 'less'
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkAny(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkBrackets(i)) tokens[i].any_child = 1;else if (l = checkParentheses(i)) tokens[i].any_child = 2;else if (l = checkString(i)) tokens[i].any_child = 3;else if (l = checkVariablesList(i)) tokens[i].any_child = 4;else if (l = checkVariable(i)) tokens[i].any_child = 5;else if (l = checkPercentage(i)) tokens[i].any_child = 6;else if (l = checkDimension(i)) tokens[i].any_child = 7;else if (l = checkUnicodeRange(i)) tokens[i].any_child = 15;else if (l = checkNumber(i)) tokens[i].any_child = 8;else if (l = checkUri(i)) tokens[i].any_child = 9;else if (l = checkExpression(i)) tokens[i].any_child = 10;else if (l = checkFunction(i)) tokens[i].any_child = 11;else if (l = checkIdent(i)) tokens[i].any_child = 12;else if (l = checkClass(i)) tokens[i].any_child = 13;else if (l = checkUnary(i)) tokens[i].any_child = 14;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getAny() {
|
|
var childType = tokens[pos].any_child;
|
|
|
|
if (childType === 1) return getBrackets();
|
|
if (childType === 2) return getParentheses();
|
|
if (childType === 3) return getString();
|
|
if (childType === 4) return getVariablesList();
|
|
if (childType === 5) return getVariable();
|
|
if (childType === 6) return getPercentage();
|
|
if (childType === 7) return getDimension();
|
|
if (childType === 15) return getUnicodeRange();
|
|
if (childType === 8) return getNumber();
|
|
if (childType === 9) return getUri();
|
|
if (childType === 10) return getExpression();
|
|
if (childType === 11) return getFunction();
|
|
if (childType === 12) return getIdent();
|
|
if (childType === 13) return getClass();
|
|
if (childType === 14) return getUnary();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of mixin's arguments.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkArguments(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
while (i < tokens[start].right) {
|
|
if (l = checkArgument(i)) i += l;else return 0;
|
|
}
|
|
|
|
return tokens[start].right - start + 1;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getArguments() {
|
|
var type = NodeType.ArgumentsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var body = void 0;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
while (pos < tokensLength && tokens[pos].type !== TokenType.RightParenthesis) {
|
|
if (checkDeclaration(pos)) content.push(getDeclaration());else if (checkArgument(pos)) {
|
|
body = getArgument();
|
|
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
|
|
} else if (checkClass(pos)) content.push(getClass());else throwError(pos);
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is valid to be part of arguments list.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkArgument(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkEscapedString(i)) tokens[i].argument_child = 1;else if (l = checkDeclaration(i)) tokens[i].argument_child = 2;else if (l = checkVariablesList(i)) tokens[i].argument_child = 3;else if (l = checkVariable(i)) tokens[i].argument_child = 4;else if (l = checkSC(i)) tokens[i].argument_child = 5;else if (l = checkUnary(i)) tokens[i].argument_child = 6;else if (l = checkOperator(i)) tokens[i].argument_child = 7;else if (l = checkDelim(i)) tokens[i].argument_child = 8;else if (l = checkDeclDelim(i)) tokens[i].argument_child = 9;else if (l = checkString(i)) tokens[i].argument_child = 10;else if (l = checkPercentage(i)) tokens[i].argument_child = 11;else if (l = checkDimension(i)) tokens[i].argument_child = 12;else if (l = checkNumber(i)) tokens[i].argument_child = 13;else if (l = checkUri(i)) tokens[i].argument_child = 14;else if (l = checkFunction(i)) tokens[i].argument_child = 15;else if (l = checkIdent(i)) tokens[i].argument_child = 16;else if (l = checkVhash(i)) tokens[i].argument_child = 17;else if (l = checkBlock(i)) tokens[i].argument_child = 18;else if (l = checkParentheses(i)) tokens[i].argument_child = 19;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} Node that is part of arguments list.
|
|
*/
|
|
function getArgument() {
|
|
var childType = tokens[pos].argument_child;
|
|
|
|
if (childType === 1) return getEscapedString();
|
|
if (childType === 2) return getDeclaration();
|
|
if (childType === 3) return getVariablesList();
|
|
if (childType === 4) return getVariable();
|
|
if (childType === 5) return getSC();
|
|
if (childType === 6) return getUnary();
|
|
if (childType === 7) return getOperator();
|
|
if (childType === 8) return getDelim();
|
|
if (childType === 9) return getDeclDelim();
|
|
if (childType === 10) return getString();
|
|
if (childType === 11) return getPercentage();
|
|
if (childType === 12) return getDimension();
|
|
if (childType === 13) return getNumber();
|
|
if (childType === 14) return getUri();
|
|
if (childType === 15) return getFunction();
|
|
if (childType === 16) return getIdent();
|
|
if (childType === 17) return getVhash();
|
|
if (childType === 18) return getBlock();
|
|
if (childType === 19) return getParentheses();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an @-word (e.g. `@import`, `@include`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkAtkeyword(i) {
|
|
var l = void 0;
|
|
|
|
// Check that token is `@`:
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.CommercialAt) return 0;
|
|
|
|
return (l = checkIdent(i)) ? l + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with @-word
|
|
* @returns {Array} `['atkeyword', ['ident', x]]` where `x` is
|
|
* an identifier without
|
|
* `@` (e.g. `import`, `include`)
|
|
*/
|
|
function getAtkeyword() {
|
|
var type = NodeType.AtkeywordType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `@`.
|
|
pos++;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a part of an @-rule
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of @-rule
|
|
*/
|
|
function checkAtrule(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// If token already has a record of being part of an @-rule,
|
|
// return the @-rule's length:
|
|
if (tokens[i].atrule_l !== undefined) return tokens[i].atrule_l;
|
|
|
|
// If token is part of an @-rule, save the rule's type to token.
|
|
if (l = checkKeyframesRule(i)) tokens[i].atrule_type = 4;
|
|
// @-rule with ruleset:
|
|
else if (l = checkAtruler(i)) tokens[i].atrule_type = 1;
|
|
// Block @-rule:
|
|
else if (l = checkAtruleb(i)) tokens[i].atrule_type = 2;
|
|
// Single-line @-rule:
|
|
else if (l = checkAtrules(i)) tokens[i].atrule_type = 3;else return 0;
|
|
|
|
// If token is part of an @-rule, save the rule's length to token:
|
|
tokens[i].atrule_l = l;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* Get node with @-rule
|
|
* @returns {Array}
|
|
*/
|
|
function getAtrule() {
|
|
var childType = tokens[pos].atrule_type;
|
|
|
|
if (childType === 1) return getAtruler(); // @-rule with ruleset
|
|
if (childType === 2) return getAtruleb(); // Block @-rule
|
|
if (childType === 3) return getAtrules(); // Single-line @-rule
|
|
if (childType === 4) return getKeyframesRule();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a block @-rule
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the @-rule
|
|
*/
|
|
function checkAtruleb(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a block @-rule
|
|
* @returns {Array} `['atruleb', ['atkeyword', x], y, ['block', z]]`
|
|
*/
|
|
function getAtruleb() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an @-rule with ruleset
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the @-rule
|
|
*/
|
|
function checkAtruler(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
|
|
|
|
if (l = checkAtrulers(i)) i += l;
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with an @-rule with ruleset
|
|
* @returns {Array} ['atruler', ['atkeyword', x], y, z]
|
|
*/
|
|
function getAtruler() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets(), getAtrulers());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkAtrulers(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkSC(i)) tokens[i].atrulers_child = 1;else if (l = checkAtrule(i)) tokens[i].atrulers_child = 2;else if (l = checkRuleset(i)) tokens[i].atrulers_child = 3;else break;
|
|
i += l;
|
|
}
|
|
|
|
if (i < tokensLength) tokens[i].atrulers_end = 1;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `['atrulers', x]`
|
|
*/
|
|
function getAtrulers() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `{`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC());
|
|
|
|
while (pos < tokensLength && !tokens[pos].atrulers_end) {
|
|
var childType = tokens[pos].atrulers_child;
|
|
if (childType === 1) content = content.concat(getSC());else if (childType === 2) content.push(getAtrule());else if (childType === 3) content.push(getRuleset());else break;
|
|
}
|
|
|
|
content = content.concat(getSC());
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `}`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkAtrules(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `['atrules', ['atkeyword', x], y]`
|
|
*/
|
|
function getAtrules() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a block (e.g. `{...}`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the block
|
|
*/
|
|
function checkBlock(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket ? tokens[i].right - i + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a block
|
|
* @returns {Array} `['block', x]`
|
|
*/
|
|
function getBlock() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = tokens[pos].right;
|
|
var content = [];
|
|
|
|
// Skip `{`.
|
|
pos++;
|
|
|
|
while (pos < end) {
|
|
if (checkBlockdecl(pos)) content = content.concat(getBlockdecl());else throwError(pos);
|
|
}
|
|
|
|
var end_ = getLastPosition(content, line, column, 1);
|
|
pos = end + 1;
|
|
|
|
return newNode(type, content, line, column, end_);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a declaration (property-value pair)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the declaration
|
|
*/
|
|
function checkBlockdecl(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkBlockdecl1(i)) tokens[i].bd_type = 1;else if (l = checkBlockdecl2(i)) tokens[i].bd_type = 2;else if (l = checkBlockdecl3(i)) tokens[i].bd_type = 3;else if (l = checkBlockdecl4(i)) tokens[i].bd_type = 4;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getBlockdecl() {
|
|
var childType = tokens[pos].bd_type;
|
|
|
|
if (childType === 1) return getBlockdecl1();
|
|
if (childType === 2) return getBlockdecl2();
|
|
if (childType === 3) return getBlockdecl3();
|
|
if (childType === 4) return getBlockdecl4();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkBlockdecl1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkCondition(i)) tokens[i].bd_kind = 1;else if (l = checkExtend(i)) tokens[i].bd_kind = 6;else if (l = checkRuleset(i)) tokens[i].bd_kind = 2;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 3;else if (l = checkAtrule(i)) tokens[i].bd_kind = 4;else if (l = checkInclude(i)) tokens[i].bd_kind = 5;else return 0;
|
|
|
|
i += l;
|
|
|
|
if (i < tokensLength && (l = checkDeclDelim(i))) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getBlockdecl1() {
|
|
var sc = getSC();
|
|
var content = void 0;
|
|
|
|
switch (tokens[pos].bd_kind) {
|
|
case 1:
|
|
content = getCondition();
|
|
break;
|
|
case 2:
|
|
content = getRuleset();
|
|
break;
|
|
case 3:
|
|
content = getDeclaration();
|
|
break;
|
|
case 4:
|
|
content = getAtrule();
|
|
break;
|
|
case 5:
|
|
content = getInclude();
|
|
break;
|
|
case 6:
|
|
content = getExtend();
|
|
break;
|
|
}
|
|
|
|
return sc.concat(content, getDeclDelim(), getSC());
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkBlockdecl2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkCondition(i)) tokens[i].bd_kind = 1;else if (l = checkExtend(i)) tokens[i].bd_kind = 3;else if (l = checkRuleset(i)) tokens[i].bd_kind = 6;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 4;else if (l = checkAtrule(i)) tokens[i].bd_kind = 5;else if (l = checkInclude(i)) tokens[i].bd_kind = 2;else return 0;
|
|
|
|
i += l;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getBlockdecl2() {
|
|
var sc = getSC();
|
|
var content = void 0;
|
|
|
|
switch (tokens[pos].bd_kind) {
|
|
case 1:
|
|
content = getCondition();
|
|
break;
|
|
case 2:
|
|
content = getInclude();
|
|
break;
|
|
case 3:
|
|
content = getExtend();
|
|
break;
|
|
case 4:
|
|
content = getDeclaration();
|
|
break;
|
|
case 5:
|
|
content = getAtrule();
|
|
break;
|
|
case 6:
|
|
content = getRuleset();
|
|
break;
|
|
}
|
|
|
|
return sc.concat(content, getSC());
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkBlockdecl3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkDeclDelim(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `[s0, ['declDelim'], s1]` where `s0` and `s1` are
|
|
* are optional whitespaces.
|
|
*/
|
|
function getBlockdecl3() {
|
|
return [].concat(getSC(), getDeclDelim(), getSC());
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkBlockdecl4(i) {
|
|
return checkSC(i);
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getBlockdecl4() {
|
|
return getSC();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of text inside square brackets, e.g. `[1]`
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkBrackets(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
|
|
// Skip `[`.
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
if (i < tokens[start].right) {
|
|
var l = checkTsets(i);
|
|
if (l) i += l;else return 0;
|
|
}
|
|
|
|
// Skip `]`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with text inside square brackets, e.g. `[1]`
|
|
* @returns {Node}
|
|
*/
|
|
function getBrackets() {
|
|
var type = NodeType.BracketsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var right = token.right;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
if (pos < right) {
|
|
content = getTsets();
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a class selector (e.g. `.abc`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the class selector
|
|
*/
|
|
function checkClass(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].class_l) return tokens[i].class_l;
|
|
|
|
if (tokens[i++].type === TokenType.FullStop) {
|
|
if (l = checkInterpolatedVariable(i)) tokens[i].class_child = 1;else if (l = checkIdent(i)) tokens[i].class_child = 2;else return 0;
|
|
|
|
tokens[i].class_l = l + 1;
|
|
return l + 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a class selector
|
|
* @returns {Array} `['class', ['ident', x]]` where x is a class's
|
|
* identifier (without `.`, e.g. `abc`).
|
|
*/
|
|
function getClass() {
|
|
var type = NodeType.ClassType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `.`
|
|
pos++;
|
|
|
|
var childType = tokens[pos].class_child;
|
|
if (childType === 1) content.push(getInterpolatedVariable());else content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkCombinator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkCombinator1(i)) tokens[i].combinatorType = 1;else if (l = checkCombinator2(i)) tokens[i].combinatorType = 2;else if (l = checkCombinator3(i)) tokens[i].combinatorType = 3;else if (l = checkCombinator4(i)) tokens[i].combinatorType = 4;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getCombinator() {
|
|
var type = tokens[pos].combinatorType;
|
|
if (type === 1) return getCombinator1();
|
|
if (type === 2) return getCombinator2();
|
|
if (type === 3) return getCombinator3();
|
|
if (type === 4) return getCombinator4();
|
|
}
|
|
|
|
/**
|
|
* (1) `>>>`
|
|
*
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkCombinator1(i) {
|
|
if (i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign) return 3;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator1() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '>>>';
|
|
|
|
// Skip combinator
|
|
pos += 3;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `||`
|
|
* (2) `>>`
|
|
*
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkCombinator2(i) {
|
|
if (i + 1 >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.VerticalLine && tokens[i + 1].type === TokenType.VerticalLine) return 2;
|
|
|
|
if (tokens[i].type === TokenType.GreaterThanSign && tokens[i + 1].type === TokenType.GreaterThanSign) return 2;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator2() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '' + token.value + tokens[pos + 1].value;
|
|
|
|
// Skip combinator
|
|
pos += 2;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `>`
|
|
* (2) `+`
|
|
* (3) `~`
|
|
*
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkCombinator3(i) {
|
|
var type = tokens[i].type;
|
|
if (type === TokenType.PlusSign || type === TokenType.GreaterThanSign || type === TokenType.Tilde) return 1;else return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator3() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
// Skip combinator
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `/panda/`
|
|
*/
|
|
function checkCombinator4(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator4() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `/`.
|
|
pos++;
|
|
|
|
var ident = getIdent();
|
|
|
|
// Skip `/`.
|
|
pos++;
|
|
|
|
var content = '/' + ident.content + '/';
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a multiline comment.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is a multiline comment, otherwise `0`
|
|
*/
|
|
function checkCommentML(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.CommentML ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a multiline comment
|
|
* @returns {Array} `['commentML', x]` where `x`
|
|
* is the comment's text (without `/*` and `* /`).
|
|
*/
|
|
function getCommentML() {
|
|
var type = NodeType.CommentMLType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos].value.substring(2);
|
|
var l = content.length;
|
|
|
|
if (content.charAt(l - 2) === '*' && content.charAt(l - 1) === '/') content = content.substring(0, l - 2);
|
|
|
|
var end = getLastPosition(content, line, column, 2);
|
|
if (end[0] === line) end[1] += 2;
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a single-line comment.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is a single-line comment, otherwise `0`
|
|
*/
|
|
function checkCommentSL(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.CommentSL ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a single-line comment.
|
|
* @returns {Array}
|
|
*/
|
|
function getCommentSL() {
|
|
var type = NodeType.CommentSLType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos++].value.substring(2);
|
|
var end = getLastPosition(content, line, column + 2);
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a condition.
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the condition
|
|
*/
|
|
function checkCondition(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if ((l = checkIdent(i)) && tokens[i].value === 'when') i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkBlock(i)) {
|
|
tokens[i].condition_child = 0;
|
|
break;
|
|
} else if (l = checkFunction(i)) tokens[i].condition_child = 1;else if (l = checkBrackets(i)) tokens[i].condition_child = 2;else if (l = checkParentheses(i)) tokens[i].condition_child = 3;else if (l = checkVariable(i)) tokens[i].condition_child = 4;else if (l = checkIdent(i)) tokens[i].condition_child = 5;else if (l = checkNumber(i)) tokens[i].condition_child = 6;else if (l = checkDelim(i)) tokens[i].condition_child = 7;else if (l = checkOperator(i)) tokens[i].condition_child = 8;else if (l = checkCombinator(i)) tokens[i].condition_child = 9;else if (l = checkSC(i)) tokens[i].condition_child = 10;else if (l = checkString(i)) tokens[i].condition_child = 11;else return 0;
|
|
|
|
i += l;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a condition.
|
|
* @returns {Array} `['condition', x]`
|
|
*/
|
|
function getCondition() {
|
|
var type = NodeType.ConditionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(getIdent());
|
|
|
|
while (pos < tokensLength) {
|
|
var childType = tokens[pos].condition_child;
|
|
|
|
if (childType === 0) break;else if (childType === 1) content.push(getFunction());else if (childType === 2) content.push(getBrackets());else if (childType === 3) content.push(getParentheses());else if (childType === 4) content.push(getVariable());else if (childType === 5) content.push(getIdent());else if (childType === 6) content.push(getNumber());else if (childType === 7) content.push(getDelim());else if (childType === 8) content.push(getOperator());else if (childType === 9) content.push(getCombinator());else if (childType === 10) content = content.concat(getSC());else if (childType === 11) content.push(getString());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a declaration (property-value pair)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the declaration
|
|
*/
|
|
function checkDeclaration(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkProperty(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkPropertyDelim(i)) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkValue(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a declaration
|
|
* @returns {Array} `['declaration', ['property', x], ['propertyDelim'],
|
|
* ['value', y]]`
|
|
*/
|
|
function getDeclaration() {
|
|
var type = NodeType.DeclarationType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getProperty(), getSC(), getPropertyDelim(), getSC(), getValue());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a semicolon
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is a semicolon, otherwise `0`
|
|
*/
|
|
function checkDeclDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Semicolon ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a semicolon
|
|
* @returns {Array} `['declDelim']`
|
|
*/
|
|
function getDeclDelim() {
|
|
var type = NodeType.DeclDelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ';';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a comma
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is a comma, otherwise `0`
|
|
*/
|
|
function checkDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Comma ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a comma
|
|
* @returns {Array} `['delim']`
|
|
*/
|
|
function getDelim() {
|
|
var type = NodeType.DelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ',';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number with dimension unit (e.g. `10px`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkDimension(i) {
|
|
var ln = checkNumber(i);
|
|
var li = void 0;
|
|
|
|
if (i >= tokensLength || !ln || i + ln >= tokensLength) return 0;
|
|
|
|
return (li = checkUnit(i + ln)) ? ln + li : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node of a number with dimension unit
|
|
* @return {Node}
|
|
*/
|
|
function getDimension() {
|
|
var type = NodeType.DimensionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getNumber(), getUnit()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an escaped string (e.g. `~"ms:something"`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the string (including `~` and quotes)
|
|
*/
|
|
function checkEscapedString(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.Tilde && (l = checkString(i + 1))) return i + l - start;else return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an escaped string
|
|
* @returns {Array} `['escapedString', ['string', x]]` where `x` is a string
|
|
* without `~` but with quotes
|
|
*/
|
|
function getEscapedString() {
|
|
var type = NodeType.EscapedStringType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
pos++;
|
|
|
|
var content = tokens[pos].value;
|
|
var end = getLastPosition(content, line, column + 1);
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkExpression(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength || tokens[i++].value !== 'expression' || i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) {
|
|
return 0;
|
|
}
|
|
|
|
return tokens[i].right - start + 1;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getExpression() {
|
|
var type = NodeType.ExpressionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
pos++;
|
|
|
|
var content = joinValues(pos + 1, tokens[pos].right - 1);
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
if (end[0] === line) end[1] += 11;
|
|
pos = tokens[pos].right + 1;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkExtend(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
|
|
if (l = checkExtend1(i)) tokens[i].extendType = 1;else if (l = checkExtend2(i)) tokens[i].extendType = 2;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getExtend() {
|
|
var childType = tokens[pos].extendType;
|
|
|
|
if (childType === 1) return getExtend1();
|
|
if (childType === 2) return getExtend2();
|
|
}
|
|
|
|
/**
|
|
* (1) `selector:extend(selector) {...}`
|
|
*/
|
|
function checkExtend1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkExtendSelector(i)) i += l;else return 0;
|
|
|
|
if (tokens[i + 1] && tokens[i + 1].value === 'extend' && (l = checkPseudoc(i))) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getExtend1() {
|
|
var type = NodeType.ExtendType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getExtendSelector(), getPseudoc(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `selector:extend(selector)`
|
|
*/
|
|
function checkExtend2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkExtendSelector(i)) i += l;else return 0;
|
|
|
|
if (tokens[i + 1] && tokens[i + 1].value === 'extend' && (l = checkPseudoc(i))) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getExtend2() {
|
|
var type = NodeType.ExtendType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getExtendSelector(), getPseudoc());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkExtendSelector(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkParentSelectorWithExtension(i)) tokens[i].extend_type = 1;else if (l = checkIdent(i)) tokens[i].extend_type = 2;else if (l = checkClass(i)) tokens[i].extend_type = 3;else if (l = checkShash(i)) tokens[i].extend_type = 4;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getExtendSelector() {
|
|
var childType = tokens[pos].extend_type;
|
|
|
|
if (childType === 1) return getParentSelectorWithExtension();
|
|
if (childType === 2) return [getIdent()];
|
|
if (childType === 3) return [getClass()];
|
|
if (childType === 4) return [getShash()];
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkFunction(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i < tokensLength && tokens[i].type === TokenType.LeftParenthesis ? tokens[i].right - start + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getFunction() {
|
|
var type = NodeType.FunctionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getIdent(), getArguments());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an identifier
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the identifier
|
|
*/
|
|
function checkIdent(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.HyphenMinus) i++;
|
|
|
|
if (tokens[i].type === TokenType.LowLine || tokens[i].type === TokenType.Identifier) i++;else return 0;
|
|
|
|
for (; i < tokensLength; i++) {
|
|
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
|
|
}
|
|
|
|
tokens[start].ident_last = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with an identifier
|
|
* @returns {Array} `['ident', x]` where `x` is identifier's name
|
|
*/
|
|
function getIdent() {
|
|
var type = NodeType.IdentType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, tokens[pos].ident_last);
|
|
|
|
pos = tokens[pos].ident_last + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the identifier
|
|
*/
|
|
function checkPartialIdent(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
for (; i < tokensLength; i++) {
|
|
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
|
|
}
|
|
|
|
tokens[start].ident_last = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of `!important` word
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkImportant(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].value === 'important') {
|
|
tokens[start].importantEnd = i;
|
|
return i - start + 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get node with `!important` word
|
|
* @returns {Array} `['important', sc]` where `sc` is optional whitespace
|
|
*/
|
|
function getImportant() {
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, token.importantEnd);
|
|
|
|
pos = token.importantEnd + 1;
|
|
|
|
return newNode(NodeType.ImportantType, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an include (`@include` or `@extend` directive).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkInclude(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkInclude1(i)) tokens[i].include_type = 1;else if (l = checkInclude2(i)) tokens[i].include_type = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin
|
|
* @returns {Array} `['include', x]`
|
|
*/
|
|
function getInclude() {
|
|
var type = tokens[pos].include_type;
|
|
|
|
if (type === 1) return getInclude1();
|
|
if (type === 2) return getInclude2();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkInclude1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkClass(i) || checkShash(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkClass(i) || checkShash(i) || checkSC(i)) i += l;else if (tokens[i].type === TokenType.GreaterThanSign) i++;else break;
|
|
}
|
|
|
|
if (l = checkArguments(i)) i += l;else return 0;
|
|
|
|
if (i < tokensLength && (l = checkSC(i))) i += l;
|
|
|
|
if (i < tokensLength && (l = checkImportant(i))) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `['include', x]`
|
|
*/
|
|
function getInclude1() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(checkClass(pos) ? getClass() : getShash());
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkClass(pos)) content.push(getClass());else if (checkShash(pos)) content.push(getShash());else if (checkSC(pos)) content = content.concat(getSC());else if (checkOperator(pos)) content.push(getOperator());else break;
|
|
}
|
|
|
|
content.push(getArguments());
|
|
|
|
content = content.concat(getSC());
|
|
|
|
if (checkImportant(pos)) content.push(getImportant());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkInclude2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkClass(i) || checkShash(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkClass(i) || checkShash(i) || checkSC(i)) i += l;else if (tokens[i].type === TokenType.GreaterThanSign) i++;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `['include', x]`
|
|
*/
|
|
function getInclude2() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(checkClass(pos) ? getClass() : getShash());
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkClass(pos)) content.push(getClass());else if (checkShash(pos)) content.push(getShash());else if (checkSC(pos)) content = content.concat(getSC());else if (checkOperator(pos)) content.push(getOperator());else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of LESS interpolated variable
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkInterpolatedVariable(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type !== TokenType.CommercialAt || !tokens[i + 1] || tokens[i + 1].type !== TokenType.LeftCurlyBracket) {
|
|
return 0;
|
|
}
|
|
|
|
i += 2;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return tokens[i].type === TokenType.RightCurlyBracket ? i - start + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with LESS interpolated variable
|
|
* @returns {Array} `['interpolatedVariable', x]`
|
|
*/
|
|
function getInterpolatedVariable() {
|
|
var type = NodeType.InterpolatedVariableType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `@{`:
|
|
pos += 2;
|
|
|
|
content.push(getIdent());
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `}`:
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkKeyframesBlock(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkKeyframesSelector(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getKeyframesBlock() {
|
|
var type = NodeType.RulesetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getKeyframesSelector(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkKeyframesBlocks(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlock(i)) i += l;else return 0;
|
|
|
|
while (tokens[i].type !== TokenType.RightCurlyBracket) {
|
|
if (l = checkSC(i)) i += l;else if (l = checkKeyframesBlock(i)) i += l;else break;
|
|
}
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getKeyframesBlocks() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var keyframesBlocksEnd = token.right;
|
|
var content = [];
|
|
|
|
// Skip `{`.
|
|
pos++;
|
|
|
|
while (pos < keyframesBlocksEnd) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkKeyframesBlock(pos)) content.push(getKeyframesBlock());
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `}`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a @keyframes rule.
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the @keyframes rule
|
|
*/
|
|
function checkKeyframesRule(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
var atruleName = joinValues2(i - l, l);
|
|
if (atruleName.toLowerCase().indexOf('keyframes') === -1) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getKeyframesRule() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdent(), getSC(), getKeyframesBlocks());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkKeyframesSelector(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) {
|
|
// Valid selectors are only `from` and `to`.
|
|
var selector = joinValues2(i, l);
|
|
if (selector !== 'from' && selector !== 'to') return 0;
|
|
|
|
i += l;
|
|
tokens[start].keyframesSelectorType = 1;
|
|
} else if (l = checkPercentage(i)) {
|
|
i += l;
|
|
tokens[start].keyframesSelectorType = 2;
|
|
} else {
|
|
return 0;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getKeyframesSelector() {
|
|
var keyframesSelectorType = NodeType.KeyframesSelectorType;
|
|
var selectorType = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (token.keyframesSelectorType === 1) {
|
|
content.push(getIdent());
|
|
} else {
|
|
content.push(getPercentage());
|
|
}
|
|
|
|
var keyframesSelector = newNode(keyframesSelectorType, content, line, column);
|
|
|
|
return newNode(selectorType, [keyframesSelector], line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a LESS mixin
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the mixin
|
|
*/
|
|
function checkMixin(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkMixin1(i)) tokens[i].mixin_type = 1;else if (l = checkMixin2(i)) tokens[i].mixin_type = 2;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getMixin() {
|
|
var type = tokens[pos].mixin_type;
|
|
|
|
if (type === 1) return getMixin1();
|
|
if (type === 2) return getMixin2();
|
|
}
|
|
|
|
function checkMixin1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkClass(i) || checkShash(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a mixin
|
|
* @returns {Array} `['mixin', x]`
|
|
*/
|
|
function getMixin1() {
|
|
var type = NodeType.MixinType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(checkClass(pos) ? getClass() : getShash());
|
|
|
|
content = content.concat(getSC());
|
|
|
|
if (checkArguments(pos)) content.push(getArguments());
|
|
|
|
content = content.concat(getSC());
|
|
|
|
if (checkBlock(pos)) content.push(getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a LESS mixin
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the mixin
|
|
*/
|
|
function checkMixin2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkClass(i) || checkShash(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a mixin
|
|
* @returns {Array} `['mixin', x]`
|
|
*/
|
|
function getMixin2() {
|
|
var type = NodeType.MixinType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(checkClass(pos) ? getClass() : getShash());
|
|
|
|
content = content.concat(getSC());
|
|
|
|
if (checkArguments(pos)) content.push(getArguments());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a namespace sign (`|`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is `|`, `0` if not
|
|
*/
|
|
function checkNamespace(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.VerticalLine ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a namespace sign
|
|
* @returns {Array} `['namespace']`
|
|
*/
|
|
function getNamespace() {
|
|
var type = NodeType.NamespaceType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '|';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkNmName2(i) {
|
|
if (tokens[i].type === TokenType.Identifier) return 1;else if (tokens[i].type !== TokenType.DecimalNumber) return 0;
|
|
|
|
i++;
|
|
|
|
return i < tokensLength && tokens[i].type === TokenType.Identifier ? 2 : 1;
|
|
}
|
|
|
|
/**
|
|
* @returns {String}
|
|
*/
|
|
function getNmName2() {
|
|
var s = tokens[pos].value;
|
|
|
|
if (tokens[pos++].type === TokenType.DecimalNumber && pos < tokensLength && tokens[pos].type === TokenType.Identifier) s += tokens[pos++].value;
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of number
|
|
*/
|
|
function checkNumber(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].number_l) return tokens[i].number_l;
|
|
|
|
// `10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && (!tokens[i + 1] || tokens[i + 1] && tokens[i + 1].type !== TokenType.FullStop)) {
|
|
tokens[i].number_l = 1;
|
|
return 1;
|
|
}
|
|
|
|
// `10.`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && (!tokens[i + 2] || tokens[i + 2].type !== TokenType.DecimalNumber)) {
|
|
tokens[i].number_l = 2;
|
|
return 2;
|
|
}
|
|
|
|
// `.10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.FullStop && tokens[i + 1].type === TokenType.DecimalNumber) {
|
|
tokens[i].number_l = 2;
|
|
return 2;
|
|
}
|
|
|
|
// `10.10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && tokens[i + 2] && tokens[i + 2].type === TokenType.DecimalNumber) {
|
|
tokens[i].number_l = 3;
|
|
return 3;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with number
|
|
* @returns {Array} `['number', x]` where `x` is a number converted
|
|
* to string.
|
|
*/
|
|
function getNumber() {
|
|
var type = NodeType.NumberType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var l = tokens[pos].number_l;
|
|
var content = '';
|
|
|
|
for (var j = 0; j < l; j++) {
|
|
content += tokens[pos + j].value;
|
|
}
|
|
|
|
pos += l;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is an operator (`/`, `,`, `:`, `=`, `>`, `<` or `*`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is an operator, otherwise `0`
|
|
*/
|
|
function checkOperator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
switch (tokens[i].type) {
|
|
case TokenType.Solidus:
|
|
case TokenType.Comma:
|
|
case TokenType.Colon:
|
|
case TokenType.EqualsSign:
|
|
case TokenType.LessThanSign:
|
|
case TokenType.GreaterThanSign:
|
|
case TokenType.Asterisk:
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an operator
|
|
* @returns {Array} `['operator', x]` where `x` is an operator converted
|
|
* to string.
|
|
*/
|
|
function getOperator() {
|
|
var type = NodeType.OperatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of text inside parentheses, e.g. `(1)`
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkParentheses(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
if (tokens[i].type === TokenType.LeftParenthesis) i++;else return 0;
|
|
|
|
if (i < right) {
|
|
var l = checkTsets(i);
|
|
if (l) i += l;else return 0;
|
|
}
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with text inside parentheses, e.g. `(1)`
|
|
* @return {Node}
|
|
*/
|
|
function getParentheses() {
|
|
var type = NodeType.ParenthesesType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var right = token.right;
|
|
var content = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
if (pos < right) {
|
|
content = getTsets();
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a parent selector (`&`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkParentSelector(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Ampersand ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a parent selector
|
|
* @returns {Array} `['parentSelector']`
|
|
*/
|
|
function getParentSelector() {
|
|
var type = NodeType.ParentSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '&';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkParentSelectorExtension(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkNumber(i) || checkPartialIdent(i)) i += l;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getParentSelectorExtension() {
|
|
var type = NodeType.ParentSelectorExtensionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkNumber(pos)) {
|
|
content.push(getNumber());
|
|
} else if (checkPartialIdent(pos)) {
|
|
content.push(getIdent());
|
|
} else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkParentSelectorWithExtension(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkParentSelector(i)) i += l;else return 0;
|
|
|
|
if (l = checkParentSelectorExtension(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getParentSelectorWithExtension() {
|
|
var content = [getParentSelector()];
|
|
|
|
if (checkParentSelectorExtension(pos)) content.push(getParentSelectorExtension());
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number with percent sign (e.g. `10%`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkPercentage(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkNumber(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Skip `%`.
|
|
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node of number with percent sign
|
|
* @returns {Array} `['percentage', ['number', x]]` where `x` is a number
|
|
* (without percent sign) converted to string.
|
|
*/
|
|
function getPercentage() {
|
|
var type = NodeType.PercentageType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getNumber()];
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `%`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkProgid(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (joinValues2(i, 6) === 'progid:DXImageTransform.Microsoft.') i += 6;else return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.LeftParenthesis) {
|
|
tokens[start].progid_end = tokens[i].right;
|
|
i = tokens[i].right + 1;
|
|
} else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getProgid() {
|
|
var type = NodeType.ProgidType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var progid_end = token.progid_end;
|
|
var content = joinValues(pos, progid_end);
|
|
|
|
pos = progid_end + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a property
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the property
|
|
*/
|
|
function checkProperty(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkVariable(i) || checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a property
|
|
* @returns {Array} `['property', x]`
|
|
*/
|
|
function getProperty() {
|
|
var type = NodeType.PropertyType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkVariable(pos)) {
|
|
content.push(getVariable());
|
|
} else {
|
|
content.push(getIdent());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a colon
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is a colon, otherwise `0`
|
|
*/
|
|
function checkPropertyDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Colon ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a colon
|
|
* @returns {Array} `['propertyDelim']`
|
|
*/
|
|
function getPropertyDelim() {
|
|
var type = NodeType.PropertyDelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ':';
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkPseudo(i) {
|
|
return checkPseudoe(i) || checkPseudoc(i);
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getPseudo() {
|
|
if (checkPseudoe(pos)) return getPseudoe();
|
|
if (checkPseudoc(pos)) return getPseudoc();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkPseudoe(i) {
|
|
var l = void 0;
|
|
|
|
// Check `::`
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.Colon || i + 1 >= tokensLength || tokens[i + 1].type !== TokenType.Colon) return 0;
|
|
|
|
if (l = checkPseudoElement1(i)) tokens[i].pseudoElementType = 1;else if (l = checkPseudoElement2(i)) tokens[i].pseudoElementType = 2;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Node}
|
|
*/
|
|
function getPseudoe() {
|
|
var childType = tokens[pos].pseudoElementType;
|
|
if (childType === 1) return getPseudoElement1();
|
|
if (childType === 2) return getPseudoElement2();
|
|
}
|
|
|
|
/**
|
|
* (1) `::slotted(selector)`
|
|
* (2) `::slotted(selector, selector)`
|
|
*/
|
|
function checkPseudoElement1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `::`.
|
|
i += 2;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* (1) `::slotted(selector)`
|
|
* (2) `::slotted(selector, selector)`
|
|
*/
|
|
function getPseudoElement1() {
|
|
var type = NodeType.PseudoeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `::`.
|
|
pos += 2;
|
|
|
|
content.push(getIdent());
|
|
|
|
{
|
|
var _type = NodeType.ArgumentsType;
|
|
var _token = tokens[pos];
|
|
var _line = _token.ln;
|
|
var _column = _token.col;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
|
|
|
|
var end = getLastPosition(selectorContent, _line, _column, 1);
|
|
var args = newNode(_type, selectorContent, _line, _column, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkPseudoElement2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `::`.
|
|
i += 2;
|
|
|
|
if (l = checkInterpolatedVariable(i) || checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getPseudoElement2() {
|
|
var type = NodeType.PseudoeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `::`.
|
|
pos += 2;
|
|
|
|
var content = [];
|
|
|
|
if (checkInterpolatedVariable(pos)) {
|
|
content.push(getInterpolatedVariable());
|
|
} else {
|
|
content.push(getIdent());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkPseudoc(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.Colon) return 0;
|
|
|
|
if (l = checkPseudoClass3(i)) tokens[i].pseudoClassType = 3;else if (l = checkPseudoClass4(i)) tokens[i].pseudoClassType = 4;else if (l = checkPseudoClass5(i)) tokens[i].pseudoClassType = 5;else if (l = checkPseudoClass1(i)) tokens[i].pseudoClassType = 1;else if (l = checkPseudoClass2(i)) tokens[i].pseudoClassType = 2;else if (l = checkPseudoClass6(i)) tokens[i].pseudoClassType = 6;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getPseudoc() {
|
|
var childType = tokens[pos].pseudoClassType;
|
|
if (childType === 1) return getPseudoClass1();
|
|
if (childType === 2) return getPseudoClass2();
|
|
if (childType === 3) return getPseudoClass3();
|
|
if (childType === 4) return getPseudoClass4();
|
|
if (childType === 5) return getPseudoClass5();
|
|
if (childType === 6) return getPseudoClass6();
|
|
}
|
|
|
|
/**
|
|
* (1) `:not(selector)`
|
|
* (2) `:extend(selector, selector)`
|
|
*/
|
|
function checkPseudoClass1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* (-) `:not(panda)`
|
|
*/
|
|
function getPseudoClass1() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content.push(getIdent());
|
|
|
|
{
|
|
var _type2 = NodeType.ArgumentsType;
|
|
var _token2 = tokens[pos];
|
|
var _line2 = _token2.ln;
|
|
var _column2 = _token2.col;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
|
|
|
|
var end = getLastPosition(selectorContent, _line2, _column2, 1);
|
|
var args = newNode(_type2, selectorContent, _line2, _column2, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `:nth-child(odd)`
|
|
* (2) `:nth-child(even)`
|
|
* (3) `:lang(de-DE)`
|
|
*/
|
|
function checkPseudoClass2(i) {
|
|
var start = i;
|
|
var l = 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass2() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content.push(getIdent());
|
|
|
|
{
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [].concat(getSC(), getIdent(), getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(-3n + 2)`
|
|
*/
|
|
function checkPseudoClass3(i) {
|
|
var start = i;
|
|
var l = 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].value === 'n') i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].value === '+' || tokens[i].value === '-') i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass3() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content.push(getIdent());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
|
|
{
|
|
var _l = tokens[pos].ln;
|
|
var _c = tokens[pos].col;
|
|
var _content = tokens[pos].value;
|
|
var ident = newNode(NodeType.IdentType, _content, _l, _c);
|
|
value.push(ident);
|
|
pos++;
|
|
}
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(-3n)`
|
|
*/
|
|
function checkPseudoClass4(i) {
|
|
var start = i;
|
|
var l = 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;
|
|
|
|
if (tokens[i].value === 'n') i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass4() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content.push(getIdent());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
if (checkIdent(pos)) value.push(getIdent());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(+8)`
|
|
*/
|
|
function checkPseudoClass5(i) {
|
|
var start = i;
|
|
var l = 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass5() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content.push(getIdent());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:checked`
|
|
*/
|
|
function checkPseudoClass6(i) {
|
|
var start = i;
|
|
var l = 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkInterpolatedVariable(i)) i += l;else if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass6() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
var ident = checkInterpolatedVariable(pos) ? getInterpolatedVariable() : getIdent();
|
|
content.push(ident);
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkRuleset(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getRuleset() {
|
|
var type = NodeType.RulesetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getSelectorsGroup(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is marked as a space (if it's a space or a tab
|
|
* or a line break).
|
|
* @param {Number} i
|
|
* @returns {Number} Number of spaces in a row starting with the given token.
|
|
*/
|
|
function checkS(i) {
|
|
return i < tokensLength && tokens[i].ws ? tokens[i].ws_last - i + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with spaces
|
|
* @returns {Array} `['s', x]` where `x` is a string containing spaces
|
|
*/
|
|
function getS() {
|
|
var type = NodeType.SType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, tokens[pos].ws_last);
|
|
|
|
pos = tokens[pos].ws_last + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a space or a comment.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Number of similar (space or comment) tokens
|
|
* in a row starting with the given token.
|
|
*/
|
|
function checkSC(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
var lsc = 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (!(l = checkS(i)) && !(l = checkCommentML(i)) && !(l = checkCommentSL(i))) break;
|
|
i += l;
|
|
lsc += l;
|
|
}
|
|
|
|
return lsc || 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with spaces and comments
|
|
* @returns {Array} Array containing nodes with spaces (if there are any)
|
|
* and nodes with comments (if there are any):
|
|
* `[['s', x]*, ['comment', y]*]` where `x` is a string of spaces
|
|
* and `y` is a comment's text (without `/*` and `* /`).
|
|
*/
|
|
function getSC() {
|
|
var sc = [];
|
|
|
|
if (pos >= tokensLength) return sc;
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkS(pos)) sc.push(getS());else if (checkCommentML(pos)) sc.push(getCommentML());else if (checkCommentSL(pos)) sc.push(getCommentSL());else break;
|
|
}
|
|
|
|
return sc;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
|
|
* a simple selector
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkShash(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.NumberSign) return 0;
|
|
|
|
if (l = checkInterpolatedVariable(i + 1) || checkIdent(i + 1)) return l + 1;else return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a hexadecimal number (e.g. `#fff`) inside a simple
|
|
* selector
|
|
* @returns {Array} `['shash', x]` where `x` is a hexadecimal number
|
|
* converted to string (without `#`, e.g. `fff`)
|
|
*/
|
|
function getShash() {
|
|
var type = NodeType.ShashType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `#`.
|
|
pos++;
|
|
|
|
if (checkInterpolatedVariable(pos)) content.push(getInterpolatedVariable());else content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a string (text wrapped in quotes)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is part of a string, `0` if not
|
|
*/
|
|
function checkString(i) {
|
|
if (i >= tokensLength) {
|
|
return 0;
|
|
}
|
|
|
|
if (tokens[i].type === TokenType.StringSQ || tokens[i].type === TokenType.StringDQ) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get string's node
|
|
* @returns {Array} `['string', x]` where `x` is a string (including
|
|
* quotes).
|
|
*/
|
|
function getString() {
|
|
var type = NodeType.StringType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Validate stylesheet: it should consist of any number (0 or more) of
|
|
* rulesets (sets of rules with selectors), @-rules, whitespaces or
|
|
* comments.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkStylesheet(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Check every token:
|
|
while (i < tokensLength) {
|
|
if (l = checkSC(i) || checkRuleset(i) || checkDeclaration(i) || checkDeclDelim(i) || checkAtrule(i) || checkMixin(i)) i += l;else throwError(i);
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `['stylesheet', x]` where `x` is all stylesheet's
|
|
* nodes.
|
|
*/
|
|
function getStylesheet() {
|
|
var type = NodeType.StylesheetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkRuleset(pos)) content.push(getRuleset());else if (checkDeclaration(pos)) content.push(getDeclaration());else if (checkDeclDelim(pos)) content.push(getDeclDelim());else if (checkAtrule(pos)) content.push(getAtrule());else if (checkMixin(pos)) content.push(getMixin());else throwError(pos);
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkTset(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkVhash(i)) tokens[i].tset_child = 1;else if (l = checkAny(i)) tokens[i].tset_child = 2;else if (l = checkSC(i)) tokens[i].tset_child = 3;else if (l = checkOperator(i)) tokens[i].tset_child = 4;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getTset() {
|
|
var childType = tokens[pos].tset_child;
|
|
|
|
if (childType === 1) return getVhash();
|
|
if (childType === 2) return getAny();
|
|
if (childType === 3) return getSC();
|
|
if (childType === 4) return getOperator();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkTsets(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (l = checkTset(i)) {
|
|
i += l;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getTsets() {
|
|
var content = [];
|
|
var t = void 0;
|
|
|
|
while (checkTset(pos)) {
|
|
t = getTset();
|
|
if (typeof t.content === 'string') content.push(t);else content = content.concat(t);
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is an unary (arithmetical) sign (`+` or `-`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is an unary sign, `0` if not
|
|
*/
|
|
function checkUnary(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.HyphenMinus || tokens[i].type === TokenType.PlusSign) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an unary (arithmetical) sign (`+` or `-`)
|
|
* @returns {Array} `['unary', x]` where `x` is an unary sign
|
|
* converted to string.
|
|
*/
|
|
function getUnary() {
|
|
var type = NodeType.OperatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a unicode range (single or multiple <urange> nodes)
|
|
* @param {number} i Token's index
|
|
* @return {number} Unicode range node's length
|
|
*/
|
|
function checkUnicodeRange(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkUrange(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
var comma = checkDelim(i + spaceBefore);
|
|
if (!comma) break;
|
|
|
|
var spaceAfter = checkSC(i + spaceBefore + comma);
|
|
if (l = checkUrange(i + spaceBefore + comma + spaceAfter)) {
|
|
i += spaceBefore + comma + spaceAfter + l;
|
|
} else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a unicode range node
|
|
* @return {Node}
|
|
*/
|
|
function getUnicodeRange() {
|
|
var type = NodeType.UnicodeRangeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkDelim(pos)) content.push(getDelim());else if (checkUrange(pos)) content.push(getUrange());else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is unit
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkUnit(i) {
|
|
var units = ['em', 'ex', 'ch', 'rem', 'vh', 'vw', 'vmin', 'vmax', 'px', 'mm', 'q', 'cm', 'in', 'pt', 'pc', 'deg', 'grad', 'rad', 'turn', 's', 'ms', 'Hz', 'kHz', 'dpi', 'dpcm', 'dppx'];
|
|
|
|
return units.indexOf(tokens[i].value) !== -1 ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get unit node of type ident
|
|
* @return {Node} An ident node containing the unit value
|
|
*/
|
|
function getUnit() {
|
|
var type = NodeType.IdentType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a u-range (part of a unicode-range)
|
|
* (1) `U+416`
|
|
* (2) `U+400-4ff`
|
|
* (3) `U+4??`
|
|
* @param {number} i Token's index
|
|
* @return {number} Urange node's length
|
|
*/
|
|
function checkUrange(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Check for unicode prefix (u+ or U+)
|
|
if (tokens[i].value === 'U' || tokens[i].value === 'u') i += 1;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].value === '+') i += 1;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = _checkUnicodeWildcard(i)) i += l;else break;
|
|
}
|
|
|
|
tokens[start].urangeEnd = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a u-range node (part of a unicode-range)
|
|
* @return {Node}
|
|
*/
|
|
function getUrange() {
|
|
var startPos = pos;
|
|
var type = NodeType.UrangeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content = joinValues(startPos, tokens[startPos].urangeEnd);
|
|
pos = tokens[startPos].urangeEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check for unicode wildcard characters `?`
|
|
* @param {number} i Token's index
|
|
* @return {number} Wildcard length
|
|
*/
|
|
function _checkUnicodeWildcard(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (tokens[i].type === TokenType.QuestionMark) i += 1;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of URI (e.g. `url('/css/styles.css')`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of URI
|
|
*/
|
|
function checkUri(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength || tokens[i++].value !== 'url' || i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
return tokens[i].right - start + 1;
|
|
}
|
|
|
|
/**
|
|
* Get node with URI
|
|
* @returns {Array} `['uri', x]` where `x` is URI's nodes (without `url`
|
|
* and braces, e.g. `['string', ''/css/styles.css'']`).
|
|
*/
|
|
function getUri() {
|
|
var startPos = pos;
|
|
var uriExcluding = {};
|
|
var uri = void 0;
|
|
var token = void 0;
|
|
var l = void 0;
|
|
var raw = void 0;
|
|
|
|
pos += 2;
|
|
|
|
uriExcluding[TokenType.Space] = 1;
|
|
uriExcluding[TokenType.Tab] = 1;
|
|
uriExcluding[TokenType.Newline] = 1;
|
|
uriExcluding[TokenType.LeftParenthesis] = 1;
|
|
uriExcluding[TokenType.RightParenthesis] = 1;
|
|
|
|
if (checkUri1(pos)) {
|
|
uri = [].concat(getSC()).concat([getString()]).concat(getSC());
|
|
} else {
|
|
uri = getSC();
|
|
l = checkExcluding(uriExcluding, pos);
|
|
token = tokens[pos];
|
|
raw = newNode(NodeType.RawType, joinValues(pos, pos + l), token.ln, token.col);
|
|
|
|
uri.push(raw);
|
|
|
|
pos += l + 1;
|
|
|
|
uri = uri.concat(getSC());
|
|
}
|
|
|
|
token = tokens[startPos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = getLastPosition(uri, line, column, 1);
|
|
pos++;
|
|
|
|
return newNode(NodeType.UriType, uri, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkUri1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type !== TokenType.StringDQ && tokens[i].type !== TokenType.StringSQ) {
|
|
return 0;
|
|
}
|
|
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a value
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the value
|
|
*/
|
|
function checkValue(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
var s = void 0;
|
|
var _i = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
s = checkSC(i);
|
|
_i = i + s;
|
|
|
|
if (l = _checkValue(_i)) i += l + s;
|
|
if (!l || checkBlock(_i)) break;
|
|
}
|
|
|
|
tokens[start].value_end = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getValue() {
|
|
var type = NodeType.ValueType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = tokens[pos].value_end;
|
|
var content = [];
|
|
var _pos = void 0;
|
|
var s = void 0;
|
|
|
|
while (pos < end) {
|
|
s = checkSC(pos);
|
|
_pos = pos + s;
|
|
|
|
if (!_checkValue(_pos)) break;
|
|
|
|
if (s) content = content.concat(getSC());
|
|
content.push(_getValue());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function _checkValue(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkEscapedString(i)) tokens[i].value_child = 1;else if (l = checkInterpolatedVariable(i)) tokens[i].value_child = 2;else if (l = checkVariable(i)) tokens[i].value_child = 3;else if (l = checkVhash(i)) tokens[i].value_child = 4;else if (l = checkBlock(i)) tokens[i].value_child = 5;else if (l = checkProgid(i)) tokens[i].value_child = 6;else if (l = checkAny(i)) tokens[i].value_child = 7;else if (l = checkAtkeyword(i)) tokens[i].value_child = 8;else if (l = checkOperator(i)) tokens[i].value_child = 9;else if (l = checkImportant(i)) tokens[i].value_child = 10;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function _getValue() {
|
|
var childType = tokens[pos].value_child;
|
|
if (childType === 1) return getEscapedString();
|
|
if (childType === 2) return getInterpolatedVariable();
|
|
if (childType === 3) return getVariable();
|
|
if (childType === 4) return getVhash();
|
|
if (childType === 5) return getBlock();
|
|
if (childType === 6) return getProgid();
|
|
if (childType === 7) return getAny();
|
|
if (childType === 8) return getAtkeyword();
|
|
if (childType === 9) return getOperator();
|
|
if (childType === 10) return getImportant();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of LESS variable
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the variable
|
|
*/
|
|
function checkVariable(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.CommercialAt) return 0;
|
|
|
|
if (tokens[i - 1] && tokens[i - 1].type === TokenType.CommercialAt && tokens[i - 2] && tokens[i - 2].type === TokenType.CommercialAt) return 0;
|
|
|
|
return (l = checkVariable(i + 1) || checkIdent(i + 1)) ? l + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a variable
|
|
* @returns {Array} `['variable', ['ident', x]]` where `x` is
|
|
* a variable name.
|
|
*/
|
|
function getVariable() {
|
|
var type = NodeType.VariableType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `$`.
|
|
pos++;
|
|
|
|
if (checkVariable(pos)) content.push(getVariable());else content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a variables list (e.g. `@rest...`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkVariablesList(i) {
|
|
var d = 0; // Number of dots
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkVariable(i)) i += l;else return 0;
|
|
|
|
while (tokens[i] && tokens[i].type === TokenType.FullStop) {
|
|
d++;
|
|
i++;
|
|
}
|
|
|
|
return d === 3 ? l + d : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a variables list
|
|
* @returns {Array} `['variableslist', ['variable', ['ident', x]]]` where
|
|
* `x` is a variable name.
|
|
*/
|
|
function getVariablesList() {
|
|
var type = NodeType.VariablesListType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getVariable()];
|
|
var end = getLastPosition(content, line, column, 3);
|
|
|
|
// Skip `...`.
|
|
pos += 3;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
|
|
* some value
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkVhash(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Skip `#`.
|
|
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
|
|
|
|
if (l = checkNmName2(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a hexadecimal number (e.g. `#fff`) inside some value
|
|
* @returns {Array} `['vhash', x]` where `x` is a hexadecimal number
|
|
* converted to string (without `#`, e.g. `'fff'`).
|
|
*/
|
|
function getVhash() {
|
|
var type = NodeType.VhashType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `#`.
|
|
pos++;
|
|
|
|
var content = getNmName2();
|
|
var end = getLastPosition(content, line, column + 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkSelectorsGroup(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSelector(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
var comma = checkDelim(i + spaceBefore);
|
|
if (!comma) break;
|
|
|
|
var spaceAfter = checkSC(i + spaceBefore + comma);
|
|
if (l = checkSelector(i + spaceBefore + comma + spaceAfter)) {
|
|
i += spaceBefore + comma + spaceAfter + l;
|
|
} else break;
|
|
}
|
|
|
|
tokens[start].selectorsGroupEnd = i;
|
|
return i - start;
|
|
}
|
|
|
|
function getSelectorsGroup() {
|
|
var selectorsGroup = [];
|
|
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
|
|
|
|
selectorsGroup.push(getSelector());
|
|
|
|
while (pos < selectorsGroupEnd) {
|
|
selectorsGroup = selectorsGroup.concat(getSC(), getDelim(), getSC(), getSelector());
|
|
}
|
|
|
|
return selectorsGroup;
|
|
}
|
|
|
|
function checkSelector(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkSelector1(i)) tokens[i].selectorType = 1;else if (l = checkSelector2(i)) tokens[i].selectorType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getSelector() {
|
|
var selectorType = tokens[pos].selectorType;
|
|
if (selectorType === 1) return getSelector1();else return getSelector2();
|
|
}
|
|
|
|
/**
|
|
* Checks for selector which starts with a compound selector.
|
|
*/
|
|
function checkSelector1(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkCompoundSelector(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var space = checkSC(i);
|
|
var comma = checkCombinator(i + space);
|
|
if (!space && !comma) break;
|
|
|
|
if (comma) {
|
|
i += space + comma;
|
|
space = checkSC(i);
|
|
}
|
|
|
|
if (l = checkCompoundSelector(i + space)) i += space + l;else break;
|
|
}
|
|
|
|
tokens[start].selectorEnd = i;
|
|
return i - start;
|
|
}
|
|
|
|
function getSelector1() {
|
|
var type = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var selectorEnd = token.selectorEnd;
|
|
var content = getCompoundSelector();
|
|
|
|
while (pos < selectorEnd) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Checks for a selector that starts with a combinator.
|
|
*/
|
|
function checkSelector2(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkCombinator(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
if (l = checkCompoundSelector(i + spaceBefore)) i += spaceBefore + l;else break;
|
|
|
|
var spaceAfter = checkSC(i);
|
|
var comma = checkCombinator(i + spaceAfter);
|
|
if (!spaceAfter && !comma) break;
|
|
if (comma) {
|
|
i += spaceAfter + comma;
|
|
}
|
|
}
|
|
|
|
tokens[start].selectorEnd = i;
|
|
return i - start;
|
|
}
|
|
|
|
function getSelector2() {
|
|
var type = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var selectorEnd = token.selectorEnd;
|
|
var content = [getCombinator()];
|
|
|
|
while (pos < selectorEnd) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkCompoundSelector(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkCompoundSelector1(i)) {
|
|
tokens[i].compoundSelectorType = 1;
|
|
} else if (l = checkCompoundSelector2(i)) {
|
|
tokens[i].compoundSelectorType = 2;
|
|
}
|
|
|
|
return l;
|
|
}
|
|
|
|
function getCompoundSelector() {
|
|
var type = tokens[pos].compoundSelectorType;
|
|
if (type === 1) return getCompoundSelector1();
|
|
if (type === 2) return getCompoundSelector2();
|
|
}
|
|
|
|
function checkCompoundSelector1(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkUniversalSelector(i) || checkTypeSelector(i) || checkParentSelectorWithExtension(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var _l2 = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i);
|
|
if (_l2) i += _l2;else break;
|
|
}
|
|
|
|
tokens[start].compoundSelectorEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getCompoundSelector1() {
|
|
var sequence = [];
|
|
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
|
|
|
|
if (checkUniversalSelector(pos)) sequence.push(getUniversalSelector());else if (checkTypeSelector(pos)) sequence.push(getTypeSelector());else if (checkParentSelectorWithExtension(pos)) sequence = sequence.concat(getParentSelectorWithExtension());
|
|
|
|
while (pos < compoundSelectorEnd) {
|
|
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());
|
|
}
|
|
|
|
return sequence;
|
|
}
|
|
|
|
function checkCompoundSelector2(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
|
|
while (i < tokensLength) {
|
|
var l = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i);
|
|
if (l) i += l;else break;
|
|
}
|
|
|
|
tokens[start].compoundSelectorEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getCompoundSelector2() {
|
|
var sequence = [];
|
|
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
|
|
|
|
while (pos < compoundSelectorEnd) {
|
|
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());
|
|
}
|
|
|
|
return sequence;
|
|
}
|
|
|
|
function checkUniversalSelector(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.Asterisk) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getUniversalSelector() {
|
|
var type = NodeType.UniversalSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var end = void 0;
|
|
|
|
if (checkNamePrefix(pos)) {
|
|
content.push(getNamePrefix());
|
|
end = getLastPosition(content, line, column, 1);
|
|
}
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkTypeSelector(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getTypeSelector() {
|
|
var type = NodeType.TypeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkNamePrefix(pos)) content.push(getNamePrefix());
|
|
|
|
content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeSelector(i) {
|
|
var l = void 0;
|
|
if (l = checkAttributeSelector1(i)) tokens[i].attributeSelectorType = 1;else if (l = checkAttributeSelector2(i)) tokens[i].attributeSelectorType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getAttributeSelector() {
|
|
var type = tokens[pos].attributeSelectorType;
|
|
if (type === 1) return getAttributeSelector1();else return getAttributeSelector2();
|
|
}
|
|
|
|
/**
|
|
* (1) `[panda=nani]`
|
|
* (2) `[panda='nani']`
|
|
* (3) `[panda='nani' i]`
|
|
*
|
|
*/
|
|
function checkAttributeSelector1(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeName(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeMatch(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeValue(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeFlags(i)) {
|
|
i += l;
|
|
if (l = checkSC(i)) i += l;
|
|
}
|
|
|
|
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeSelector1() {
|
|
var type = NodeType.AttributeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC(), getAttributeName(), getSC(), getAttributeMatch(), getSC(), getAttributeValue(), getSC());
|
|
|
|
if (checkAttributeFlags(pos)) {
|
|
content.push(getAttributeFlags());
|
|
content = content.concat(getSC());
|
|
}
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* (1) `[panda]`
|
|
*/
|
|
function checkAttributeSelector2(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeName(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeSelector2() {
|
|
var type = NodeType.AttributeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC(), getAttributeName(), getSC());
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkAttributeName(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeName() {
|
|
var type = NodeType.AttributeNameType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkNamePrefix(pos)) content.push(getNamePrefix());
|
|
content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeMatch(i) {
|
|
var l = void 0;
|
|
if (l = checkAttributeMatch1(i)) tokens[i].attributeMatchType = 1;else if (l = checkAttributeMatch2(i)) tokens[i].attributeMatchType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getAttributeMatch() {
|
|
var type = tokens[pos].attributeMatchType;
|
|
if (type === 1) return getAttributeMatch1();else return getAttributeMatch2();
|
|
}
|
|
|
|
function checkAttributeMatch1(i) {
|
|
var start = i;
|
|
|
|
var type = tokens[i].type;
|
|
if (type === TokenType.Tilde || type === TokenType.VerticalLine || type === TokenType.CircumflexAccent || type === TokenType.DollarSign || type === TokenType.Asterisk) i++;else return 0;
|
|
|
|
if (tokens[i].type === TokenType.EqualsSign) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeMatch1() {
|
|
var type = NodeType.AttributeMatchType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos].value + tokens[pos + 1].value;
|
|
pos += 2;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeMatch2(i) {
|
|
if (tokens[i].type === TokenType.EqualsSign) return 1;else return 0;
|
|
}
|
|
|
|
function getAttributeMatch2() {
|
|
var type = NodeType.AttributeMatchType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '=';
|
|
|
|
pos++;
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeValue(i) {
|
|
return checkString(i) || checkIdent(i);
|
|
}
|
|
|
|
function getAttributeValue() {
|
|
var type = NodeType.AttributeValueType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkString(pos)) content.push(getString());else content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeFlags(i) {
|
|
return checkIdent(i);
|
|
}
|
|
|
|
function getAttributeFlags() {
|
|
var type = NodeType.AttributeFlagsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkNamePrefix(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkNamePrefix1(i)) tokens[i].namePrefixType = 1;else if (l = checkNamePrefix2(i)) tokens[i].namePrefixType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getNamePrefix() {
|
|
var type = tokens[pos].namePrefixType;
|
|
if (type === 1) return getNamePrefix1();else return getNamePrefix2();
|
|
}
|
|
|
|
/**
|
|
* (1) `panda|`
|
|
* (2) `panda<comment>|`
|
|
*/
|
|
function checkNamePrefix1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamespacePrefix(i)) i += l;else return 0;
|
|
|
|
if (l = checkCommentML(i)) i += l;
|
|
|
|
if (l = checkNamespaceSeparator(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getNamePrefix1() {
|
|
var type = NodeType.NamePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(getNamespacePrefix());
|
|
|
|
if (checkCommentML(pos)) content.push(getCommentML());
|
|
|
|
content.push(getNamespaceSeparator());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `|`
|
|
*/
|
|
function checkNamePrefix2(i) {
|
|
return checkNamespaceSeparator(i);
|
|
}
|
|
|
|
function getNamePrefix2() {
|
|
var type = NodeType.NamePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getNamespaceSeparator()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `*`
|
|
* (2) `panda`
|
|
*/
|
|
function checkNamespacePrefix(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
|
|
if (tokens[i].type === TokenType.Asterisk) return 1;else if (l = checkIdent(i)) return l;else return 0;
|
|
}
|
|
|
|
function getNamespacePrefix() {
|
|
var type = NodeType.NamespacePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (token.type === TokenType.Asterisk) {
|
|
var asteriskNode = newNode(NodeType.IdentType, '*', line, column);
|
|
content.push(asteriskNode);
|
|
pos++;
|
|
} else if (checkIdent(pos)) content.push(getIdent());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `|`
|
|
*/
|
|
function checkNamespaceSeparator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type !== TokenType.VerticalLine) return 0;
|
|
|
|
// Return false if `|=` - [attr|=value]
|
|
if (tokens[i + 1] && tokens[i + 1].type === TokenType.EqualsSign) return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
function getNamespaceSeparator() {
|
|
var type = NodeType.NamespaceSeparatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '|';
|
|
|
|
pos++;
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
module.exports = function (_tokens, context) {
|
|
tokens = _tokens;
|
|
tokensLength = tokens.length;
|
|
pos = 0;
|
|
|
|
return contexts[context]();
|
|
};
|
|
|
|
/***/ }),
|
|
/* 20 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
module.exports = function (css, tabSize) {
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
var tokens = [];
|
|
var urlMode = false;
|
|
var c = void 0; // Current character
|
|
var cn = void 0; // Next character
|
|
var pos = 0;
|
|
var tn = 0;
|
|
var ln = 1;
|
|
var col = 1;
|
|
|
|
var Punctuation = {
|
|
' ': TokenType.Space,
|
|
'\n': TokenType.Newline,
|
|
'\r': TokenType.Newline,
|
|
'\t': TokenType.Tab,
|
|
'!': TokenType.ExclamationMark,
|
|
'"': TokenType.QuotationMark,
|
|
'#': TokenType.NumberSign,
|
|
'$': TokenType.DollarSign,
|
|
'%': TokenType.PercentSign,
|
|
'&': TokenType.Ampersand,
|
|
'\'': TokenType.Apostrophe,
|
|
'(': TokenType.LeftParenthesis,
|
|
')': TokenType.RightParenthesis,
|
|
'*': TokenType.Asterisk,
|
|
'+': TokenType.PlusSign,
|
|
',': TokenType.Comma,
|
|
'-': TokenType.HyphenMinus,
|
|
'.': TokenType.FullStop,
|
|
'/': TokenType.Solidus,
|
|
':': TokenType.Colon,
|
|
';': TokenType.Semicolon,
|
|
'<': TokenType.LessThanSign,
|
|
'=': TokenType.EqualsSign,
|
|
'>': TokenType.GreaterThanSign,
|
|
'?': TokenType.QuestionMark,
|
|
'@': TokenType.CommercialAt,
|
|
'[': TokenType.LeftSquareBracket,
|
|
']': TokenType.RightSquareBracket,
|
|
'^': TokenType.CircumflexAccent,
|
|
'_': TokenType.LowLine,
|
|
'{': TokenType.LeftCurlyBracket,
|
|
'|': TokenType.VerticalLine,
|
|
'}': TokenType.RightCurlyBracket,
|
|
'~': TokenType.Tilde
|
|
};
|
|
|
|
/**
|
|
* Add a token to the token list
|
|
* @param {string} type
|
|
* @param {string} value
|
|
*/
|
|
function pushToken(type, value, column) {
|
|
tokens.push({
|
|
tn: tn++,
|
|
ln: ln,
|
|
col: column,
|
|
type: type,
|
|
value: value
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Check if a character is a decimal digit
|
|
* @param {string} c Character
|
|
* @returns {boolean}
|
|
*/
|
|
function isDecimalDigit(c) {
|
|
return '0123456789'.indexOf(c) >= 0;
|
|
}
|
|
|
|
/**
|
|
* Parse spaces
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseSpaces(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a non-space character:
|
|
for (; pos < css.length; pos++) {
|
|
if (css.charAt(pos) !== ' ') break;
|
|
}
|
|
|
|
// Add a substring containing only spaces to tokens:
|
|
pushToken(TokenType.Space, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse a string within quotes
|
|
* @param {string} css Unparsed part of CSS string
|
|
* @param {string} q Quote (either `'` or `"`)
|
|
*/
|
|
function parseString(css, q) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a matching quote:
|
|
for (pos++; pos < css.length; pos++) {
|
|
// Skip escaped quotes:
|
|
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) === q) break;
|
|
}
|
|
|
|
// Add the string (including quotes) to tokens:
|
|
var type = q === '"' ? TokenType.StringDQ : TokenType.StringSQ;
|
|
pushToken(type, css.substring(start, pos + 1), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse numbers
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseDecimalNumber(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a character that's not a digit:
|
|
for (; pos < css.length; pos++) {
|
|
if (!isDecimalDigit(css.charAt(pos))) break;
|
|
}
|
|
|
|
// Add the number to tokens:
|
|
pushToken(TokenType.DecimalNumber, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse identifier
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseIdentifier(css) {
|
|
var start = pos;
|
|
|
|
// Skip all opening slashes:
|
|
while (css.charAt(pos) === '/') {
|
|
pos++;
|
|
} // Read the string until we meet a punctuation mark:
|
|
for (; pos < css.length; pos++) {
|
|
// Skip all '\':
|
|
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) in Punctuation) break;
|
|
}
|
|
|
|
var ident = css.substring(start, pos--);
|
|
|
|
// Enter url mode if parsed substring is `url`:
|
|
if (!urlMode && ident === 'url' && css.charAt(pos + 1) === '(') {
|
|
urlMode = true;
|
|
}
|
|
|
|
// Add identifier to tokens:
|
|
pushToken(TokenType.Identifier, ident, col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse a multiline comment
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseMLComment(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet `*/`.
|
|
// Since we already know first 2 characters (`/*`), start reading
|
|
// from `pos + 2`:
|
|
for (pos = pos + 2; pos < css.length; pos++) {
|
|
if (css.charAt(pos) === '*' && css.charAt(pos + 1) === '/') {
|
|
pos++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Add full comment (including `/*` and `*/`) to the list of tokens:
|
|
var comment = css.substring(start, pos + 1);
|
|
pushToken(TokenType.CommentML, comment, col);
|
|
|
|
var newlines = comment.split('\n');
|
|
if (newlines.length > 1) {
|
|
ln += newlines.length - 1;
|
|
col = newlines[newlines.length - 1].length;
|
|
} else {
|
|
col += pos - start;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parse a single line comment
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseSLComment(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet line break.
|
|
// Since we already know first 2 characters (`//`), start reading
|
|
// from `pos + 2`:
|
|
for (pos += 2; pos < css.length; pos++) {
|
|
if (css.charAt(pos) === '\n' || css.charAt(pos) === '\r') {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Add comment (including `//` and line break) to the list of tokens:
|
|
pushToken(TokenType.CommentSL, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Convert a CSS string to a list of tokens
|
|
* @param {string} css CSS string
|
|
* @returns {Array} List of tokens
|
|
* @private
|
|
*/
|
|
function getTokens(css) {
|
|
// Parse string, character by character:
|
|
for (pos = 0; pos < css.length; col++, pos++) {
|
|
c = css.charAt(pos);
|
|
cn = css.charAt(pos + 1);
|
|
|
|
// If we meet `/*`, it's a start of a multiline comment.
|
|
// Parse following characters as a multiline comment:
|
|
if (c === '/' && cn === '*') {
|
|
parseMLComment(css);
|
|
}
|
|
|
|
// If we meet `//` and it is not a part of url:
|
|
else if (!urlMode && c === '/' && cn === '/') {
|
|
// If we're currently inside a block, treat `//` as a start
|
|
// of identifier. Else treat `//` as a start of a single-line
|
|
// comment:
|
|
parseSLComment(css);
|
|
}
|
|
|
|
// If current character is a double or single quote, it's a start
|
|
// of a string:
|
|
else if (c === '"' || c === "'") {
|
|
parseString(css, c);
|
|
}
|
|
|
|
// If current character is a space:
|
|
else if (c === ' ') {
|
|
parseSpaces(css);
|
|
}
|
|
|
|
// If current character is a punctuation mark:
|
|
else if (c in Punctuation) {
|
|
// Add it to the list of tokens:
|
|
pushToken(Punctuation[c], c, col);
|
|
if (c === '\n' || c === '\r') {
|
|
ln++;
|
|
col = 0;
|
|
} // Go to next line
|
|
if (c === ')') urlMode = false; // Exit url mode
|
|
else if (c === '\t' && tabSize > 1) col += tabSize - 1;
|
|
}
|
|
|
|
// If current character is a decimal digit:
|
|
else if (isDecimalDigit(c)) {
|
|
parseDecimalNumber(css);
|
|
}
|
|
|
|
// If current character is anything else:
|
|
else {
|
|
parseIdentifier(css);
|
|
}
|
|
}
|
|
|
|
return tokens;
|
|
}
|
|
|
|
return getTokens(css);
|
|
};
|
|
|
|
/***/ }),
|
|
/* 21 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
exports.__esModule = true;
|
|
exports.default = {
|
|
mark: __webpack_require__(22),
|
|
parse: __webpack_require__(23),
|
|
stringify: __webpack_require__(5),
|
|
tokenizer: __webpack_require__(24)
|
|
};
|
|
module.exports = exports['default'];
|
|
|
|
/***/ }),
|
|
/* 22 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
module.exports = function () {
|
|
/**
|
|
* Mark whitespaces and comments
|
|
*/
|
|
function markSC(tokens) {
|
|
var tokensLength = tokens.length;
|
|
var ws = -1; // Flag for whitespaces
|
|
var sc = -1; // Flag for whitespaces and comments
|
|
var t = void 0; // Current token
|
|
|
|
// For every token in the token list, mark spaces and line breaks
|
|
// as spaces (set both `ws` and `sc` flags). Mark multiline comments
|
|
// with `sc` flag.
|
|
// If there are several spaces or tabs or line breaks or multiline
|
|
// comments in a row, group them: take the last one's index number
|
|
// and save it to the first token in the group as a reference:
|
|
// e.g., `ws_last = 7` for a group of whitespaces or `sc_last = 9`
|
|
// for a group of whitespaces and comments.
|
|
for (var i = 0; i < tokensLength; i++) {
|
|
t = tokens[i];
|
|
switch (t.type) {
|
|
case TokenType.Space:
|
|
case TokenType.Tab:
|
|
t.ws = true;
|
|
t.sc = true;
|
|
|
|
if (ws === -1) ws = i;
|
|
if (sc === -1) sc = i;
|
|
|
|
break;
|
|
case TokenType.Newline:
|
|
t.ws = true;
|
|
t.sc = true;
|
|
|
|
ws = ws === -1 ? i : ws;
|
|
sc = sc === -1 ? i : ws;
|
|
|
|
tokens[ws].ws_last = i - 1;
|
|
tokens[sc].sc_last = i - 1;
|
|
tokens[i].ws_last = i;
|
|
tokens[i].sc_last = i;
|
|
|
|
ws = -1;
|
|
sc = -1;
|
|
|
|
break;
|
|
case TokenType.CommentML:
|
|
case TokenType.CommentSL:
|
|
if (ws !== -1) {
|
|
tokens[ws].ws_last = i - 1;
|
|
ws = -1;
|
|
}
|
|
|
|
t.sc = true;
|
|
|
|
break;
|
|
default:
|
|
if (ws !== -1) {
|
|
tokens[ws].ws_last = i - 1;
|
|
ws = -1;
|
|
}
|
|
|
|
if (sc !== -1) {
|
|
tokens[sc].sc_last = i - 1;
|
|
sc = -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ws !== -1) tokens[ws].ws_last = i - 1;
|
|
if (sc !== -1) tokens[sc].sc_last = i - 1;
|
|
}
|
|
|
|
/**
|
|
* Pair brackets
|
|
*/
|
|
function markBrackets(tokens) {
|
|
var tokensLength = tokens.length;
|
|
var ps = []; // Parentheses
|
|
var sbs = []; // Square brackets
|
|
var cbs = []; // Curly brackets
|
|
var t = void 0; // Current token
|
|
|
|
// For every token in the token list, if we meet an opening (left)
|
|
// bracket, push its index number to a corresponding array.
|
|
// If we then meet a closing (right) bracket, look at the corresponding
|
|
// array. If there are any elements (records about previously met
|
|
// left brackets), take a token of the last left bracket (take
|
|
// the last index number from the array and find a token with
|
|
// this index number) and save right bracket's index as a reference:
|
|
for (var i = 0; i < tokensLength; i++) {
|
|
t = tokens[i];
|
|
switch (t.type) {
|
|
case TokenType.LeftParenthesis:
|
|
ps.push(i);
|
|
break;
|
|
case TokenType.RightParenthesis:
|
|
if (ps.length) {
|
|
t.left = ps.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
break;
|
|
case TokenType.LeftSquareBracket:
|
|
sbs.push(i);
|
|
break;
|
|
case TokenType.RightSquareBracket:
|
|
if (sbs.length) {
|
|
t.left = sbs.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
break;
|
|
case TokenType.LeftCurlyBracket:
|
|
cbs.push(i);
|
|
break;
|
|
case TokenType.RightCurlyBracket:
|
|
if (cbs.length) {
|
|
t.left = cbs.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
function markBlocks(tokens) {
|
|
var i = 0;
|
|
var l = tokens.length;
|
|
var lines = [];
|
|
var whitespaceOnlyLines = [];
|
|
|
|
for (i = 0; i < l; i++) {
|
|
var lineStart = i;
|
|
var currentLineIndent = 0;
|
|
|
|
// Get all spaces.
|
|
while (i < l && (tokens[i].type === TokenType.Space || tokens[i].type === TokenType.Tab)) {
|
|
currentLineIndent += tokens[i].value.length;
|
|
i++;
|
|
}
|
|
|
|
lines.push([lineStart, currentLineIndent]);
|
|
|
|
var x = i;
|
|
while (i < l && tokens[i].type !== TokenType.Newline) {
|
|
i++;
|
|
}
|
|
|
|
if (x === i) {
|
|
whitespaceOnlyLines.push(lines.length - 1);
|
|
}
|
|
}
|
|
|
|
var levels = [0];
|
|
var blockStarts = [];
|
|
|
|
for (i = 0; i < lines.length; i++) {
|
|
var line = lines[i];
|
|
var token = line[0];
|
|
var indent = line[1];
|
|
var lastLevel = levels[levels.length - 1];
|
|
|
|
if (indent > lastLevel) {
|
|
blockStarts.push(token);
|
|
levels.push(indent);
|
|
} else {
|
|
// Check if line is whitespace-only.
|
|
var p = i;
|
|
|
|
while (true) {
|
|
if (whitespaceOnlyLines.indexOf(p) === -1) break;
|
|
p++;
|
|
}
|
|
|
|
if (i === p && indent === lastLevel) continue;
|
|
|
|
if (!lines[p]) {
|
|
continue;
|
|
}
|
|
|
|
indent = lines[p][1];
|
|
|
|
if (indent === lastLevel) {
|
|
i = p;
|
|
continue;
|
|
}
|
|
|
|
if (indent > lastLevel) {
|
|
blockStarts.push(token);
|
|
levels.push(lines[p][1]);
|
|
i = p;
|
|
continue;
|
|
}
|
|
|
|
while (true) {
|
|
var _lastLevel = levels.pop();
|
|
if (indent < _lastLevel) {
|
|
var start = blockStarts.pop();
|
|
tokens[start].block_end = token - 1;
|
|
} else {
|
|
levels.push(indent);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
blockStarts.forEach(function (start) {
|
|
tokens[start].block_end = tokens.length - 1;
|
|
});
|
|
}
|
|
|
|
return function (tokens) {
|
|
markBrackets(tokens);
|
|
markSC(tokens);
|
|
markBlocks(tokens);
|
|
};
|
|
}();
|
|
|
|
/***/ }),
|
|
/* 23 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var Node = __webpack_require__(1);
|
|
var NodeType = __webpack_require__(15);
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
var tokens = void 0;
|
|
var tokensLength = void 0;
|
|
var pos = void 0;
|
|
|
|
var contexts = {
|
|
'arguments': function _arguments() {
|
|
return checkArguments(pos) && getArguments();
|
|
},
|
|
'atkeyword': function atkeyword() {
|
|
return checkAtkeyword(pos) && getAtkeyword();
|
|
},
|
|
'atrule': function atrule() {
|
|
return checkAtrule(pos) && getAtrule();
|
|
},
|
|
'attributeSelector': function attributeSelector() {
|
|
return checkAttributeSelector(pos) && getAttributeSelector();
|
|
},
|
|
'block': function block() {
|
|
return checkBlock(pos) && getBlock();
|
|
},
|
|
'brackets': function brackets() {
|
|
return checkBrackets(pos) && getBrackets();
|
|
},
|
|
'class': function _class() {
|
|
return checkClass(pos) && getClass();
|
|
},
|
|
'combinator': function combinator() {
|
|
return checkCombinator(pos) && getCombinator();
|
|
},
|
|
'commentML': function commentML() {
|
|
return checkCommentML(pos) && getCommentML();
|
|
},
|
|
'commentSL': function commentSL() {
|
|
return checkCommentSL(pos) && getCommentSL();
|
|
},
|
|
'condition': function condition() {
|
|
return checkCondition(pos) && getCondition();
|
|
},
|
|
'conditionalStatement': function conditionalStatement() {
|
|
return checkConditionalStatement(pos) && getConditionalStatement();
|
|
},
|
|
'declaration': function declaration() {
|
|
return checkDeclaration(pos) && getDeclaration();
|
|
},
|
|
'declDelim': function declDelim() {
|
|
return checkDeclDelim(pos) && getDeclDelim();
|
|
},
|
|
'default': function _default() {
|
|
return checkDefault(pos) && getDefault();
|
|
},
|
|
'delim': function delim() {
|
|
return checkDelim(pos) && getDelim();
|
|
},
|
|
'dimension': function dimension() {
|
|
return checkDimension(pos) && getDimension();
|
|
},
|
|
'expression': function expression() {
|
|
return checkExpression(pos) && getExpression();
|
|
},
|
|
'extend': function extend() {
|
|
return checkExtend(pos) && getExtend();
|
|
},
|
|
'function': function _function() {
|
|
return checkFunction(pos) && getFunction();
|
|
},
|
|
'global': function global() {
|
|
return checkGlobal(pos) && getGlobal();
|
|
},
|
|
'ident': function ident() {
|
|
return checkIdent(pos) && getIdent();
|
|
},
|
|
'important': function important() {
|
|
return checkImportant(pos) && getImportant();
|
|
},
|
|
'include': function include() {
|
|
return checkInclude(pos) && getInclude();
|
|
},
|
|
'interpolation': function interpolation() {
|
|
return checkInterpolation(pos) && getInterpolation();
|
|
},
|
|
'loop': function loop() {
|
|
return checkLoop(pos) && getLoop();
|
|
},
|
|
'mixin': function mixin() {
|
|
return checkMixin(pos) && getMixin();
|
|
},
|
|
'namespace': function namespace() {
|
|
return checkNamespace(pos) && getNamespace();
|
|
},
|
|
'number': function number() {
|
|
return checkNumber(pos) && getNumber();
|
|
},
|
|
'operator': function operator() {
|
|
return checkOperator(pos) && getOperator();
|
|
},
|
|
'optional': function optional() {
|
|
return checkOptional(pos) && getOptional();
|
|
},
|
|
'parentheses': function parentheses() {
|
|
return checkParentheses(pos) && getParentheses();
|
|
},
|
|
'parentselector': function parentselector() {
|
|
return checkParentSelector(pos) && getParentSelector();
|
|
},
|
|
'percentage': function percentage() {
|
|
return checkPercentage(pos) && getPercentage();
|
|
},
|
|
'placeholder': function placeholder() {
|
|
return checkPlaceholder(pos) && getPlaceholder();
|
|
},
|
|
'progid': function progid() {
|
|
return checkProgid(pos) && getProgid();
|
|
},
|
|
'property': function property() {
|
|
return checkProperty(pos) && getProperty();
|
|
},
|
|
'propertyDelim': function propertyDelim() {
|
|
return checkPropertyDelim(pos) && getPropertyDelim();
|
|
},
|
|
'pseudoc': function pseudoc() {
|
|
return checkPseudoc(pos) && getPseudoc();
|
|
},
|
|
'pseudoe': function pseudoe() {
|
|
return checkPseudoe(pos) && getPseudoe();
|
|
},
|
|
'ruleset': function ruleset() {
|
|
return checkRuleset(pos) && getRuleset();
|
|
},
|
|
's': function s() {
|
|
return checkS(pos) && getS();
|
|
},
|
|
'selector': function selector() {
|
|
return checkSelector(pos) && getSelector();
|
|
},
|
|
'shash': function shash() {
|
|
return checkShash(pos) && getShash();
|
|
},
|
|
'string': function string() {
|
|
return checkString(pos) && getString();
|
|
},
|
|
'stylesheet': function stylesheet() {
|
|
return checkStylesheet(pos) && getStylesheet();
|
|
},
|
|
'typeSelector': function typeSelector() {
|
|
return checkTypeSelector(pos) && getTypeSelector();
|
|
},
|
|
'unary': function unary() {
|
|
return checkUnary(pos) && getUnary();
|
|
},
|
|
'unicodeRange': function unicodeRange() {
|
|
return checkUnicodeRange(pos) && getUnicodeRange();
|
|
},
|
|
'universalSelector': function universalSelector() {
|
|
return checkUniversalSelector(pos) && getUniversalSelector();
|
|
},
|
|
'urange': function urange() {
|
|
return checkUrange(pos) && getUrange();
|
|
},
|
|
'uri': function uri() {
|
|
return checkUri(pos) && getUri();
|
|
},
|
|
'value': function value() {
|
|
return checkValue(pos) && getValue();
|
|
},
|
|
'variable': function variable() {
|
|
return checkVariable(pos) && getVariable();
|
|
},
|
|
'variableslist': function variableslist() {
|
|
return checkVariablesList(pos) && getVariablesList();
|
|
},
|
|
'vhash': function vhash() {
|
|
return checkVhash(pos) && getVhash();
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Stops parsing and display error.
|
|
*
|
|
* @param {number=} i Token's index number
|
|
*/
|
|
function throwError(i) {
|
|
var ln = tokens[i].ln;
|
|
|
|
throw { line: ln, syntax: 'sass' };
|
|
}
|
|
|
|
/**
|
|
* @param {number} start
|
|
* @param {number} finish
|
|
* @return {string}
|
|
*/
|
|
function joinValues(start, finish) {
|
|
var s = '';
|
|
|
|
for (var i = start; i < finish + 1; i++) {
|
|
s += tokens[i].value;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* @param {number} start
|
|
* @param {number} num
|
|
* @return {string}
|
|
*/
|
|
function joinValues2(start, num) {
|
|
if (start + num - 1 >= tokensLength) return;
|
|
|
|
var s = '';
|
|
|
|
for (var i = 0; i < num; i++) {
|
|
s += tokens[start + i].value;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* @param {string|!Array} content
|
|
* @param {number} line
|
|
* @param {number} column
|
|
* @param {number} colOffset
|
|
*/
|
|
function getLastPosition(content, line, column, colOffset) {
|
|
return typeof content === 'string' ? getLastPositionForString(content, line, column, colOffset) : getLastPositionForArray(content, line, column, colOffset);
|
|
}
|
|
|
|
/**
|
|
* @param {string} content
|
|
* @param {number} line
|
|
* @param {number} column
|
|
* @param {number} colOffset
|
|
*/
|
|
function getLastPositionForString(content, line, column, colOffset) {
|
|
var position = [];
|
|
|
|
if (!content) {
|
|
position = [line, column];
|
|
if (colOffset) position[1] += colOffset - 1;
|
|
return position;
|
|
}
|
|
|
|
var lastLinebreak = content.lastIndexOf('\n');
|
|
var endsWithLinebreak = lastLinebreak === content.length - 1;
|
|
var splitContent = content.split('\n');
|
|
var linebreaksCount = splitContent.length - 1;
|
|
var prevLinebreak = linebreaksCount === 0 || linebreaksCount === 1 ? -1 : content.length - splitContent[linebreaksCount - 1].length - 2;
|
|
|
|
// Line:
|
|
var offset = endsWithLinebreak ? linebreaksCount - 1 : linebreaksCount;
|
|
position[0] = line + offset;
|
|
|
|
// Column:
|
|
if (endsWithLinebreak) {
|
|
offset = prevLinebreak !== -1 ? content.length - prevLinebreak : content.length - 1;
|
|
} else {
|
|
offset = linebreaksCount !== 0 ? content.length - lastLinebreak - column - 1 : content.length - 1;
|
|
}
|
|
position[1] = column + offset;
|
|
|
|
if (!colOffset) return position;
|
|
|
|
if (endsWithLinebreak) {
|
|
position[0]++;
|
|
position[1] = colOffset;
|
|
} else {
|
|
position[1] += colOffset;
|
|
}
|
|
|
|
return position;
|
|
}
|
|
|
|
/**
|
|
* @param {!Array} content
|
|
* @param {number} line
|
|
* @param {number} column
|
|
* @param {number} colOffset
|
|
*/
|
|
function getLastPositionForArray(content, line, column, colOffset) {
|
|
var position = void 0;
|
|
|
|
if (content.length === 0) {
|
|
position = [line, column];
|
|
} else {
|
|
var c = content[content.length - 1];
|
|
if (c.hasOwnProperty('end')) {
|
|
position = [c.end.line, c.end.column];
|
|
} else {
|
|
position = getLastPosition(c.content, line, column);
|
|
}
|
|
}
|
|
|
|
if (!colOffset) return position;
|
|
|
|
if (tokens[pos - 1] && tokens[pos - 1].type !== 'Newline') {
|
|
position[1] += colOffset;
|
|
} else {
|
|
position[0]++;
|
|
position[1] = 1;
|
|
}
|
|
|
|
return position;
|
|
}
|
|
|
|
/**
|
|
* @param {string} type
|
|
* @param {string|!Array} content
|
|
* @param {number} line
|
|
* @param {number} column
|
|
* @param {!Array} end
|
|
*/
|
|
function newNode(type, content, line, column, end) {
|
|
if (!end) end = getLastPosition(content, line, column);
|
|
return new Node({
|
|
type: type,
|
|
content: content,
|
|
start: {
|
|
line: line,
|
|
column: column
|
|
},
|
|
end: {
|
|
line: end[0],
|
|
column: end[1]
|
|
},
|
|
syntax: 'sass'
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkAny(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkBrackets(i)) tokens[i].any_child = 1;else if (l = checkParentheses(i)) tokens[i].any_child = 2;else if (l = checkString(i)) tokens[i].any_child = 3;else if (l = checkVariablesList(i)) tokens[i].any_child = 4;else if (l = checkVariable(i)) tokens[i].any_child = 5;else if (l = checkPlaceholder(i)) tokens[i].any_child = 6;else if (l = checkPercentage(i)) tokens[i].any_child = 7;else if (l = checkDimension(i)) tokens[i].any_child = 8;else if (l = checkUnicodeRange(i)) tokens[i].any_child = 9;else if (l = checkNumber(i)) tokens[i].any_child = 10;else if (l = checkUri(i)) tokens[i].any_child = 11;else if (l = checkExpression(i)) tokens[i].any_child = 12;else if (l = checkFunctionsList(i)) tokens[i].any_child = 13;else if (l = checkFunction(i)) tokens[i].any_child = 14;else if (l = checkInterpolation(i)) tokens[i].any_child = 15;else if (l = checkIdent(i)) tokens[i].any_child = 16;else if (l = checkClass(i)) tokens[i].any_child = 17;else if (l = checkUnary(i)) tokens[i].any_child = 18;else if (l = checkParentSelector(i)) tokens[i].any_child = 19;else if (l = checkImportant(i)) tokens[i].any_child = 20;else if (l = checkGlobal(i)) tokens[i].any_child = 21;else if (l = checkDefault(i)) tokens[i].any_child = 22;else if (l = checkOptional(i)) tokens[i].any_child = 23;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {!Node}
|
|
*/
|
|
function getAny() {
|
|
var childType = tokens[pos].any_child;
|
|
|
|
if (childType === 1) return getBrackets();
|
|
if (childType === 2) return getParentheses();
|
|
if (childType === 3) return getString();
|
|
if (childType === 4) return getVariablesList();
|
|
if (childType === 5) return getVariable();
|
|
if (childType === 6) return getPlaceholder();
|
|
if (childType === 7) return getPercentage();
|
|
if (childType === 8) return getDimension();
|
|
if (childType === 9) return getUnicodeRange();
|
|
if (childType === 10) return getNumber();
|
|
if (childType === 11) return getUri();
|
|
if (childType === 12) return getExpression();
|
|
if (childType === 13) return getFunctionsList();
|
|
if (childType === 14) return getFunction();
|
|
if (childType === 15) return getInterpolation();
|
|
if (childType === 16) return getIdent();
|
|
if (childType === 17) return getClass();
|
|
if (childType === 18) return getUnary();
|
|
if (childType === 19) return getParentSelector();
|
|
if (childType === 20) return getImportant();
|
|
if (childType === 21) return getGlobal();
|
|
if (childType === 22) return getDefault();
|
|
if (childType === 23) return getOptional();
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of mixin's arguments.
|
|
*
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of arguments
|
|
*/
|
|
function checkArguments(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
while (i < tokens[start].right) {
|
|
if (l = checkArgument(i)) i += l;else return 0;
|
|
}
|
|
|
|
return tokens[start].right - start + 1;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getArguments() {
|
|
var type = NodeType.ArgumentsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var body = void 0;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
while (pos < tokensLength && tokens[pos].type !== TokenType.RightParenthesis) {
|
|
if (checkSingleValueDeclaration(pos)) {
|
|
content.push(getSingleValueDeclaration());
|
|
} else if (checkArgument(pos)) {
|
|
body = getArgument();
|
|
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
|
|
} else if (checkClass(pos)) content.push(getClass());else throwError(pos);
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Checks if token is valid to be part of arguments list
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of argument
|
|
*/
|
|
function checkArgument(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkBrackets(i)) tokens[i].argument_child = 1;else if (l = checkParentheses(i)) tokens[i].argument_child = 2;else if (l = checkSingleValueDeclaration(i)) tokens[i].argument_child = 3;else if (l = checkFunctionsList(i)) tokens[i].argument_child = 4;else if (l = checkFunction(i)) tokens[i].argument_child = 5;else if (l = checkVariablesList(i)) tokens[i].argument_child = 6;else if (l = checkVariable(i)) tokens[i].argument_child = 7;else if (l = checkSC(i)) tokens[i].argument_child = 8;else if (l = checkDelim(i)) tokens[i].argument_child = 9;else if (l = checkDeclDelim(i)) tokens[i].argument_child = 10;else if (l = checkString(i)) tokens[i].argument_child = 11;else if (l = checkPercentage(i)) tokens[i].argument_child = 12;else if (l = checkDimension(i)) tokens[i].argument_child = 13;else if (l = checkNumber(i)) tokens[i].argument_child = 14;else if (l = checkUri(i)) tokens[i].argument_child = 15;else if (l = checkInterpolation(i)) tokens[i].argument_child = 16;else if (l = checkIdent(i)) tokens[i].argument_child = 17;else if (l = checkVhash(i)) tokens[i].argument_child = 18;else if (l = checkCustomProperty(i)) tokens[i].argument_child = 19;else if (l = checkOperator(i)) tokens[i].argument_child = 20;else if (l = checkUnary(i)) tokens[i].argument_child = 21;else if (l = checkParentSelector(i)) tokens[i].argument_child = 22;else if (l = checkImportant(i)) tokens[i].argument_child = 23;else if (l = checkGlobal(i)) tokens[i].argument_child = 24;else if (l = checkDefault(i)) tokens[i].argument_child = 25;else if (l = checkOptional(i)) tokens[i].argument_child = 26;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {!Node}
|
|
*/
|
|
function getArgument() {
|
|
var childType = tokens[pos].argument_child;
|
|
|
|
if (childType === 1) return getBrackets();
|
|
if (childType === 2) return getParentheses();
|
|
if (childType === 3) return getSingleValueDeclaration();
|
|
if (childType === 4) return getFunctionsList();
|
|
if (childType === 5) return getFunction();
|
|
if (childType === 6) return getVariablesList();
|
|
if (childType === 7) return getVariable();
|
|
if (childType === 8) return getSC();
|
|
if (childType === 9) return getDelim();
|
|
if (childType === 10) return getDeclDelim();
|
|
if (childType === 11) return getString();
|
|
if (childType === 12) return getPercentage();
|
|
if (childType === 13) return getDimension();
|
|
if (childType === 14) return getNumber();
|
|
if (childType === 15) return getUri();
|
|
if (childType === 16) return getInterpolation();
|
|
if (childType === 17) return getIdent();
|
|
if (childType === 18) return getVhash();
|
|
if (childType === 19) return getCustomProperty();
|
|
if (childType === 20) return getOperator();
|
|
if (childType === 21) return getUnary();
|
|
if (childType === 22) return getParentSelector();
|
|
if (childType === 23) return getImportant();
|
|
if (childType === 24) return getGlobal();
|
|
if (childType === 25) return getDefault();
|
|
if (childType === 26) return getOptional();
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of an @-word (e.g. `@import`, `@include`).
|
|
*
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkAtkeyword(i) {
|
|
var l = void 0;
|
|
|
|
// Check that token is `@`:
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.CommercialAt) return 0;
|
|
|
|
return (l = checkIdentOrInterpolation(i)) ? l + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Gets node with @-word.
|
|
*
|
|
* @return {!Node}
|
|
*/
|
|
function getAtkeyword() {
|
|
var type = NodeType.AtkeywordType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `@`.
|
|
pos++;
|
|
|
|
var content = getIdentOrInterpolation();
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Checks if token is a part of an @-rule.
|
|
*
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of @-rule
|
|
*/
|
|
function checkAtrule(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// If token already has a record of being part of an @-rule,
|
|
// return the @-rule's length:
|
|
if (tokens[i].atrule_l !== undefined) return tokens[i].atrule_l;
|
|
|
|
// If token is part of an @-rule, save the rule's type to token.
|
|
// @keyframes:
|
|
if (l = checkKeyframesRule(i)) tokens[i].atrule_type = 4;
|
|
// @-rule with ruleset:
|
|
else if (l = checkAtruler(i)) tokens[i].atrule_type = 1;
|
|
// Block @-rule:
|
|
else if (l = checkAtruleb(i)) tokens[i].atrule_type = 2;
|
|
// Single-line @-rule:
|
|
else if (l = checkAtrules(i)) tokens[i].atrule_type = 3;else return 0;
|
|
|
|
// If token is part of an @-rule, save the rule's length to token:
|
|
tokens[i].atrule_l = l;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* Gets node with @-rule.
|
|
*
|
|
* @return {!Node}
|
|
*/
|
|
function getAtrule() {
|
|
var childType = tokens[pos].atrule_type;
|
|
|
|
if (childType === 1) return getAtruler(); // @-rule with ruleset
|
|
if (childType === 2) return getAtruleb(); // Block @-rule
|
|
if (childType === 3) return getAtrules(); // Single-line @-rule
|
|
if (childType === 4) return getKeyframesRule();
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of a block @-rule.
|
|
*
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the @-rule
|
|
*/
|
|
function checkAtruleb(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Gets node with a block @-rule.
|
|
*
|
|
* @return {!Node}
|
|
*/
|
|
function getAtruleb() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of an @-rule with ruleset.
|
|
*
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the @-rule
|
|
*/
|
|
function checkAtruler(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
if (l = checkAtrulers(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Gets node with an @-rule with ruleset.
|
|
*
|
|
* @return {!Node}
|
|
*/
|
|
function getAtruler() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets(), getAtrulers());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkAtrulers(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var blockEnd = tokens[i].block_end;
|
|
if (!blockEnd) return 0;
|
|
|
|
while (i < blockEnd) {
|
|
if (l = checkSC(i)) tokens[i].atrulers_child = 1;else if (l = checkAtrule(i)) tokens[i].atrulers_child = 2;else if (l = checkRuleset(i)) tokens[i].atrulers_child = 3;else return 0;
|
|
i += l;
|
|
}
|
|
|
|
if (i < tokensLength) tokens[i].atrulers_end = 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {!Node}
|
|
*/
|
|
function getAtrulers() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = getSC();
|
|
|
|
while (pos < tokensLength && !tokens[pos].atrulers_end) {
|
|
var childType = tokens[pos].atrulers_child;
|
|
if (childType === 1) content = content.concat(getSC());else if (childType === 2) content.push(getAtrule());else if (childType === 3) content.push(getRuleset());else break;
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column);
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkAtrules(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {!Node}
|
|
*/
|
|
function getAtrules() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of a block (e.g. `{...}`).
|
|
*
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the block
|
|
*/
|
|
function checkBlock(i) {
|
|
return i < tokensLength && tokens[i].block_end ? tokens[i].block_end - i + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Gets node with a block.
|
|
*
|
|
* @return {!Node}
|
|
*/
|
|
function getBlock() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = tokens[pos].block_end;
|
|
var content = [];
|
|
|
|
while (pos < end) {
|
|
if (checkBlockdecl(pos)) content = content.concat(getBlockdecl());else throwError(pos);
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of a declaration (property-value pair).
|
|
*
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the declaration
|
|
*/
|
|
function checkBlockdecl(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkBlockdecl7(i)) tokens[i].bd_type = 7;else if (l = checkBlockdecl5(i)) tokens[i].bd_type = 5;else if (l = checkBlockdecl6(i)) tokens[i].bd_type = 6;else if (l = checkBlockdecl1(i)) tokens[i].bd_type = 1;else if (l = checkBlockdecl2(i)) tokens[i].bd_type = 2;else if (l = checkBlockdecl3(i)) tokens[i].bd_type = 3;else if (l = checkBlockdecl4(i)) tokens[i].bd_type = 4;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {!Array}
|
|
*/
|
|
function getBlockdecl() {
|
|
var childType = tokens[pos].bd_type;
|
|
|
|
if (childType === 1) return getBlockdecl1();
|
|
if (childType === 2) return getBlockdecl2();
|
|
if (childType === 3) return getBlockdecl3();
|
|
if (childType === 4) return getBlockdecl4();
|
|
if (childType === 5) return getBlockdecl5();
|
|
if (childType === 6) return getBlockdecl6();
|
|
if (childType === 7) return getBlockdecl7();
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkBlockdecl1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkInclude(i)) tokens[i].bd_kind = 2;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 5;else if (l = checkAtrule(i)) tokens[i].bd_kind = 6;else return 0;
|
|
|
|
i += l;
|
|
|
|
if (tokens[start].bd_kind === 2 && [2, 4, 6, 8].indexOf(tokens[start].include_type) === -1) return 0;
|
|
|
|
if (tokens[start].bd_kind === 6 && tokens[start].atrule_type === 3) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkDeclDelim(i)) return i + l - start;
|
|
|
|
if (l = checkS(i)) i += l;else if (l = checkCommentSL(i)) i += l;else break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {!Array}
|
|
*/
|
|
function getBlockdecl1() {
|
|
var content = [];
|
|
var _content = [];
|
|
|
|
switch (tokens[pos].bd_kind) {
|
|
case 2:
|
|
content.push(getInclude());
|
|
break;
|
|
case 5:
|
|
content.push(getDeclaration());
|
|
break;
|
|
case 6:
|
|
content.push(getAtrule());
|
|
break;
|
|
}
|
|
|
|
while (pos < tokensLength) {
|
|
var _pos = pos;
|
|
if (checkDeclDelim(pos)) {
|
|
_content.push(getDeclDelim());
|
|
content = content.concat(_content);
|
|
break;
|
|
}
|
|
|
|
if (checkS(pos)) _content.push(getS());else if (checkCommentSL(pos)) _content.push(getCommentSL());else {
|
|
pos = _pos;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkBlockdecl2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkConditionalStatement(i)) tokens[i].bd_kind = 1;else if (l = checkInclude(i)) tokens[i].bd_kind = 2;else if (l = checkExtend(i)) tokens[i].bd_kind = 4;else if (l = checkMixin(i)) tokens[i].bd_kind = 8;else if (l = checkLoop(i)) tokens[i].bd_kind = 3;else if (l = checkRuleset(i)) tokens[i].bd_kind = 7;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 5;else if (l = checkAtrule(i)) tokens[i].bd_kind = 6;else return 0;
|
|
|
|
i += l;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkS(i)) i += l;else if (l = checkCommentSL(i)) i += l;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {!Array}
|
|
*/
|
|
function getBlockdecl2() {
|
|
var content = [];
|
|
|
|
switch (tokens[pos].bd_kind) {
|
|
case 1:
|
|
content.push(getConditionalStatement());
|
|
break;
|
|
case 2:
|
|
content.push(getInclude());
|
|
break;
|
|
case 3:
|
|
content.push(getLoop());
|
|
break;
|
|
case 4:
|
|
content.push(getExtend());
|
|
break;
|
|
case 5:
|
|
content.push(getDeclaration());
|
|
break;
|
|
case 6:
|
|
content.push(getAtrule());
|
|
break;
|
|
case 7:
|
|
content.push(getRuleset());
|
|
break;
|
|
case 8:
|
|
content.push(getMixin());
|
|
break;
|
|
}
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkS(pos)) content.push(getS());else if (checkCommentSL(pos)) content.push(getCommentSL());else break;
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkBlockdecl3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkConditionalStatement(i)) tokens[i].bd_kind = 1;else if (l = checkInclude(i)) tokens[i].bd_kind = 2;else if (l = checkExtend(i)) tokens[i].bd_kind = 4;else if (l = checkLoop(i)) tokens[i].bd_kind = 3;else if (l = checkRuleset(i)) tokens[i].bd_kind = 7;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 5;else if (l = checkAtrule(i)) tokens[i].bd_kind = 6;else return 0;
|
|
|
|
i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {!Array}
|
|
*/
|
|
function getBlockdecl3() {
|
|
var content = void 0;
|
|
|
|
switch (tokens[pos].bd_kind) {
|
|
case 1:
|
|
content = getConditionalStatement();
|
|
break;
|
|
case 2:
|
|
content = getInclude();
|
|
break;
|
|
case 3:
|
|
content = getLoop();
|
|
break;
|
|
case 4:
|
|
content = getExtend();
|
|
break;
|
|
case 5:
|
|
content = getDeclaration();
|
|
break;
|
|
case 6:
|
|
content = getAtrule();
|
|
break;
|
|
case 7:
|
|
content = getRuleset();
|
|
break;
|
|
}
|
|
|
|
return [content];
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkBlockdecl4(i) {
|
|
return checkSC(i);
|
|
}
|
|
|
|
/**
|
|
* @return {!Array}
|
|
*/
|
|
function getBlockdecl4() {
|
|
return getSC();
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkBlockdecl5(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkInclude(i)) i += l;else if (l = checkRuleset(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkS(i)) i += l;else if (l = checkCommentSL(i)) i += l;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {!Array}
|
|
*/
|
|
function getBlockdecl5() {
|
|
var content = [];
|
|
|
|
if (checkInclude(pos)) content.push(getInclude());else content.push(getRuleset());
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkS(pos)) content.push(getS());else if (checkCommentSL(pos)) content.push(getCommentSL());else break;
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkBlockdecl6(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkInclude(i)) i += l;else if (l = checkRuleset(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {!Array}
|
|
*/
|
|
function getBlockdecl6() {
|
|
var content = void 0;
|
|
|
|
if (checkInclude(pos)) content = getInclude();else content = getRuleset();
|
|
|
|
return [content];
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkBlockdecl7(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkInclude(i)) i += l;else return 0;
|
|
|
|
if ([2, 4, 6, 8].indexOf(tokens[start].include_type) === -1) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkDeclDelim(i)) return i + l - start;
|
|
|
|
if (l = checkS(i)) i += l;else if (l = checkCommentSL(i)) i += l;else break;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {!Array}
|
|
*/
|
|
function getBlockdecl7() {
|
|
var content = [];
|
|
var _content = [];
|
|
|
|
content.push(getInclude());
|
|
|
|
while (pos < tokensLength) {
|
|
var _pos = pos;
|
|
if (checkDeclDelim(pos)) {
|
|
_content.push(getDeclDelim());
|
|
content = content.concat(_content);
|
|
break;
|
|
}
|
|
|
|
if (checkS(pos)) _content.push(getS());else if (checkCommentSL(pos)) _content.push(getCommentSL());else {
|
|
pos = _pos;
|
|
break;
|
|
}
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of text inside square brackets, e.g. `[1]`.
|
|
*
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkBrackets(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
|
|
// Skip `[`.
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
if (i < tokens[start].right) {
|
|
var l = checkTsets(i);
|
|
if (l) i += l;else return 0;
|
|
}
|
|
|
|
// Skip `]`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Gets node with text inside square brackets, e.g. `[1]`.
|
|
*
|
|
* @return {!Node}
|
|
*/
|
|
function getBrackets() {
|
|
var type = NodeType.BracketsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var right = token.right;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
if (pos < right) {
|
|
content = getTsets();
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of a class selector (e.g. `.abc`).
|
|
*
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the class selector
|
|
*/
|
|
function checkClass(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].class_l) return tokens[i].class_l;
|
|
|
|
// Skip `.`.
|
|
if (tokens[i].type === TokenType.FullStop) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkIdentOrInterpolation(i)) {
|
|
tokens[start].class_l = l + 1;
|
|
i += l;
|
|
} else break;
|
|
}
|
|
|
|
tokens[start].classEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Gets node with a class selector.
|
|
*
|
|
* @return {!Node}
|
|
*/
|
|
function getClass() {
|
|
var type = NodeType.ClassType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = token.classEnd;
|
|
var content = [];
|
|
|
|
// Skip `.`
|
|
pos++;
|
|
|
|
while (pos < end) {
|
|
if (checkIdentOrInterpolation(pos)) {
|
|
content = content.concat(getIdentOrInterpolation());
|
|
} else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i
|
|
* @return {number}
|
|
*/
|
|
function checkCombinator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkCombinator1(i)) tokens[i].combinatorType = 1;else if (l = checkCombinator2(i)) tokens[i].combinatorType = 2;else if (l = checkCombinator3(i)) tokens[i].combinatorType = 3;else if (l = checkCombinator4(i)) tokens[i].combinatorType = 4;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {!Node}
|
|
*/
|
|
function getCombinator() {
|
|
var type = tokens[pos].combinatorType;
|
|
if (type === 1) return getCombinator1();
|
|
if (type === 2) return getCombinator2();
|
|
if (type === 3) return getCombinator3();
|
|
if (type === 4) return getCombinator4();
|
|
}
|
|
|
|
/**
|
|
* (1) `>>>`
|
|
*
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkCombinator1(i) {
|
|
if (i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign) return 3;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator1() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '>>>';
|
|
|
|
// Skip combinator
|
|
pos += 3;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `||`
|
|
* (2) `>>`
|
|
*
|
|
* @param {number} i
|
|
* @return {number}
|
|
*/
|
|
function checkCombinator2(i) {
|
|
if (i + 1 >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.VerticalLine && tokens[i + 1].type === TokenType.VerticalLine) return 2;
|
|
|
|
if (tokens[i].type === TokenType.GreaterThanSign && tokens[i + 1].type === TokenType.GreaterThanSign) return 2;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {!Node}
|
|
*/
|
|
function getCombinator2() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '' + token.value + tokens[pos + 1].value;
|
|
|
|
// Skip combinator
|
|
pos += 2;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `>`
|
|
* (2) `+`
|
|
* (3) `~`
|
|
*
|
|
* @param {number} i
|
|
* @return {number}
|
|
*/
|
|
function checkCombinator3(i) {
|
|
var type = tokens[i].type;
|
|
if (type === TokenType.PlusSign || type === TokenType.GreaterThanSign || type === TokenType.Tilde) return 1;else return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator3() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
// Skip combinator
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `/panda/`
|
|
*/
|
|
function checkCombinator4(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator4() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `/`.
|
|
pos++;
|
|
|
|
var ident = getIdent();
|
|
|
|
// Skip `/`.
|
|
pos++;
|
|
|
|
var content = '/' + ident.content + '/';
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a multiline comment.
|
|
* @param {number} i Token's index number
|
|
* @return {number} `1` if token is a multiline comment, otherwise `0`
|
|
*/
|
|
function checkCommentML(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.CommentML ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a multiline comment
|
|
* @return {Array} `['commentML', x]` where `x`
|
|
* is the comment's text (without `/*` and `* /`).
|
|
*/
|
|
function getCommentML() {
|
|
var type = NodeType.CommentMLType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos].value.substring(2);
|
|
|
|
var end = getLastPosition(content, line, column + 2);
|
|
|
|
if (content.endsWith('*/')) {
|
|
content = content.substring(0, content.length - 2);
|
|
}
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a single-line comment.
|
|
* @param {number} i Token's index number
|
|
* @return {number} `1` if token is a single-line comment, otherwise `0`
|
|
*/
|
|
function checkCommentSL(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.CommentSL ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a single-line comment.
|
|
* @return {Array} `['commentSL', x]` where `x` is comment's message
|
|
* (without `//`)
|
|
*/
|
|
function getCommentSL() {
|
|
var type = NodeType.CommentSLType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos++].value.substring(2);
|
|
var end = !content ? [line, column + 1] : getLastPosition(content, line, column + 2);
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a condition
|
|
* (e.g. `@if ...`, `@else if ...` or `@else ...`).
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the condition
|
|
*/
|
|
function checkCondition(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
var _i = void 0;
|
|
var s = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (['if', 'else'].indexOf(tokens[start + 1].value) < 0) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkBlock(i)) break;
|
|
|
|
s = checkSC(i);
|
|
_i = i + s;
|
|
|
|
if (l = _checkCondition(_i)) i += l + s;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function _checkCondition(i) {
|
|
return checkVariable(i) || checkNumber(i) || checkInterpolation(i) || checkIdent(i) || checkOperator(i) || checkCombinator(i) || checkString(i);
|
|
}
|
|
|
|
/**
|
|
* Get node with a condition.
|
|
* @return {Array} `['condition', x]`
|
|
*/
|
|
function getCondition() {
|
|
var type = NodeType.ConditionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var s = void 0;
|
|
var _pos = void 0;
|
|
|
|
content.push(getAtkeyword());
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkBlock(pos)) break;
|
|
|
|
s = checkSC(pos);
|
|
_pos = pos + s;
|
|
|
|
if (!_checkCondition(_pos)) break;
|
|
|
|
if (s) content = content.concat(getSC());
|
|
content.push(_getCondition());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function _getCondition() {
|
|
if (checkVariable(pos)) return getVariable();
|
|
if (checkNumber(pos)) return getNumber();
|
|
if (checkInterpolation(pos)) return getInterpolation();
|
|
if (checkIdent(pos)) return getIdent();
|
|
if (checkOperator(pos)) return getOperator();
|
|
if (checkCombinator(pos)) return getCombinator();
|
|
if (checkString(pos)) return getString();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a conditional statement
|
|
* (e.g. `@if ... {} @else if ... {} @else ... {}`).
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the condition
|
|
*/
|
|
function checkConditionalStatement(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkCondition(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a condition.
|
|
* @return {Array} `['condition', x]`
|
|
*/
|
|
function getConditionalStatement() {
|
|
var type = NodeType.ConditionalStatementType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getCondition(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a declaration (property-value pair)
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the declaration
|
|
*/
|
|
function checkDeclaration(i) {
|
|
return checkDeclaration1(i) || checkDeclaration2(i);
|
|
}
|
|
|
|
/**
|
|
* Get node with a declaration
|
|
* @return {Array} `['declaration', ['property', x], ['propertyDelim'],
|
|
* ['value', y]]`
|
|
*/
|
|
function getDeclaration() {
|
|
return checkDeclaration1(pos) ? getDeclaration1() : getDeclaration2();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a declaration (property-value pair)
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the declaration
|
|
*/
|
|
function checkDeclaration1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkProperty(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkPropertyDelim(i)) i++;else return 0;
|
|
|
|
if (l = checkValue(i)) return i + l - start;
|
|
|
|
if (l = checkS(i)) i += l;
|
|
|
|
if (l = checkValue(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a declaration
|
|
* @return {Array} `['declaration', ['property', x], ['propertyDelim'],
|
|
* ['value', y]]`
|
|
*/
|
|
function getDeclaration1() {
|
|
var type = NodeType.DeclarationType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(getProperty());
|
|
if (checkS(pos)) content.push(getS());
|
|
content.push(getPropertyDelim());
|
|
if (checkS(pos)) content.push(getS());
|
|
content.push(getValue());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a declaration (property-value pair)
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the declaration
|
|
*/
|
|
function checkDeclaration2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkPropertyDelim(i)) i++;else return 0;
|
|
|
|
if (l = checkProperty(i)) i += l;else return 0;
|
|
|
|
if (l = checkValue(i)) return i + l - start;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkValue(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a declaration
|
|
* @return {Array} `['declaration', ['propertyDelim'], ['property', x],
|
|
* ['value', y]]`
|
|
*/
|
|
function getDeclaration2() {
|
|
var type = NodeType.DeclarationType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getPropertyDelim(), getProperty(), getSC(), getValue());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the declaration
|
|
*/
|
|
function checkSingleValueDeclaration(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkProperty(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkPropertyDelim(i)) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkSingleValue(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a declaration
|
|
* @returns {Array} `['declaration', ['property', x], ['propertyDelim'],
|
|
* ['value', y]]`
|
|
*/
|
|
function getSingleValueDeclaration() {
|
|
var type = NodeType.DeclarationType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getProperty(), getSC(), getPropertyDelim(), getSC(), getSingleValue());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a semicolon
|
|
* @param {number} i Token's index number
|
|
* @return {number} `1` if token is a semicolon, otherwise `0`
|
|
*/
|
|
function checkDeclDelim(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
return tokens[i].type === TokenType.Newline || tokens[i].type === TokenType.Semicolon ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a semicolon
|
|
* @return {Array} `['declDelim']`
|
|
*/
|
|
function getDeclDelim() {
|
|
var type = NodeType.DeclDelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '\n';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token if part of `!default` word.
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the `!default` word
|
|
*/
|
|
function checkDefault(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].value === 'default') {
|
|
tokens[start].defaultEnd = i;
|
|
return i - start + 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get node with a `!default` word
|
|
* @return {Array} `['default', sc]` where `sc` is optional whitespace
|
|
*/
|
|
function getDefault() {
|
|
var type = NodeType.DefaultType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, token.defaultEnd);
|
|
|
|
pos = token.defaultEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a comma
|
|
* @param {number} i Token's index number
|
|
* @return {number} `1` if token is a comma, otherwise `0`
|
|
*/
|
|
function checkDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Comma ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a comma
|
|
* @return {Array} `['delim']`
|
|
*/
|
|
function getDelim() {
|
|
var type = NodeType.DelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ',';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number with dimension unit (e.g. `10px`)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkDimension(i) {
|
|
var ln = checkNumber(i);
|
|
var li = void 0;
|
|
|
|
if (i >= tokensLength || !ln || i + ln >= tokensLength) return 0;
|
|
|
|
return (li = checkUnit(i + ln)) ? ln + li : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node of a number with dimension unit
|
|
* @return {Node}
|
|
*/
|
|
function getDimension() {
|
|
var type = NodeType.DimensionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getNumber(), getUnit()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkExpression(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength || tokens[i++].value !== 'expression' || i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) {
|
|
return 0;
|
|
}
|
|
|
|
return tokens[i].right - start + 1;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getExpression() {
|
|
var type = NodeType.ExpressionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
pos++;
|
|
|
|
var content = joinValues(pos + 1, tokens[pos].right - 1);
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
if (end[0] === line) end[1] += 11;
|
|
pos = tokens[pos].right + 1;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkExtend(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
|
|
if (l = checkExtend1(i)) tokens[i].extend_child = 1;else if (l = checkExtend2(i)) tokens[i].extend_child = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getExtend() {
|
|
var childType = tokens[pos].extend_child;
|
|
|
|
if (childType === 1) return getExtend1();
|
|
if (childType === 2) return getExtend2();
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of an extend with `!optional` flag.
|
|
* @param {number} i
|
|
*/
|
|
function checkExtend1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'extend') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkOptional(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getExtend1() {
|
|
var type = NodeType.ExtendType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getSelectorsGroup(), getSC(), getOptional());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of an extend without `!optional` flag.
|
|
* @param {number} i
|
|
*/
|
|
function checkExtend2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'extend') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getExtend2() {
|
|
var type = NodeType.ExtendType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getSelectorsGroup());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkFunction(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i < tokensLength && tokens[i].type === TokenType.LeftParenthesis ? tokens[i].right - start + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getFunction() {
|
|
var type = NodeType.FunctionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getIdentOrInterpolation(), getArguments());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a functions list (e.g. `function(value)...`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkFunctionsList(i) {
|
|
var d = 0; // Number of dots
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkFunction(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength && tokens[i].type === TokenType.FullStop) {
|
|
d++;
|
|
i++;
|
|
}
|
|
|
|
return d === 3 ? l + d : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a functions list
|
|
* @returns {Array}
|
|
*/
|
|
function getFunctionsList() {
|
|
var type = NodeType.FunctionsListType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getFunction()];
|
|
var end = getLastPosition(content, line, column, 3);
|
|
|
|
// Skip `...`.
|
|
pos += 3;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of `!global` word
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkGlobal(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].value === 'global') {
|
|
tokens[start].globalEnd = i;
|
|
return i - start + 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get node with `!global` word
|
|
*/
|
|
function getGlobal() {
|
|
var type = NodeType.GlobalType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, token.globalEnd);
|
|
|
|
pos = token.globalEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an identifier
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the identifier
|
|
*/
|
|
function checkIdent(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Check if token is part of a negative number
|
|
if (tokens[i].type === TokenType.HyphenMinus && tokens[i + 1].type === TokenType.DecimalNumber) return 0;
|
|
|
|
if (tokens[i].type === TokenType.HyphenMinus) i++;
|
|
|
|
if (checkInterpolation(i)) {
|
|
tokens[start].ident_last = i - 1;
|
|
return i - start;
|
|
}
|
|
|
|
if (tokens[i].type === TokenType.LowLine || tokens[i].type === TokenType.Identifier) i++;else return 0;
|
|
|
|
for (; i < tokensLength; i++) {
|
|
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
|
|
}
|
|
|
|
tokens[start].ident_last = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with an identifier
|
|
* @return {Array} `['ident', x]` where `x` is identifier's name
|
|
*/
|
|
function getIdent() {
|
|
var type = NodeType.IdentType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, tokens[pos].ident_last);
|
|
|
|
pos = tokens[pos].ident_last + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the identifier
|
|
*/
|
|
function checkPartialIdent(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
for (; i < tokensLength; i++) {
|
|
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
|
|
}
|
|
|
|
tokens[start].ident_last = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the identifier
|
|
*/
|
|
function checkIdentOrInterpolation(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
var prevIsInterpolation = false;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkInterpolation(i)) {
|
|
tokens[i].ii_type = 1;
|
|
i += l;
|
|
prevIsInterpolation = true;
|
|
} else if (l = checkIdent(i)) {
|
|
tokens[i].ii_type = 2;
|
|
i += l;
|
|
prevIsInterpolation = false;
|
|
} else if (prevIsInterpolation && (l = checkPartialIdent(i))) {
|
|
tokens[i].ii_type = 3;
|
|
i += l;
|
|
prevIsInterpolation = false;
|
|
} else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getIdentOrInterpolation() {
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
var tokenType = tokens[pos].ii_type;
|
|
|
|
if (tokenType === 1) {
|
|
content.push(getInterpolation());
|
|
} else if (tokenType === 2 || tokenType === 3) {
|
|
content.push(getIdent());
|
|
} else break;
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of `!important` word
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkImportant(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].value === 'important') {
|
|
tokens[start].importantEnd = i;
|
|
return i - start + 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get node with `!important` word
|
|
* @return {Array} `['important', sc]` where `sc` is optional whitespace
|
|
*/
|
|
function getImportant() {
|
|
var type = NodeType.ImportantType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, token.importantEnd);
|
|
|
|
pos = token.importantEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin (`@include` or `@extend`
|
|
* directive).
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the included mixin
|
|
*/
|
|
function checkInclude(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIncludeWithKeyframes1(i)) tokens[i].include_type = 9;else if (l = checkInclude1(i)) tokens[i].include_type = 1;else if (l = checkInclude2(i)) tokens[i].include_type = 2;else if (l = checkInclude3(i)) tokens[i].include_type = 3;else if (l = checkInclude4(i)) tokens[i].include_type = 4;else if (l = checkIncludeWithKeyframes2(i)) tokens[i].include_type = 10;else if (l = checkInclude5(i)) tokens[i].include_type = 5;else if (l = checkInclude6(i)) tokens[i].include_type = 6;else if (l = checkInclude7(i)) tokens[i].include_type = 7;else if (l = checkInclude8(i)) tokens[i].include_type = 8;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin
|
|
* @return {Array} `['include', x]`
|
|
*/
|
|
function getInclude() {
|
|
var type = tokens[pos].include_type;
|
|
|
|
if (type === 1) return getInclude1();
|
|
if (type === 2) return getInclude2();
|
|
if (type === 3) return getInclude3();
|
|
if (type === 4) return getInclude4();
|
|
if (type === 5) return getInclude5();
|
|
if (type === 6) return getInclude6();
|
|
if (type === 7) return getInclude7();
|
|
if (type === 8) return getInclude8();
|
|
if (type === 9) return getIncludeWithKeyframes1();
|
|
if (type === 10) return getIncludeWithKeyframes2();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin like `@include nani(foo) {...}`
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the include
|
|
*/
|
|
function checkInclude1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'include') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin like `@include nani(foo) {...}`
|
|
* @return {Array} `['include', ['atkeyword', x], sc, ['selector', y], sc,
|
|
* ['arguments', z], sc, ['block', q], sc` where `x` is `include` or
|
|
* `extend`, `y` is mixin's identifier (selector), `z` are arguments
|
|
* passed to the mixin, `q` is block passed to the mixin and `sc`
|
|
* are optional whitespaces
|
|
*/
|
|
function getInclude1() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin like `@include nani(foo)`
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the include
|
|
*/
|
|
function checkInclude2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'include') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin like `@include nani(foo)`
|
|
* @return {Array} `['include', ['atkeyword', x], sc, ['selector', y], sc,
|
|
* ['arguments', z], sc]` where `x` is `include` or `extend`, `y` is
|
|
* mixin's identifier (selector), `z` are arguments passed to the
|
|
* mixin and `sc` are optional whitespaces
|
|
*/
|
|
function getInclude2() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin with a content block passed
|
|
* as an argument (e.g. `@include nani {...}`)
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the mixin
|
|
*/
|
|
function checkInclude3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'include') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with an included mixin with a content block passed
|
|
* as an argument (e.g. `@include nani {...}`)
|
|
* @return {Array} `['include', x]`
|
|
*/
|
|
function getInclude3() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkInclude4(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'include') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array} `['include', x]`
|
|
*/
|
|
function getInclude4() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin like `+nani(foo) {...}`
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the include
|
|
*/
|
|
function checkInclude5(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (tokens[i].type === TokenType.PlusSign) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin like `+nani(foo) {...}`
|
|
* @return {Array} `['include', ['operator', '+'], ['selector', x], sc,
|
|
* ['arguments', y], sc, ['block', z], sc` where `x` is
|
|
* mixin's identifier (selector), `y` are arguments passed to the
|
|
* mixin, `z` is block passed to mixin and `sc` are optional whitespaces
|
|
*/
|
|
function getInclude5() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getOperator(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin like `+nani(foo)`
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the include
|
|
*/
|
|
function checkInclude6(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (tokens[i].type === TokenType.PlusSign) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin like `+nani(foo)`
|
|
* @return {Array} `['include', ['operator', '+'], ['selector', y], sc,
|
|
* ['arguments', z], sc]` where `y` is
|
|
* mixin's identifier (selector), `z` are arguments passed to the
|
|
* mixin and `sc` are optional whitespaces
|
|
*/
|
|
function getInclude6() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getOperator(), getIdentOrInterpolation(), getSC(), getArguments());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin with a content block passed
|
|
* as an argument (e.g. `+nani {...}`)
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the mixin
|
|
*/
|
|
function checkInclude7(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (tokens[i].type === TokenType.PlusSign) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with an included mixin with a content block passed
|
|
* as an argument (e.g. `+nani {...}`)
|
|
* @return {Array} `['include', x]`
|
|
*/
|
|
function getInclude7() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getOperator(), getIdentOrInterpolation(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkInclude8(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (tokens[i].type === TokenType.PlusSign) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array} `['include', x]`
|
|
*/
|
|
function getInclude8() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getOperator(), getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin with keyfames selector like
|
|
* `@include nani(foo) { 0% {}}`
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the include
|
|
*/
|
|
function checkIncludeWithKeyframes1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'include') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin with keyfames selector like
|
|
* `@include nani(foo) { 0% {}}`
|
|
* @return {!Node}
|
|
*/
|
|
function getIncludeWithKeyframes1() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getKeyframesBlocks());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin with keyfames selector like
|
|
* `+nani(foo) { 0% {}}`
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the include
|
|
*/
|
|
function checkIncludeWithKeyframes2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (tokens[i].type === TokenType.PlusSign) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin with keyfames selector like
|
|
* `+nani(foo) { 0% {}}`
|
|
* @return {!Node}
|
|
*/
|
|
function getIncludeWithKeyframes2() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getOperator(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getKeyframesBlocks());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an interpolated variable (e.g. `#{$nani}`).
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkInterpolation(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type !== TokenType.NumberSign || !tokens[i + 1] || tokens[i + 1].type !== TokenType.LeftCurlyBracket) return 0;
|
|
|
|
i += 2;
|
|
|
|
while (tokens[i].type !== TokenType.RightCurlyBracket) {
|
|
if (l = checkArgument(i)) i += l;else return 0;
|
|
}
|
|
|
|
return tokens[i].type === TokenType.RightCurlyBracket ? i - start + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an interpolated variable
|
|
* @return {Array} `['interpolation', x]`
|
|
*/
|
|
function getInterpolation() {
|
|
var type = NodeType.InterpolationType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `#{`:
|
|
pos += 2;
|
|
|
|
while (pos < tokensLength && tokens[pos].type !== TokenType.RightCurlyBracket) {
|
|
var body = getArgument();
|
|
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `}`:
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check a single keyframe block - `5% {}`
|
|
* @param {number} i
|
|
* @return {number}
|
|
*/
|
|
function checkKeyframesBlock(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkKeyframesSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a single keyframe block - `5% {}`
|
|
* @return {Node}
|
|
*/
|
|
function getKeyframesBlock() {
|
|
var type = NodeType.RulesetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getKeyframesSelectorsGroup(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check all keyframe blocks - `5% {} 100% {}`
|
|
* @param {number} i
|
|
* @return {number}
|
|
*/
|
|
function checkKeyframesBlocks(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var blockEnd = tokens[i].block_end;
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (!blockEnd) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlock(i)) i += l;
|
|
|
|
while (i < blockEnd) {
|
|
if (l = checkSC(i)) i += l;else if (l = checkKeyframesBlock(i)) i += l;else if (l = checkAtrule(i)) i += l;else break;
|
|
}
|
|
|
|
if (i !== blockEnd + 1) return 0;
|
|
|
|
return blockEnd + 1 - start;
|
|
}
|
|
|
|
/**
|
|
* Get all keyframe blocks - `5% {} 100% {}`
|
|
* @return {Node}
|
|
*/
|
|
function getKeyframesBlocks() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var keyframesBlocksEnd = token.block_end;
|
|
var content = [];
|
|
|
|
while (pos < keyframesBlocksEnd) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkKeyframesBlock(pos)) content.push(getKeyframesBlock());else if (checkAtrule(pos)) content.push(getAtrule()); // @content
|
|
else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a @keyframes rule.
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the @keyframes rule
|
|
*/
|
|
function checkKeyframesRule(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
var atruleName = joinValues2(i - l, l);
|
|
if (atruleName.toLowerCase().indexOf('keyframes') === -1) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i) || checkPseudoc(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getKeyframesRule() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC());
|
|
|
|
if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());else if (checkPseudoc(pos)) {
|
|
content = content.concat(getPseudoc());
|
|
}
|
|
|
|
content = content.concat(getSC(), getKeyframesBlocks());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check a single keyframe selector - `5%`, `from` etc
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkKeyframesSelector(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) {
|
|
// Valid selectors are only `from` and `to`.
|
|
var selector = joinValues2(i, l);
|
|
if (selector !== 'from' && selector !== 'to') return 0;
|
|
|
|
i += l;
|
|
tokens[start].keyframesSelectorType = 1;
|
|
} else if (l = checkPercentage(i)) {
|
|
i += l;
|
|
tokens[start].keyframesSelectorType = 2;
|
|
} else if (l = checkInterpolation(i)) {
|
|
i += l;
|
|
tokens[start].keyframesSelectorType = 3;
|
|
} else {
|
|
return 0;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a single keyframe selector
|
|
* @return {Node}
|
|
*/
|
|
function getKeyframesSelector() {
|
|
var keyframesSelectorType = NodeType.KeyframesSelectorType;
|
|
var selectorType = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (token.keyframesSelectorType === 1) {
|
|
content.push(getIdent());
|
|
} else if (token.keyframesSelectorType === 2) {
|
|
content.push(getPercentage());
|
|
} else if (token.keyframesSelectorType === 3) {
|
|
content.push(getInterpolation());
|
|
}
|
|
|
|
var keyframesSelector = newNode(keyframesSelectorType, content, line, column);
|
|
|
|
return newNode(selectorType, [keyframesSelector], line, column);
|
|
}
|
|
|
|
/**
|
|
* Check the keyframe's selector groups
|
|
* @param {number} i
|
|
* @return {number}
|
|
*/
|
|
function checkKeyframesSelectorsGroup(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkKeyframesSelector(i)) i += l;else return 0;
|
|
|
|
// Check for trailing space
|
|
if (l = checkSC(i) && tokens[i].type !== TokenType.Newline) i += l;
|
|
|
|
while (i < tokensLength) {
|
|
var tempStart = i;
|
|
var tempIndex = i;
|
|
var tempLength = void 0;
|
|
|
|
if (tempLength = checkDelim(tempIndex)) tempIndex += tempLength;else break;
|
|
|
|
// Check for maxmimum space usage - 'space', '\n', 'space'
|
|
if (tempLength = checkSC(tempIndex)) tempIndex += tempLength;
|
|
if (tempLength = checkSC(tempIndex)) tempIndex += tempLength;
|
|
if (tempLength = checkSC(tempIndex)) tempIndex += tempLength;
|
|
|
|
if (tempLength = checkKeyframesSelector(tempIndex)) tempIndex += tempLength;else break;
|
|
|
|
// Check for trailing space
|
|
if (tempLength = checkSC(tempIndex) && tokens[tempIndex].type !== TokenType.Newline) {
|
|
tempIndex += tempLength;
|
|
}
|
|
|
|
i += tempIndex - tempStart;
|
|
}
|
|
|
|
tokens[start].selectorsGroupEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get the keyframe's selector groups
|
|
* @return {Array} An array of keyframe selectors
|
|
*/
|
|
function getKeyframesSelectorsGroup() {
|
|
var selectorsGroup = [];
|
|
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
|
|
|
|
selectorsGroup.push(getKeyframesSelector());
|
|
|
|
if (checkSC(pos) && tokens[pos].type !== TokenType.Newline) {
|
|
selectorsGroup = selectorsGroup.concat(getSC());
|
|
}
|
|
|
|
while (pos < selectorsGroupEnd) {
|
|
selectorsGroup = selectorsGroup.concat(getDelim(), getSC(), getSC(), getSC(), getKeyframesSelector());
|
|
|
|
if (checkSC(pos) && tokens[pos].type !== TokenType.Newline) {
|
|
selectorsGroup = selectorsGroup.concat(getSC());
|
|
}
|
|
}
|
|
|
|
return selectorsGroup;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a loop.
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the loop
|
|
*/
|
|
function checkLoop(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (['for', 'each', 'while'].indexOf(tokens[start + 1].value) < 0) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkBlock(i)) {
|
|
i += l;
|
|
break;
|
|
} else if (l = checkVariable(i) || checkNumber(i) || checkInterpolation(i) || checkIdent(i) || checkSC(i) || checkOperator(i) || checkCombinator(i) || checkString(i)) i += l;else return 0;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a loop.
|
|
* @return {Array} `['loop', x]`
|
|
*/
|
|
function getLoop() {
|
|
var type = NodeType.LoopType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(getAtkeyword());
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkBlock(pos)) {
|
|
content.push(getBlock());
|
|
break;
|
|
} else if (checkVariable(pos)) content.push(getVariable());else if (checkNumber(pos)) content.push(getNumber());else if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkIdent(pos)) content.push(getIdent());else if (checkOperator(pos)) content.push(getOperator());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkSC(pos)) content = content.concat(getSC());else if (checkString(pos)) content.push(getString());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a mixin
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the mixin
|
|
*/
|
|
function checkMixin(i) {
|
|
return checkMixin1(i) || checkMixin2(i);
|
|
}
|
|
|
|
/**
|
|
* Get node with a mixin
|
|
* @return {Array} `['mixin', x]`
|
|
*/
|
|
function getMixin() {
|
|
return checkMixin1(pos) ? getMixin1() : getMixin2();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a mixin
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the mixin
|
|
*/
|
|
function checkMixin1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if ((l = checkAtkeyword(i)) && tokens[i + 1].value === 'mixin') i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else {
|
|
if (l = checkArguments(i)) i += l;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a mixin
|
|
* @return {Array} `['mixin', x]`
|
|
*/
|
|
function getMixin1() {
|
|
var type = NodeType.MixinType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC());
|
|
|
|
if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());
|
|
|
|
content = content.concat(getSC());
|
|
|
|
if (checkBlock(pos)) content.push(getBlock());else {
|
|
if (checkArguments(pos)) content.push(getArguments());
|
|
|
|
content = content.concat(getSC());
|
|
|
|
content.push(getBlock());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a mixin
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the mixin
|
|
*/
|
|
function checkMixin2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.EqualsSign) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else {
|
|
if (l = checkArguments(i)) i += l;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a mixin
|
|
* @return {Array} `['mixin', x]`
|
|
*/
|
|
function getMixin2() {
|
|
var type = NodeType.MixinType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getOperator(), getSC());
|
|
|
|
if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());
|
|
|
|
content = content.concat(getSC());
|
|
|
|
if (checkBlock(pos)) content.push(getBlock());else {
|
|
if (checkArguments(pos)) content.push(getArguments());
|
|
|
|
content = content.concat(getSC());
|
|
|
|
content.push(getBlock());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a namespace sign (`|`)
|
|
* @param {number} i Token's index number
|
|
* @return {number} `1` if token is `|`, `0` if not
|
|
*/
|
|
function checkNamespace(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.VerticalLine ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a namespace sign
|
|
* @return {Array} `['namespace']`
|
|
*/
|
|
function getNamespace() {
|
|
var type = NodeType.NamespaceType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '|';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkNmName2(i) {
|
|
if (tokens[i].type === TokenType.Identifier) return 1;else if (tokens[i].type !== TokenType.DecimalNumber) return 0;
|
|
|
|
i++;
|
|
|
|
return i < tokensLength && tokens[i].type === TokenType.Identifier ? 2 : 1;
|
|
}
|
|
|
|
/**
|
|
* @return {string}
|
|
*/
|
|
function getNmName2() {
|
|
var s = tokens[pos].value;
|
|
|
|
if (tokens[pos++].type === TokenType.DecimalNumber && pos < tokensLength && tokens[pos].type === TokenType.Identifier) s += tokens[pos++].value;
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of number
|
|
*/
|
|
function checkNumber(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].number_l) return tokens[i].number_l;
|
|
|
|
// `10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && (!tokens[i + 1] || tokens[i + 1] && tokens[i + 1].type !== TokenType.FullStop)) {
|
|
tokens[i].number_l = 1;
|
|
return 1;
|
|
}
|
|
|
|
// `10.`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && (!tokens[i + 2] || tokens[i + 2].type !== TokenType.DecimalNumber)) {
|
|
tokens[i].number_l = 2;
|
|
return 2;
|
|
}
|
|
|
|
// `.10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.FullStop && tokens[i + 1].type === TokenType.DecimalNumber) {
|
|
tokens[i].number_l = 2;
|
|
return 2;
|
|
}
|
|
|
|
// `10.10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && tokens[i + 2] && tokens[i + 2].type === TokenType.DecimalNumber) {
|
|
tokens[i].number_l = 3;
|
|
return 3;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with number
|
|
* @return {Array} `['number', x]` where `x` is a number converted
|
|
* to string.
|
|
*/
|
|
function getNumber() {
|
|
var type = NodeType.NumberType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var l = tokens[pos].number_l;
|
|
var content = '';
|
|
|
|
for (var j = 0; j < l; j++) {
|
|
content += tokens[pos + j].value;
|
|
}
|
|
|
|
pos += l;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is an operator (`/`, `%`, `,`, `:` or `=`).
|
|
* @param {number} i Token's index number
|
|
* @return {number} `1` if token is an operator, otherwise `0`
|
|
*/
|
|
function checkOperator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
switch (tokens[i].type) {
|
|
case TokenType.Solidus:
|
|
case TokenType.PercentSign:
|
|
case TokenType.Comma:
|
|
case TokenType.Colon:
|
|
case TokenType.EqualsSign:
|
|
case TokenType.EqualitySign:
|
|
case TokenType.InequalitySign:
|
|
case TokenType.LessThanSign:
|
|
case TokenType.GreaterThanSign:
|
|
case TokenType.Asterisk:
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an operator
|
|
* @return {Array} `['operator', x]` where `x` is an operator converted
|
|
* to string.
|
|
*/
|
|
function getOperator() {
|
|
var type = NodeType.OperatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of `!optional` word
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkOptional(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].value === 'optional') {
|
|
tokens[start].optionalEnd = i;
|
|
return i - start + 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get node with `!optional` word
|
|
*/
|
|
function getOptional() {
|
|
var type = NodeType.OptionalType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, token.optionalEnd);
|
|
|
|
pos = token.optionalEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of text inside parentheses, e.g. `(1)`
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkParentheses(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var right = tokens[i].right;
|
|
var l = void 0;
|
|
|
|
// Skip `(`.
|
|
if (tokens[i].type === TokenType.LeftParenthesis) i++;else return 0;
|
|
|
|
if (i < right) {
|
|
if (l = checkTsets(i)) i += l;else return 0;
|
|
}
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with text inside parentheses, e.g. `(1)`
|
|
* @return {Node}
|
|
*/
|
|
function getParentheses() {
|
|
var type = NodeType.ParenthesesType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var right = token.right;
|
|
var content = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
if (pos < right) {
|
|
content = getTsets();
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a parent selector, e.g. `&`
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkParentSelector(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Ampersand ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a parent selector
|
|
* @return {Node}
|
|
*/
|
|
function getParentSelector() {
|
|
var type = NodeType.ParentSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '&';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a parent selector extension, e.g. `&--foo-bar`
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the parent selector extension
|
|
*/
|
|
function checkParentSelectorExtension(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get parent selector extension node
|
|
* @return {Node}
|
|
*/
|
|
function getParentSelectorExtension() {
|
|
var type = NodeType.ParentSelectorExtensionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkIdentOrInterpolation(pos)) {
|
|
content = content.concat(getIdentOrInterpolation());
|
|
} else if (checkPartialIdent(pos)) {
|
|
content.push(getIdent());
|
|
} else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a parent selector with an extension or not
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the parent selector and extension if applicable
|
|
*/
|
|
function checkParentSelectorWithExtension(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkParentSelector(i)) i += l;else return 0;
|
|
|
|
if (l = checkParentSelectorExtension(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get parent selector node and extension node if applicable
|
|
* @return {Array}
|
|
*/
|
|
function getParentSelectorWithExtension() {
|
|
var content = [getParentSelector()];
|
|
|
|
if (checkParentSelectorExtension(pos)) content.push(getParentSelectorExtension());
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number or an interpolation with a percent sign
|
|
* (e.g. `10%`).
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkPercentage(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkNumberOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Skip `%`.
|
|
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a percentage node that contains either a number or an interpolation
|
|
* @return {Object} The percentage node
|
|
*/
|
|
function getPercentage() {
|
|
var type = NodeType.PercentageType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = getNumberOrInterpolation();
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `%`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a number or an interpolation
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkNumberOrInterpolation(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkInterpolation(i) || checkNumber(i)) i += l;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a number and/or interpolation node
|
|
* @return {Array} An array containing a single or multiple nodes
|
|
*/
|
|
function getNumberOrInterpolation() {
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkNumber(pos)) content.push(getNumber());else break;
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a placeholder selector (e.g. `%abc`).
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the selector
|
|
*/
|
|
function checkPlaceholder(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[start].placeholder_l) return tokens[start].placeholder_l;
|
|
|
|
// Skip `%`.
|
|
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) {
|
|
i += l;
|
|
tokens[start].placeholder_l = i - start;
|
|
} else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a placeholder selector
|
|
* @return {Array} `['placeholder', ['ident', x]]` where x is a placeholder's
|
|
* identifier (without `%`, e.g. `abc`).
|
|
*/
|
|
function getPlaceholder() {
|
|
var type = NodeType.PlaceholderType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `%`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkProgid(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (joinValues2(i, 6) === 'progid:DXImageTransform.Microsoft.') i += 6;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.LeftParenthesis) {
|
|
tokens[start].progid_end = tokens[i].right;
|
|
i = tokens[i].right + 1;
|
|
} else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getProgid() {
|
|
var type = NodeType.ProgidType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var progid_end = token.progid_end;
|
|
var content = joinValues(pos, progid_end);
|
|
|
|
pos = progid_end + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a property
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the property
|
|
*/
|
|
function checkProperty(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkProperty1(i)) tokens[start].propertyType = 1;else if (l = checkProperty2(i)) tokens[start].propertyType = 2;else if (l = checkProperty3(i)) tokens[start].propertyType = 3;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* Get node with a property
|
|
* @return {!Node}
|
|
*/
|
|
function getProperty() {
|
|
var type = tokens[pos].propertyType;
|
|
|
|
if (type === 1) return getProperty1();
|
|
if (type === 2) return getProperty2();
|
|
if (type === 3) return getProperty3();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a property
|
|
* (1) `foo`
|
|
* (2) `#{$foo}`
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the property
|
|
*/
|
|
function checkProperty1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a property
|
|
* @returns {Array}
|
|
*/
|
|
function getProperty1() {
|
|
var type = NodeType.PropertyType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = getIdentOrInterpolation();
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a custom property
|
|
* (1) `--foo-bar`
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the property
|
|
*/
|
|
function checkProperty2(i) {
|
|
return checkCustomProperty(i);
|
|
}
|
|
|
|
/**
|
|
* Get node with a custom property
|
|
* @return {Node}
|
|
*/
|
|
function getProperty2() {
|
|
return getCustomProperty();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a property
|
|
* (1) `$foo`
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the property
|
|
*/
|
|
function checkProperty3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkVariable(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a property
|
|
* @returns {Array} `['property', x]`
|
|
*/
|
|
function getProperty3() {
|
|
var type = NodeType.PropertyType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getVariable()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a custom property
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the property
|
|
*/
|
|
function checkCustomProperty(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type !== TokenType.HyphenMinus || tokens[i + 1] && tokens[i + 1].type !== TokenType.HyphenMinus) return 0;
|
|
|
|
// Skip `--`
|
|
i += 2;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a custom property
|
|
* @return {Node}
|
|
*/
|
|
function getCustomProperty() {
|
|
var type = NodeType.CustomPropertyType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `--`
|
|
pos += 2;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a colon
|
|
* @param {number} i Token's index number
|
|
* @return {number} `1` if token is a colon, otherwise `0`
|
|
*/
|
|
function checkPropertyDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Colon ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a colon
|
|
* @return {Array} `['propertyDelim']`
|
|
*/
|
|
function getPropertyDelim() {
|
|
var type = NodeType.PropertyDelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ':';
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkPseudo(i) {
|
|
return checkPseudoe(i) || checkPseudoc(i);
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getPseudo() {
|
|
if (checkPseudoe(pos)) return getPseudoe();
|
|
if (checkPseudoc(pos)) return getPseudoc();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkPseudoe(i) {
|
|
var l = void 0;
|
|
|
|
// Check `::`
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.Colon || i + 1 >= tokensLength || tokens[i + 1].type !== TokenType.Colon) return 0;
|
|
|
|
if (l = checkPseudoElement1(i)) tokens[i].pseudoElementType = 1;else if (l = checkPseudoElement2(i)) tokens[i].pseudoElementType = 2;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getPseudoe() {
|
|
var childType = tokens[pos].pseudoElementType;
|
|
if (childType === 1) return getPseudoElement1();
|
|
if (childType === 2) return getPseudoElement2();
|
|
}
|
|
|
|
/**
|
|
* (1) `::slotted(selector)`
|
|
* (2) `::slotted(selector, selector)`
|
|
*/
|
|
function checkPseudoElement1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `::`.
|
|
i += 2;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* (1) `::slotted(selector)`
|
|
* (2) `::slotted(selector, selector)`
|
|
*/
|
|
function getPseudoElement1() {
|
|
var type = NodeType.PseudoeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `::`.
|
|
pos += 2;
|
|
|
|
content.push(getIdent());
|
|
|
|
{
|
|
var _type = NodeType.ArgumentsType;
|
|
var _token = tokens[pos];
|
|
var _line = _token.ln;
|
|
var _column = _token.col;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
|
|
|
|
var end = getLastPosition(selectorContent, _line, _column, 1);
|
|
var args = newNode(_type, selectorContent, _line, _column, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkPseudoElement2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `::`.
|
|
i += 2;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getPseudoElement2() {
|
|
var type = NodeType.PseudoeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `::`.
|
|
pos += 2;
|
|
|
|
var content = getIdentOrInterpolation();
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkPseudoc(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.Colon) return 0;
|
|
|
|
if (l = checkPseudoClass3(i)) tokens[i].pseudoClassType = 3;else if (l = checkPseudoClass4(i)) tokens[i].pseudoClassType = 4;else if (l = checkPseudoClass5(i)) tokens[i].pseudoClassType = 5;else if (l = checkPseudoClass1(i)) tokens[i].pseudoClassType = 1;else if (l = checkPseudoClass2(i)) tokens[i].pseudoClassType = 2;else if (l = checkPseudoClass6(i)) tokens[i].pseudoClassType = 6;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getPseudoc() {
|
|
var childType = tokens[pos].pseudoClassType;
|
|
if (childType === 1) return getPseudoClass1();
|
|
if (childType === 2) return getPseudoClass2();
|
|
if (childType === 3) return getPseudoClass3();
|
|
if (childType === 4) return getPseudoClass4();
|
|
if (childType === 5) return getPseudoClass5();
|
|
if (childType === 6) return getPseudoClass6();
|
|
}
|
|
|
|
/**
|
|
* (-) `:not(panda)`
|
|
*/
|
|
function checkPseudoClass1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* (-) `:not(panda)`
|
|
*/
|
|
function getPseudoClass1() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
{
|
|
var _type2 = NodeType.ArgumentsType;
|
|
var _token2 = tokens[pos];
|
|
var _line2 = _token2.ln;
|
|
var _column2 = _token2.col;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
|
|
|
|
var end = getLastPosition(selectorContent, _line2, _column2, 1);
|
|
var args = newNode(_type2, selectorContent, _line2, _column2, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `:nth-child(odd)`
|
|
* (2) `:nth-child(even)`
|
|
* (3) `:lang(de-DE)`
|
|
*/
|
|
function checkPseudoClass2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass2() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var value = [].concat(getSC(), getIdentOrInterpolation(), getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(-3n + 2)`
|
|
*/
|
|
function checkPseudoClass3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
|
|
if (l = checkNumberOrInterpolation(i)) i += l;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].value === 'n') i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.PlusSign || tokens[i].type === TokenType.HyphenMinus) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkNumberOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass3() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkNumberOrInterpolation(pos)) value = value.concat(getNumberOrInterpolation());
|
|
|
|
{
|
|
var _token3 = tokens[pos];
|
|
|
|
if (_token3.value === 'n') {
|
|
var _l = _token3.ln;
|
|
var _c = _token3.col;
|
|
var _content2 = _token3.value;
|
|
var ident = newNode(NodeType.IdentType, _content2, _l, _c);
|
|
value.push(ident);
|
|
pos++;
|
|
}
|
|
}
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkNumberOrInterpolation(pos)) value = value.concat(getNumberOrInterpolation());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(-3n)`
|
|
*/
|
|
function checkPseudoClass4(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
|
|
if (l = checkInterpolation(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;
|
|
|
|
if (tokens[i].value === 'n') i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass4() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkInterpolation(pos)) value.push(getInterpolation());
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
if (checkIdent(pos)) value.push(getIdent());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(+8)`
|
|
*/
|
|
function checkPseudoClass5(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass5() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:checked`
|
|
*/
|
|
function checkPseudoClass6(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass6() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
var content = getIdentOrInterpolation();
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkRuleset(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) {
|
|
i += l;
|
|
} else if (l = checkSC(i)) {
|
|
i += l;
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
} else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getRuleset() {
|
|
var type = NodeType.RulesetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getSelectorsGroup(), getSC());
|
|
|
|
if (checkBlock(pos)) {
|
|
content.push(getBlock());
|
|
} else {
|
|
content = content.concat(getSC(), getBlock());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is marked as a space (if it's a space or a tab
|
|
* or a line break).
|
|
* @param {number} i
|
|
* @return {number} Number of spaces in a row starting with the given token.
|
|
*/
|
|
function checkS(i) {
|
|
return i < tokensLength && tokens[i].ws ? tokens[i].ws_last - i + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with spaces
|
|
* @return {Array} `['s', x]` where `x` is a string containing spaces
|
|
*/
|
|
function getS() {
|
|
var type = NodeType.SType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, tokens[pos].ws_last);
|
|
|
|
pos = tokens[pos].ws_last + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a space, newline, or a comment.
|
|
* @param {number} i Token's index number
|
|
* @return {number} Number of similar (space, newline, or comment) tokens
|
|
* in a row starting with the given token.
|
|
*/
|
|
function checkMultilineSC(i) {
|
|
if (!tokens[i]) return 0;
|
|
|
|
var l = void 0;
|
|
var lsc = 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (!(l = checkS(i)) && !(l = checkCommentML(i)) && !(l = checkCommentSL(i))) break;
|
|
|
|
i += l;
|
|
lsc += l;
|
|
}
|
|
|
|
return lsc || 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with spaces newlines and comments
|
|
* @return {!Node}
|
|
*/
|
|
function getMultilineSC() {
|
|
var sc = [];
|
|
|
|
if (pos >= tokensLength) return sc;
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkS(pos)) sc.push(getS());else if (checkCommentML(pos)) sc.push(getCommentML());else if (checkCommentSL(pos)) sc.push(getCommentSL());else break;
|
|
}
|
|
|
|
return sc;
|
|
}
|
|
|
|
/**
|
|
* Check if token is a space or a comment.
|
|
* @param {number} i Token's index number
|
|
* @return {number} Number of similar (space or comment) tokens
|
|
* in a row starting with the given token.
|
|
*/
|
|
function checkSC(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
var lsc = 0;
|
|
var ln = tokens[i].ln;
|
|
|
|
while (i < tokensLength) {
|
|
if (tokens[i].ln !== ln) break;
|
|
|
|
if (l = checkS(i)) tokens[i].sc_child = 1;else if (l = checkCommentML(i)) tokens[i].sc_child = 2;else if (l = checkCommentSL(i)) tokens[i].sc_child = 3;else break;
|
|
|
|
i += l;
|
|
lsc += l;
|
|
|
|
if (tokens[i] && tokens[i].type === TokenType.Newline) break;
|
|
}
|
|
|
|
return lsc || 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with spaces and comments
|
|
* @return {Array} Array containing nodes with spaces (if there are any)
|
|
* and nodes with comments (if there are any):
|
|
* `[['s', x]*, ['comment', y]*]` where `x` is a string of spaces
|
|
* and `y` is a comment's text (without `/*` and `* /`).
|
|
*/
|
|
function getSC() {
|
|
var sc = [];
|
|
|
|
if (pos >= tokensLength) return sc;
|
|
|
|
var ln = tokens[pos].ln;
|
|
|
|
while (pos < tokensLength) {
|
|
if (tokens[pos].ln !== ln) break;else if (checkS(pos)) sc.push(getS());else if (checkCommentML(pos)) sc.push(getCommentML());else if (checkCommentSL(pos)) sc.push(getCommentSL());else break;
|
|
|
|
if (tokens[pos] && tokens[pos].type === TokenType.Newline) break;
|
|
}
|
|
|
|
return sc;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside a simple
|
|
* selector
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkShash(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else break;
|
|
}
|
|
|
|
tokens[start].shashEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a hexadecimal number (e.g. `#fff`) inside a simple selector
|
|
* @return {Node}
|
|
*/
|
|
function getShash() {
|
|
var type = NodeType.ShashType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = token.shashEnd;
|
|
var content = [];
|
|
|
|
// Skip `#`.
|
|
pos++;
|
|
|
|
while (pos < end) {
|
|
if (checkIdentOrInterpolation(pos)) {
|
|
content = content.concat(getIdentOrInterpolation());
|
|
} else if (checkPartialIdent(pos)) {
|
|
content.push(getIdent());
|
|
} else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a string (text wrapped in quotes)
|
|
* @param {number} i Token's index number
|
|
* @return {number} `1` if token is part of a string, `0` if not
|
|
*/
|
|
function checkString(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.StringSQ || tokens[i].type === TokenType.StringDQ) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get string's node
|
|
* @return {Array} `['string', x]` where `x` is a string (including
|
|
* quotes).
|
|
*/
|
|
function getString() {
|
|
var type = NodeType.StringType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Validate stylesheet: it should consist of any number (0 or more) of
|
|
* rulesets (sets of rules with selectors), @-rules, whitespaces or
|
|
* comments.
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkStylesheet(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkSC(i) || checkDeclaration(i) || checkDeclDelim(i) || checkInclude(i) || checkExtend(i) || checkMixin(i) || checkLoop(i) || checkConditionalStatement(i) || checkAtrule(i) || checkRuleset(i)) i += l;else throwError(i);
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array} `['stylesheet', x]` where `x` is all stylesheet's
|
|
* nodes.
|
|
*/
|
|
function getStylesheet() {
|
|
var type = NodeType.StylesheetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var node = void 0;
|
|
var wasDeclaration = false;
|
|
|
|
while (pos < tokensLength) {
|
|
if (wasDeclaration && checkDeclDelim(pos)) node = getDeclDelim();else if (checkSC(pos)) node = getSC();else if (checkRuleset(pos)) node = getRuleset();else if (checkInclude(pos)) node = getInclude();else if (checkExtend(pos)) node = getExtend();else if (checkMixin(pos)) node = getMixin();else if (checkLoop(pos)) node = getLoop();else if (checkConditionalStatement(pos)) node = getConditionalStatement();else if (checkAtrule(pos)) node = getAtrule();else if (checkDeclaration(pos)) node = getDeclaration();else throwError(pos);
|
|
|
|
wasDeclaration = node.type === NodeType.DeclarationType;
|
|
if (Array.isArray(node)) content = content.concat(node);else content.push(node);
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkTset(i) {
|
|
return checkVhash(i) || checkOperator(i) || checkAny(i) || checkSC(i);
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getTset() {
|
|
if (checkVhash(pos)) return getVhash();else if (checkOperator(pos)) return getOperator();else if (checkAny(pos)) return getAny();else if (checkSC(pos)) return getSC();
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkTsets(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (tokens[i - 1].type !== TokenType.Newline && (l = checkTset(i))) {
|
|
i += l;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getTsets() {
|
|
var content = [];
|
|
var t = void 0;
|
|
|
|
while (tokens[pos - 1].type !== TokenType.Newline && (t = getTset())) {
|
|
if (typeof t.content === 'string') content.push(t);else content = content.concat(t);
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is an unary (arithmetical) sign (`+` or `-`)
|
|
* @param {number} i Token's index number
|
|
* @return {number} `1` if token is an unary sign, `0` if not
|
|
*/
|
|
function checkUnary(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.HyphenMinus || tokens[i].type === TokenType.PlusSign) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an unary (arithmetical) sign (`+` or `-`)
|
|
* @return {Array} `['unary', x]` where `x` is an unary sign
|
|
* converted to string.
|
|
*/
|
|
function getUnary() {
|
|
var type = NodeType.OperatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a unicode range (single or multiple <urange> nodes)
|
|
* @param {number} i Token's index
|
|
* @return {number} Unicode range node's length
|
|
*/
|
|
function checkUnicodeRange(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkUrange(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
var comma = checkDelim(i + spaceBefore);
|
|
if (!comma) break;
|
|
|
|
var spaceAfter = checkSC(i + spaceBefore + comma);
|
|
if (l = checkUrange(i + spaceBefore + comma + spaceAfter)) {
|
|
i += spaceBefore + comma + spaceAfter + l;
|
|
} else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a unicode range node
|
|
* @return {Node}
|
|
*/
|
|
function getUnicodeRange() {
|
|
var type = NodeType.UnicodeRangeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkDelim(pos)) content.push(getDelim());else if (checkUrange(pos)) content.push(getUrange());else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is unit
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkUnit(i) {
|
|
var units = ['em', 'ex', 'ch', 'rem', 'vh', 'vw', 'vmin', 'vmax', 'px', 'mm', 'q', 'cm', 'in', 'pt', 'pc', 'deg', 'grad', 'rad', 'turn', 's', 'ms', 'Hz', 'kHz', 'dpi', 'dpcm', 'dppx'];
|
|
|
|
return units.indexOf(tokens[i].value) !== -1 ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get unit node of type ident
|
|
* @return {Node} An ident node containing the unit value
|
|
*/
|
|
function getUnit() {
|
|
var type = NodeType.IdentType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a u-range (part of a unicode-range)
|
|
* (1) `U+416`
|
|
* (2) `U+400-4ff`
|
|
* (3) `U+4??`
|
|
* @param {number} i Token's index
|
|
* @return {number} Urange node's length
|
|
*/
|
|
function checkUrange(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Check for unicode prefix (u+ or U+)
|
|
if (tokens[i].value === 'U' || tokens[i].value === 'u') i += 1;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].value === '+') i += 1;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = _checkUnicodeWildcard(i)) i += l;else break;
|
|
}
|
|
|
|
tokens[start].urangeEnd = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a u-range node (part of a unicode-range)
|
|
* @return {Node}
|
|
*/
|
|
function getUrange() {
|
|
var startPos = pos;
|
|
var type = NodeType.UrangeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content = joinValues(startPos, tokens[startPos].urangeEnd);
|
|
pos = tokens[startPos].urangeEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check for unicode wildcard characters `?`
|
|
* @param {number} i Token's index
|
|
* @return {number} Wildcard length
|
|
*/
|
|
function _checkUnicodeWildcard(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (tokens[i].type === TokenType.QuestionMark) i += 1;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of URI, e.g. `url('/css/styles.css')`
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of URI
|
|
*/
|
|
function checkUri(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].value !== 'url') return 0;
|
|
|
|
// Skip `url`.
|
|
i++;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
// Store the opening parenthesis token as we will reference it's `right`
|
|
// property to determine when the parentheses close
|
|
var leftParenthesis = tokens[i];
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
// Determine the type of URI
|
|
while (i < leftParenthesis.right) {
|
|
if (l = checkUri1(i)) {
|
|
i += l;
|
|
tokens[start].uriType = 1; // Raw based URI (without quotes)
|
|
} else if (l = checkUri2(i)) {
|
|
i += l;
|
|
tokens[start].uriType = 2; // Non-raw based URI (with quotes)
|
|
} else return 0;
|
|
}
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get specific type of URI node
|
|
* @return {Node} Specific type of URI node
|
|
*/
|
|
function getUri() {
|
|
var startPos = pos;
|
|
var type = NodeType.UriType;
|
|
var token = tokens[startPos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var end = void 0;
|
|
|
|
var uriType = tokens[startPos].uriType;
|
|
|
|
// Skip `url` and `(`.
|
|
pos += 2;
|
|
|
|
if (uriType === 1) content = content.concat(getUri1());else if (uriType === 2) content = content.concat(getUri2());else end = getLastPosition(content, line, column, 4);
|
|
|
|
if (!end) end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token type is valid URI character
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of raw node
|
|
*/
|
|
function checkUriRawCharacters(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else {
|
|
switch (tokens[i].type) {
|
|
case TokenType.ExclamationMark:
|
|
case TokenType.NumberSign:
|
|
case TokenType.DollarSign:
|
|
case TokenType.PercentSign:
|
|
case TokenType.Ampersand:
|
|
case TokenType.Asterisk:
|
|
case TokenType.PlusSign:
|
|
case TokenType.Comma:
|
|
case TokenType.HyphenMinus:
|
|
case TokenType.FullStop:
|
|
case TokenType.Solidus:
|
|
case TokenType.Colon:
|
|
case TokenType.Semicolon:
|
|
case TokenType.LessThanSign:
|
|
case TokenType.EqualsSign:
|
|
case TokenType.GreaterThanSign:
|
|
case TokenType.QuotationMark:
|
|
case TokenType.CommercialAt:
|
|
case TokenType.LeftSquareBracket:
|
|
case TokenType.RightSquareBracket:
|
|
case TokenType.CircumflexAccent:
|
|
case TokenType.LowLine:
|
|
case TokenType.LeftCurlyBracket:
|
|
case TokenType.VerticalLine:
|
|
case TokenType.RightCurlyBracket:
|
|
case TokenType.Tilde:
|
|
i += 1;
|
|
break;
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Check if content of URI can be contained within a raw node
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of raw node
|
|
*/
|
|
function checkUriRaw(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (checkInterpolation(i) || checkVariable(i)) break;else if (l = checkUriRawCharacters(i)) i += l;else break;
|
|
}
|
|
|
|
tokens[start].uri_raw_end = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a raw node
|
|
* @return {Node}
|
|
*/
|
|
function getUriRaw() {
|
|
var startPos = pos;
|
|
var type = NodeType.RawType;
|
|
var token = tokens[startPos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var l = void 0;
|
|
|
|
while (pos < tokens[startPos].uri_raw_end) {
|
|
if (checkInterpolation(pos) || checkVariable(pos)) break;else if (l = checkUriRawCharacters(pos)) pos += l;else break;
|
|
}
|
|
|
|
content = joinValues(startPos, pos - 1);
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check for a raw (without quotes) URI
|
|
* (1) http://foo.com/bar.png
|
|
* (2) http://foo.com/#{$bar}.png
|
|
* (3) #{$foo}/bar.png
|
|
* (4) #{$foo}
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of URI node
|
|
*/
|
|
function checkUri1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkInterpolation(i) || checkUriRaw(i)) i += l;else break;
|
|
}
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
// Check that we are at the end of the uri
|
|
if (i < tokens[start - 1].right) return 0;
|
|
|
|
tokens[start].uri_end = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a raw (without quotes) URI
|
|
node
|
|
* @return {Array}
|
|
*/
|
|
function getUri1() {
|
|
var startPos = pos;
|
|
var content = [];
|
|
|
|
if (checkSC(pos)) content = content.concat(getSC());
|
|
|
|
while (pos < tokens[startPos].uri_end) {
|
|
if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkUriRaw(pos)) content.push(getUriRaw());else break;
|
|
}
|
|
|
|
if (checkSC(pos)) content = content.concat(getSC());
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check for a non-raw (with quotes) URI
|
|
* (1) 'http://foo.com/bar.png'
|
|
* (2) 'http://foo.com/'#{$bar}.png
|
|
* (3) #{$foo}'/bar.png'
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of URI node
|
|
*/
|
|
function checkUri2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkSC(i)) i += l;else if (l = checkString(i)) i += l;else if (l = checkFunction(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = checkIdentOrInterpolation(i)) i += l;else if (l = checkVariable(i)) i += l;else break;
|
|
}
|
|
|
|
// Check that we are at the end of the uri
|
|
if (i < tokens[start - 1].right) return 0;
|
|
|
|
tokens[start].uri_end = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a non-raw (with quotes) URI node
|
|
* @return {Array}
|
|
*/
|
|
function getUri2() {
|
|
var startPos = pos;
|
|
var content = [];
|
|
|
|
while (pos < tokens[startPos].uri_end) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkUnary(pos)) content.push(getUnary());else if (_checkValue(pos)) content.push(_getValue());else break;
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a value
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the value
|
|
*/
|
|
function checkValue(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
var s = void 0;
|
|
var _i = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (checkDeclDelim(i)) break;
|
|
|
|
if (l = checkBlock(i)) {
|
|
i += l;
|
|
break;
|
|
}
|
|
|
|
s = checkS(i);
|
|
_i = i + s;
|
|
|
|
if (l = _checkValue(_i)) i += l + s;
|
|
if (!l || checkBlock(i - l)) break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function getValue() {
|
|
var type = NodeType.ValueType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var _pos = void 0;
|
|
var s = void 0;
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkDeclDelim(pos)) break;
|
|
|
|
s = checkS(pos);
|
|
_pos = pos + s;
|
|
|
|
if (checkDeclDelim(_pos)) break;
|
|
|
|
if (checkBlock(pos)) {
|
|
content.push(getBlock());
|
|
break;
|
|
}
|
|
|
|
if (!_checkValue(_pos)) break;
|
|
|
|
if (s) content.push(getS());
|
|
content.push(_getValue());
|
|
|
|
if (checkBlock(_pos)) break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function _checkValue(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkInterpolation(i)) tokens[i].value_child = 1;else if (l = checkVariable(i)) tokens[i].value_child = 2;else if (l = checkVhash(i)) tokens[i].value_child = 3;else if (l = checkBlock(i)) tokens[i].value_child = 4;else if (l = checkAtkeyword(i)) tokens[i].value_child = 5;else if (l = checkOperator(i)) tokens[i].value_child = 6;else if (l = checkImportant(i)) tokens[i].value_child = 7;else if (l = checkGlobal(i)) tokens[i].value_child = 8;else if (l = checkDefault(i)) tokens[i].value_child = 9;else if (l = checkProgid(i)) tokens[i].value_child = 10;else if (l = checkAny(i)) tokens[i].value_child = 11;else if (l = checkParentSelector(i)) tokens[i].value_child = 12;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @return {Array}
|
|
*/
|
|
function _getValue() {
|
|
var childType = tokens[pos].value_child;
|
|
if (childType === 1) return getInterpolation();
|
|
if (childType === 2) return getVariable();
|
|
if (childType === 3) return getVhash();
|
|
if (childType === 4) return getBlock();
|
|
if (childType === 5) return getAtkeyword();
|
|
if (childType === 6) return getOperator();
|
|
if (childType === 7) return getImportant();
|
|
if (childType === 8) return getGlobal();
|
|
if (childType === 9) return getDefault();
|
|
if (childType === 10) return getProgid();
|
|
if (childType === 11) return getAny();
|
|
if (childType === 12) return getParentSelector();
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the value
|
|
*/
|
|
function checkSingleValue(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
var s = void 0;
|
|
var _i = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (checkDeclDelim(i) || checkDelim(i)) break;
|
|
|
|
if (l = checkBlock(i)) {
|
|
i += l;
|
|
break;
|
|
}
|
|
|
|
s = checkSC(i);
|
|
_i = i + s;
|
|
|
|
if (l = _checkValue(_i)) i += l + s;
|
|
if (!l || checkBlock(i - l)) break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getSingleValue() {
|
|
var type = NodeType.ValueType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var _pos = void 0;
|
|
var s = void 0;
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkDeclDelim(pos) || checkDelim(pos)) break;
|
|
|
|
s = checkSC(pos);
|
|
_pos = pos + s;
|
|
|
|
if (checkDeclDelim(_pos) || checkDelim(_pos)) break;
|
|
|
|
if (checkBlock(pos)) {
|
|
content.push(getBlock());
|
|
break;
|
|
}
|
|
|
|
if (!_checkValue(_pos)) break;
|
|
|
|
if (s) content.push(getS());
|
|
content.push(_getValue());
|
|
|
|
if (checkBlock(_pos)) break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a variable
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the variable
|
|
*/
|
|
function checkVariable(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Skip `$`.
|
|
if (tokens[i].type === TokenType.DollarSign) i++;else return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a variable
|
|
* @return {Array} `['variable', ['ident', x]]` where `x` is
|
|
* a variable name.
|
|
*/
|
|
function getVariable() {
|
|
var type = NodeType.VariableType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `$`.
|
|
pos++;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a variables list (e.g. `$values...`).
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkVariablesList(i) {
|
|
var d = 0; // Number of dots
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkVariable(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength && tokens[i].type === TokenType.FullStop) {
|
|
d++;
|
|
i++;
|
|
}
|
|
|
|
return d === 3 ? l + d : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a variables list
|
|
* @return {Array} `['variableslist', ['variable', ['ident', x]]]` where
|
|
* `x` is a variable name.
|
|
*/
|
|
function getVariablesList() {
|
|
var type = NodeType.VariablesListType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getVariable()];
|
|
var end = getLastPosition(content, line, column, 3);
|
|
|
|
// Skip `...`.
|
|
pos += 3;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
|
|
* some value
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkVhash(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Skip `#`.
|
|
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
|
|
|
|
if (l = checkNmName2(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a hexadecimal number (e.g. `#fff`) inside some value
|
|
* @return {Array} `['vhash', x]` where `x` is a hexadecimal number
|
|
* converted to string (without `#`, e.g. `'fff'`).
|
|
*/
|
|
function getVhash() {
|
|
var type = NodeType.VhashType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `#`.
|
|
pos++;
|
|
|
|
var content = getNmName2();
|
|
var end = getLastPosition(content, line, column + 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkSelectorsGroup(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSelector(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
var comma = checkDelim(i + spaceBefore);
|
|
if (!comma) break;
|
|
|
|
var spaceAfter = checkMultilineSC(i + spaceBefore + comma);
|
|
var spaceEnd = spaceAfter ? checkMultilineSC(i + spaceBefore + comma + spaceAfter) : 0;
|
|
|
|
if (l = checkSelector(i + spaceBefore + comma + spaceAfter + spaceEnd)) i += spaceBefore + comma + spaceAfter + spaceEnd + l;else break;
|
|
}
|
|
|
|
tokens[start].selectorsGroupEnd = i;
|
|
return i - start;
|
|
}
|
|
|
|
function getSelectorsGroup() {
|
|
var selectorsGroup = [];
|
|
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
|
|
|
|
selectorsGroup.push(getSelector());
|
|
|
|
while (pos < selectorsGroupEnd) {
|
|
selectorsGroup = selectorsGroup.concat(getMultilineSC(), getDelim(), getMultilineSC(), getSelector());
|
|
}
|
|
|
|
return selectorsGroup;
|
|
}
|
|
|
|
function checkSelector(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkSelector1(i)) tokens[i].selectorType = 1;else if (l = checkSelector2(i)) tokens[i].selectorType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getSelector() {
|
|
var selectorType = tokens[pos].selectorType;
|
|
if (selectorType === 1) return getSelector1();else return getSelector2();
|
|
}
|
|
|
|
/**
|
|
* Checks for selector which starts with a compound selector.
|
|
*/
|
|
function checkSelector1(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkCompoundSelector(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var space = checkSC(i);
|
|
var comma = checkCombinator(i + space);
|
|
if (!space && !comma) break;
|
|
|
|
if (comma) {
|
|
i += space + comma;
|
|
space = checkSC(i);
|
|
}
|
|
|
|
if (l = checkCompoundSelector(i + space)) i += space + l;else break;
|
|
}
|
|
|
|
tokens[start].selectorEnd = i;
|
|
return i - start;
|
|
}
|
|
|
|
function getSelector1() {
|
|
var type = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var selectorEnd = token.selectorEnd;
|
|
var content = getCompoundSelector();
|
|
|
|
while (pos < selectorEnd) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Checks for a selector that starts with a combinator.
|
|
*/
|
|
function checkSelector2(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkCombinator(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
if (l = checkCompoundSelector(i + spaceBefore)) i += spaceBefore + l;else break;
|
|
|
|
var spaceAfter = checkSC(i);
|
|
var comma = checkCombinator(i + spaceAfter);
|
|
if (!spaceAfter && !comma) break;
|
|
if (comma) {
|
|
i += spaceAfter + comma;
|
|
}
|
|
}
|
|
|
|
tokens[start].selectorEnd = i;
|
|
return i - start;
|
|
}
|
|
|
|
function getSelector2() {
|
|
var type = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var selectorEnd = token.selectorEnd;
|
|
var content = [getCombinator()];
|
|
|
|
while (pos < selectorEnd) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkCompoundSelector(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkCompoundSelector1(i)) {
|
|
tokens[i].compoundSelectorType = 1;
|
|
} else if (l = checkCompoundSelector2(i)) {
|
|
tokens[i].compoundSelectorType = 2;
|
|
}
|
|
|
|
return l;
|
|
}
|
|
|
|
function getCompoundSelector() {
|
|
var type = tokens[pos].compoundSelectorType;
|
|
if (type === 1) return getCompoundSelector1();
|
|
if (type === 2) return getCompoundSelector2();
|
|
}
|
|
|
|
/**
|
|
* Check for compound selectors that start with either a type selector,
|
|
* placeholder or parent selector with extension
|
|
* (1) `foo.bar`
|
|
* (2) `foo[attr=val]`
|
|
* (3) `foo:first-of-type`
|
|
* (4) `foo%bar`
|
|
* @param {number} i Token's index
|
|
* @return {number} Compound selector's length
|
|
*/
|
|
function checkCompoundSelector1(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkUniversalSelector(i) || checkTypeSelector(i) || checkPlaceholder(i) || checkParentSelectorWithExtension(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var _l2 = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i) || checkPlaceholder(i) || checkInterpolation(i);
|
|
|
|
if (_l2) i += _l2;else break;
|
|
}
|
|
|
|
tokens[start].compoundSelectorEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array} An array of nodes that make up the compound selector
|
|
*/
|
|
function getCompoundSelector1() {
|
|
var sequence = [];
|
|
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
|
|
|
|
if (checkUniversalSelector(pos)) sequence.push(getUniversalSelector());else if (checkTypeSelector(pos)) sequence.push(getTypeSelector());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkParentSelectorWithExtension(pos)) sequence = sequence.concat(getParentSelectorWithExtension());
|
|
|
|
while (pos < compoundSelectorEnd) {
|
|
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkInterpolation(pos)) sequence.push(getInterpolation());else break;
|
|
}
|
|
|
|
return sequence;
|
|
}
|
|
|
|
/**
|
|
* Check for all other compound selectors
|
|
* (1) `.foo.bar`
|
|
* (2) `.foo[attr=val]`
|
|
* (3) `.foo:first-of-type`
|
|
* (4) `.foo%bar`
|
|
* (5) `.foo#{$bar}`
|
|
* @param {number} i Token's index
|
|
* @return {number} Compound selector's length
|
|
*/
|
|
function checkCompoundSelector2(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
|
|
while (i < tokensLength) {
|
|
var l = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i) || checkPlaceholder(i) || checkInterpolation(i);
|
|
|
|
if (l) i += l;else break;
|
|
}
|
|
|
|
tokens[start].compoundSelectorEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array} An array of nodes that make up the compound selector
|
|
*/
|
|
function getCompoundSelector2() {
|
|
var sequence = [];
|
|
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
|
|
|
|
while (pos < compoundSelectorEnd) {
|
|
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkInterpolation(pos)) sequence.push(getInterpolation());else break;
|
|
}
|
|
|
|
return sequence;
|
|
}
|
|
|
|
function checkUniversalSelector(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.Asterisk) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getUniversalSelector() {
|
|
var type = NodeType.UniversalSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var end = void 0;
|
|
|
|
if (checkNamePrefix(pos)) {
|
|
content.push(getNamePrefix());
|
|
end = getLastPosition(content, line, column, 1);
|
|
}
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a type selector
|
|
* @param {number} i Token's index
|
|
* @return {number} Type selector's length
|
|
*/
|
|
function checkTypeSelector(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get type selector node
|
|
* @return {Node}
|
|
*/
|
|
function getTypeSelector() {
|
|
var type = NodeType.TypeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkNamePrefix(pos)) content.push(getNamePrefix());
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeSelector(i) {
|
|
var l = void 0;
|
|
if (l = checkAttributeSelector1(i)) tokens[i].attributeSelectorType = 1;else if (l = checkAttributeSelector2(i)) tokens[i].attributeSelectorType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getAttributeSelector() {
|
|
var type = tokens[pos].attributeSelectorType;
|
|
if (type === 1) return getAttributeSelector1();else return getAttributeSelector2();
|
|
}
|
|
|
|
/**
|
|
* (1) `[panda=nani]`
|
|
* (2) `[panda='nani']`
|
|
* (3) `[panda='nani' i]`
|
|
*
|
|
*/
|
|
function checkAttributeSelector1(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeName(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeMatch(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeValue(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeFlags(i)) {
|
|
i += l;
|
|
if (l = checkSC(i)) i += l;
|
|
}
|
|
|
|
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeSelector1() {
|
|
var type = NodeType.AttributeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC(), getAttributeName(), getSC(), getAttributeMatch(), getSC(), getAttributeValue(), getSC());
|
|
|
|
if (checkAttributeFlags(pos)) {
|
|
content.push(getAttributeFlags());
|
|
content = content.concat(getSC());
|
|
}
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* (1) `[panda]`
|
|
*/
|
|
function checkAttributeSelector2(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeName(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeSelector2() {
|
|
var type = NodeType.AttributeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC(), getAttributeName(), getSC());
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkAttributeName(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeName() {
|
|
var type = NodeType.AttributeNameType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkNamePrefix(pos)) content.push(getNamePrefix());
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeMatch(i) {
|
|
var l = void 0;
|
|
if (l = checkAttributeMatch1(i)) tokens[i].attributeMatchType = 1;else if (l = checkAttributeMatch2(i)) tokens[i].attributeMatchType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getAttributeMatch() {
|
|
var type = tokens[pos].attributeMatchType;
|
|
if (type === 1) return getAttributeMatch1();else return getAttributeMatch2();
|
|
}
|
|
|
|
function checkAttributeMatch1(i) {
|
|
var start = i;
|
|
|
|
var type = tokens[i].type;
|
|
if (type === TokenType.Tilde || type === TokenType.VerticalLine || type === TokenType.CircumflexAccent || type === TokenType.DollarSign || type === TokenType.Asterisk) i++;else return 0;
|
|
|
|
if (tokens[i].type === TokenType.EqualsSign) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeMatch1() {
|
|
var type = NodeType.AttributeMatchType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos].value + tokens[pos + 1].value;
|
|
pos += 2;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeMatch2(i) {
|
|
if (tokens[i].type === TokenType.EqualsSign) return 1;else return 0;
|
|
}
|
|
|
|
function getAttributeMatch2() {
|
|
var type = NodeType.AttributeMatchType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '=';
|
|
|
|
pos++;
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeValue(i) {
|
|
return checkString(i) || checkIdentOrInterpolation(i);
|
|
}
|
|
|
|
function getAttributeValue() {
|
|
var type = NodeType.AttributeValueType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkString(pos)) content.push(getString());else content = content.concat(getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeFlags(i) {
|
|
return checkIdentOrInterpolation(i);
|
|
}
|
|
|
|
function getAttributeFlags() {
|
|
var type = NodeType.AttributeFlagsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = getIdentOrInterpolation();
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkNamePrefix(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkNamePrefix1(i)) tokens[i].namePrefixType = 1;else if (l = checkNamePrefix2(i)) tokens[i].namePrefixType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getNamePrefix() {
|
|
var type = tokens[pos].namePrefixType;
|
|
if (type === 1) return getNamePrefix1();else return getNamePrefix2();
|
|
}
|
|
|
|
/**
|
|
* (1) `panda|`
|
|
* (2) `panda<comment>|`
|
|
*/
|
|
function checkNamePrefix1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamespacePrefix(i)) i += l;else return 0;
|
|
|
|
if (l = checkCommentML(i)) i += l;
|
|
|
|
if (l = checkNamespaceSeparator(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getNamePrefix1() {
|
|
var type = NodeType.NamePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(getNamespacePrefix());
|
|
|
|
if (checkCommentML(pos)) content.push(getCommentML());
|
|
|
|
content.push(getNamespaceSeparator());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `|`
|
|
*/
|
|
function checkNamePrefix2(i) {
|
|
return checkNamespaceSeparator(i);
|
|
}
|
|
|
|
function getNamePrefix2() {
|
|
var type = NodeType.NamePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getNamespaceSeparator()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `*`
|
|
* (2) `panda`
|
|
*/
|
|
function checkNamespacePrefix(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
|
|
if (tokens[i].type === TokenType.Asterisk) return 1;else if (l = checkIdentOrInterpolation(i)) return l;else return 0;
|
|
}
|
|
|
|
function getNamespacePrefix() {
|
|
var type = NodeType.NamespacePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (token.type === TokenType.Asterisk) {
|
|
var asteriskNode = newNode(NodeType.IdentType, '*', line, column);
|
|
content.push(asteriskNode);
|
|
pos++;
|
|
} else if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `|`
|
|
*/
|
|
function checkNamespaceSeparator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type !== TokenType.VerticalLine) return 0;
|
|
|
|
// Return false if `|=` - [attr|=value]
|
|
if (tokens[i + 1] && tokens[i + 1].type === TokenType.EqualsSign) return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
function getNamespaceSeparator() {
|
|
var type = NodeType.NamespaceSeparatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '|';
|
|
|
|
pos++;
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
module.exports = function (_tokens, context) {
|
|
tokens = _tokens;
|
|
tokensLength = tokens.length;
|
|
pos = 0;
|
|
|
|
return contexts[context]();
|
|
};
|
|
|
|
/***/ }),
|
|
/* 24 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
module.exports = function (css, tabSize) {
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
var tokens = [];
|
|
var urlMode = false;
|
|
var c = void 0; // Current character
|
|
var cn = void 0; // Next character
|
|
var pos = 0;
|
|
var tn = 0;
|
|
var ln = 1;
|
|
var col = 1;
|
|
|
|
var Punctuation = {
|
|
' ': TokenType.Space,
|
|
'\n': TokenType.Newline,
|
|
'\r': TokenType.Newline,
|
|
'\t': TokenType.Tab,
|
|
'!': TokenType.ExclamationMark,
|
|
'"': TokenType.QuotationMark,
|
|
'#': TokenType.NumberSign,
|
|
'$': TokenType.DollarSign,
|
|
'%': TokenType.PercentSign,
|
|
'&': TokenType.Ampersand,
|
|
'\'': TokenType.Apostrophe,
|
|
'(': TokenType.LeftParenthesis,
|
|
')': TokenType.RightParenthesis,
|
|
'*': TokenType.Asterisk,
|
|
'+': TokenType.PlusSign,
|
|
',': TokenType.Comma,
|
|
'-': TokenType.HyphenMinus,
|
|
'.': TokenType.FullStop,
|
|
'/': TokenType.Solidus,
|
|
':': TokenType.Colon,
|
|
';': TokenType.Semicolon,
|
|
'<': TokenType.LessThanSign,
|
|
'=': TokenType.EqualsSign,
|
|
'==': TokenType.EqualitySign,
|
|
'!=': TokenType.InequalitySign,
|
|
'>': TokenType.GreaterThanSign,
|
|
'?': TokenType.QuestionMark,
|
|
'@': TokenType.CommercialAt,
|
|
'[': TokenType.LeftSquareBracket,
|
|
']': TokenType.RightSquareBracket,
|
|
'^': TokenType.CircumflexAccent,
|
|
'_': TokenType.LowLine,
|
|
'{': TokenType.LeftCurlyBracket,
|
|
'|': TokenType.VerticalLine,
|
|
'}': TokenType.RightCurlyBracket,
|
|
'~': TokenType.Tilde,
|
|
'`': TokenType.Backtick
|
|
};
|
|
|
|
/**
|
|
* Add a token to the token list
|
|
* @param {string} type
|
|
* @param {string} value
|
|
*/
|
|
function pushToken(type, value, column) {
|
|
tokens.push({
|
|
tn: tn++,
|
|
ln: ln,
|
|
col: column,
|
|
type: type,
|
|
value: value
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Check if a character is a decimal digit
|
|
* @param {string} c Character
|
|
* @returns {boolean}
|
|
*/
|
|
function isDecimalDigit(c) {
|
|
return '0123456789'.indexOf(c) >= 0;
|
|
}
|
|
|
|
/**
|
|
* Parse spaces
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseSpaces(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a non-space character:
|
|
for (; pos < css.length; pos++) {
|
|
if (css.charAt(pos) !== ' ') break;
|
|
}
|
|
|
|
// Add a substring containing only spaces to tokens:
|
|
pushToken(TokenType.Space, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse a string within quotes
|
|
* @param {string} css Unparsed part of CSS string
|
|
* @param {string} q Quote (either `'` or `"`)
|
|
*/
|
|
function parseString(css, q) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a matching quote:
|
|
for (pos++; pos < css.length; pos++) {
|
|
// Skip escaped quotes:
|
|
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) === q) break;
|
|
}
|
|
|
|
// Add the string (including quotes) to tokens:
|
|
var type = q === '"' ? TokenType.StringDQ : TokenType.StringSQ;
|
|
pushToken(type, css.substring(start, pos + 1), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse numbers
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseDecimalNumber(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a character that's not a digit:
|
|
for (; pos < css.length; pos++) {
|
|
if (!isDecimalDigit(css.charAt(pos))) break;
|
|
}
|
|
|
|
// Add the number to tokens:
|
|
pushToken(TokenType.DecimalNumber, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse identifier
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseIdentifier(css) {
|
|
var start = pos;
|
|
|
|
// Skip all opening slashes:
|
|
while (css.charAt(pos) === '/') {
|
|
pos++;
|
|
} // Read the string until we meet a punctuation mark:
|
|
for (; pos < css.length; pos++) {
|
|
// Skip all '\':
|
|
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) in Punctuation) break;
|
|
}
|
|
|
|
var ident = css.substring(start, pos--);
|
|
|
|
// Enter url mode if parsed substring is `url`:
|
|
if (!urlMode && ident === 'url' && css.charAt(pos + 1) === '(') {
|
|
urlMode = true;
|
|
}
|
|
|
|
// Add identifier to tokens:
|
|
pushToken(TokenType.Identifier, ident, col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse equality sign
|
|
*/
|
|
function parseEquality() {
|
|
pushToken(TokenType.EqualitySign, '==', col);
|
|
pos++;
|
|
col++;
|
|
}
|
|
|
|
/**
|
|
* Parse inequality sign
|
|
*/
|
|
function parseInequality() {
|
|
pushToken(TokenType.InequalitySign, '!=', col);
|
|
pos++;
|
|
col++;
|
|
}
|
|
|
|
/**
|
|
* Parse a multiline comment
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseMLComment(css) {
|
|
var start = pos;
|
|
var col_ = col;
|
|
|
|
// Get current indent level:
|
|
var il = 0;
|
|
for (var _pos = pos - 1; _pos > -1; _pos--) {
|
|
// TODO: Can be tabs:
|
|
if (css.charAt(_pos) === ' ') il++;else break;
|
|
}
|
|
|
|
for (pos += 2; pos < css.length; pos++) {
|
|
var ch = css.charAt(pos);
|
|
if (ch === '\n') {
|
|
var _pos2 = void 0;
|
|
// Get new line's indent level:
|
|
var _il = 0;
|
|
for (_pos2 = pos + 1; _pos2 < css.length; _pos2++) {
|
|
if (css.charAt(_pos2) === ' ') _il++;else break;
|
|
}
|
|
|
|
if (_il > il) {
|
|
col = 0;
|
|
pos += _pos2 - pos;
|
|
} else {
|
|
pos--;
|
|
break;
|
|
}
|
|
} else if (ch === '*' && css.charAt(pos + 1) === '/') {
|
|
pos++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// If CRLF is used, we need to adjust pos
|
|
if (css.charAt(pos) === '\r') pos--;
|
|
|
|
// Add full comment (including `/*`) to the list of tokens:
|
|
var comment = css.substring(start, pos + 1);
|
|
pushToken(TokenType.CommentML, comment, col_);
|
|
|
|
var newlines = comment.split('\n');
|
|
if (newlines.length > 1) {
|
|
ln += newlines.length - 1;
|
|
col = newlines[newlines.length - 1].length;
|
|
} else {
|
|
col += pos - start;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parse a single line comment
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseSLComment(css) {
|
|
var start = pos;
|
|
var col_ = col;
|
|
var _pos;
|
|
|
|
// Check if comment is the only token on the line, and if so,
|
|
// get current indent level:
|
|
var il = 0;
|
|
var onlyToken = false;
|
|
for (_pos = pos - 1; _pos > -1; _pos--) {
|
|
// TODO: Can be tabs:
|
|
if (css.charAt(_pos) === ' ') il++;else if (css.charAt(_pos) === '\n') {
|
|
onlyToken = true;
|
|
break;
|
|
} else break;
|
|
}
|
|
if (_pos === -1) onlyToken = true;
|
|
|
|
// Read the string until we meet comment end.
|
|
// Since we already know first 2 characters (`//`), start reading
|
|
// from `pos + 2`:
|
|
if (!onlyToken) {
|
|
for (pos += 2; pos < css.length; pos++) {
|
|
if (css.charAt(pos) === '\n' || css.charAt(pos) === '\r') {
|
|
break;
|
|
}
|
|
}
|
|
} else {
|
|
for (pos += 2; pos < css.length; pos++) {
|
|
var ch = css.charAt(pos);
|
|
if (ch === '\n') {
|
|
// Get new line's indent level:
|
|
var _il = 0;
|
|
for (_pos = pos + 1; _pos < css.length; _pos++) {
|
|
if (css.charAt(_pos) === ' ') _il++;else break;
|
|
}
|
|
|
|
if (_il > il) {
|
|
col = 0;
|
|
pos += _pos - pos;
|
|
} else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// If CRLF is used, we need to adjust pos
|
|
if (css.charAt(pos - 1) === '\r') pos--;
|
|
|
|
// Add comment (including `//` and line break) to the list of tokens:
|
|
var comment = css.substring(start, pos--);
|
|
pushToken(TokenType.CommentSL, comment, col_);
|
|
|
|
var newlines = comment.split('\n');
|
|
if (newlines.length > 1) {
|
|
ln += newlines.length - 1;
|
|
col = newlines[newlines.length - 1].length;
|
|
} else {
|
|
col += pos - start;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Convert a CSS string to a list of tokens
|
|
* @param {string} css CSS string
|
|
* @returns {Array} List of tokens
|
|
* @private
|
|
*/
|
|
function getTokens(css) {
|
|
// Parse string, character by character:
|
|
for (pos = 0; pos < css.length; col++, pos++) {
|
|
c = css.charAt(pos);
|
|
cn = css.charAt(pos + 1);
|
|
|
|
// If we meet `/*`, it's a start of a multiline comment.
|
|
// Parse following characters as a multiline comment:
|
|
if (c === '/' && cn === '*') {
|
|
parseMLComment(css);
|
|
}
|
|
|
|
// If we meet `//` and it is not a part of url:
|
|
else if (!urlMode && c === '/' && cn === '/') {
|
|
// If we're currently inside a block, treat `//` as a start
|
|
// of identifier. Else treat `//` as a start of a single-line
|
|
// comment:
|
|
parseSLComment(css);
|
|
}
|
|
|
|
// If current character is a double or single quote, it's a start
|
|
// of a string:
|
|
else if (c === '"' || c === "'") {
|
|
parseString(css, c);
|
|
}
|
|
|
|
// If current character is a space:
|
|
else if (c === ' ') {
|
|
parseSpaces(css);
|
|
}
|
|
|
|
// If current character is `=`, it must be combined with next `=`
|
|
else if (c === '=' && cn === '=') {
|
|
parseEquality(css);
|
|
}
|
|
|
|
// If we meet `!=`, this must be inequality
|
|
else if (c === '!' && cn === '=') {
|
|
parseInequality(css);
|
|
}
|
|
|
|
// If current character is a punctuation mark:
|
|
else if (c in Punctuation) {
|
|
// Check for CRLF here or just LF
|
|
if (c === '\r' && cn === '\n' || c === '\n') {
|
|
// If \r we know the next character is \n due to statement above
|
|
// so we push a CRLF token type to the token list and importantly
|
|
// skip the next character so as not to double count newlines or
|
|
// columns etc
|
|
if (c === '\r') {
|
|
pushToken(TokenType.Newline, '\r\n', col);
|
|
pos++; // If CRLF skip the next character and push crlf token
|
|
} else if (c === '\n') {
|
|
// If just a LF newline and not part of CRLF newline we can just
|
|
// push punctuation as usual
|
|
pushToken(Punctuation[c], c, col);
|
|
}
|
|
|
|
ln++; // Go to next line
|
|
col = 0; // Reset the column count
|
|
} else if (c !== '\r' && c !== '\n') {
|
|
// Handle all other punctuation and add to list of tokens
|
|
pushToken(Punctuation[c], c, col);
|
|
} // Go to next line
|
|
if (c === ')') urlMode = false; // Exit url mode
|
|
else if (c === '\t' && tabSize > 1) col += tabSize - 1;
|
|
}
|
|
|
|
// If current character is a decimal digit:
|
|
else if (isDecimalDigit(c)) {
|
|
parseDecimalNumber(css);
|
|
}
|
|
|
|
// If current character is anything else:
|
|
else {
|
|
parseIdentifier(css);
|
|
}
|
|
}
|
|
|
|
return tokens;
|
|
}
|
|
|
|
return getTokens(css);
|
|
};
|
|
|
|
/***/ }),
|
|
/* 25 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
exports.__esModule = true;
|
|
exports.default = {
|
|
mark: __webpack_require__(26),
|
|
parse: __webpack_require__(27),
|
|
stringify: __webpack_require__(6),
|
|
tokenizer: __webpack_require__(28)
|
|
};
|
|
module.exports = exports['default'];
|
|
|
|
/***/ }),
|
|
/* 26 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
module.exports = function () {
|
|
/**
|
|
* Mark whitespaces and comments
|
|
*/
|
|
function markSC(tokens) {
|
|
var tokensLength = tokens.length;
|
|
var ws = -1; // Flag for whitespaces
|
|
var sc = -1; // Flag for whitespaces and comments
|
|
var t = void 0; // Current token
|
|
|
|
// For every token in the token list, mark spaces and line breaks
|
|
// as spaces (set both `ws` and `sc` flags). Mark multiline comments
|
|
// with `sc` flag.
|
|
// If there are several spaces or tabs or line breaks or multiline
|
|
// comments in a row, group them: take the last one's index number
|
|
// and save it to the first token in the group as a reference:
|
|
// e.g., `ws_last = 7` for a group of whitespaces or `sc_last = 9`
|
|
// for a group of whitespaces and comments.
|
|
for (var i = 0; i < tokensLength; i++) {
|
|
t = tokens[i];
|
|
switch (t.type) {
|
|
case TokenType.Space:
|
|
case TokenType.Tab:
|
|
case TokenType.Newline:
|
|
t.ws = true;
|
|
t.sc = true;
|
|
|
|
if (ws === -1) ws = i;
|
|
if (sc === -1) sc = i;
|
|
|
|
break;
|
|
case TokenType.CommentML:
|
|
case TokenType.CommentSL:
|
|
if (ws !== -1) {
|
|
tokens[ws].ws_last = i - 1;
|
|
ws = -1;
|
|
}
|
|
|
|
t.sc = true;
|
|
|
|
break;
|
|
default:
|
|
if (ws !== -1) {
|
|
tokens[ws].ws_last = i - 1;
|
|
ws = -1;
|
|
}
|
|
|
|
if (sc !== -1) {
|
|
tokens[sc].sc_last = i - 1;
|
|
sc = -1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (ws !== -1) tokens[ws].ws_last = i - 1;
|
|
if (sc !== -1) tokens[sc].sc_last = i - 1;
|
|
}
|
|
|
|
/**
|
|
* Pair brackets
|
|
*/
|
|
function markBrackets(tokens) {
|
|
var tokensLength = tokens.length;
|
|
var ps = []; // Parentheses
|
|
var sbs = []; // Square brackets
|
|
var cbs = []; // Curly brackets
|
|
var t = void 0; // Current token
|
|
|
|
// For every token in the token list, if we meet an opening (left)
|
|
// bracket, push its index number to a corresponding array.
|
|
// If we then meet a closing (right) bracket, look at the corresponding
|
|
// array. If there are any elements (records about previously met
|
|
// left brackets), take a token of the last left bracket (take
|
|
// the last index number from the array and find a token with
|
|
// this index number) and save right bracket's index as a reference:
|
|
for (var i = 0; i < tokensLength; i++) {
|
|
t = tokens[i];
|
|
switch (t.type) {
|
|
case TokenType.LeftParenthesis:
|
|
ps.push(i);
|
|
break;
|
|
case TokenType.RightParenthesis:
|
|
if (ps.length) {
|
|
t.left = ps.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
break;
|
|
case TokenType.LeftSquareBracket:
|
|
sbs.push(i);
|
|
break;
|
|
case TokenType.RightSquareBracket:
|
|
if (sbs.length) {
|
|
t.left = sbs.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
break;
|
|
case TokenType.LeftCurlyBracket:
|
|
cbs.push(i);
|
|
break;
|
|
case TokenType.RightCurlyBracket:
|
|
if (cbs.length) {
|
|
t.left = cbs.pop();
|
|
tokens[t.left].right = i;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return function (tokens) {
|
|
markBrackets(tokens);
|
|
markSC(tokens);
|
|
};
|
|
}();
|
|
|
|
/***/ }),
|
|
/* 27 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var Node = __webpack_require__(1);
|
|
var NodeType = __webpack_require__(15);
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
var tokens = void 0;
|
|
var tokensLength = void 0;
|
|
var pos = void 0;
|
|
|
|
var contexts = {
|
|
'arguments': function _arguments() {
|
|
return checkArguments(pos) && getArguments();
|
|
},
|
|
'atkeyword': function atkeyword() {
|
|
return checkAtkeyword(pos) && getAtkeyword();
|
|
},
|
|
'atrule': function atrule() {
|
|
return checkAtrule(pos) && getAtrule();
|
|
},
|
|
'attributeSelector': function attributeSelector() {
|
|
return checkAttributeSelector(pos) && getAttributeSelector();
|
|
},
|
|
'block': function block() {
|
|
return checkBlock(pos) && getBlock();
|
|
},
|
|
'brackets': function brackets() {
|
|
return checkBrackets(pos) && getBrackets();
|
|
},
|
|
'class': function _class() {
|
|
return checkClass(pos) && getClass();
|
|
},
|
|
'combinator': function combinator() {
|
|
return checkCombinator(pos) && getCombinator();
|
|
},
|
|
'commentML': function commentML() {
|
|
return checkCommentML(pos) && getCommentML();
|
|
},
|
|
'commentSL': function commentSL() {
|
|
return checkCommentSL(pos) && getCommentSL();
|
|
},
|
|
'condition': function condition() {
|
|
return checkCondition(pos) && getCondition();
|
|
},
|
|
'conditionalStatement': function conditionalStatement() {
|
|
return checkConditionalStatement(pos) && getConditionalStatement();
|
|
},
|
|
'declaration': function declaration() {
|
|
return checkDeclaration(pos) && getDeclaration();
|
|
},
|
|
'declDelim': function declDelim() {
|
|
return checkDeclDelim(pos) && getDeclDelim();
|
|
},
|
|
'default': function _default() {
|
|
return checkDefault(pos) && getDefault();
|
|
},
|
|
'delim': function delim() {
|
|
return checkDelim(pos) && getDelim();
|
|
},
|
|
'dimension': function dimension() {
|
|
return checkDimension(pos) && getDimension();
|
|
},
|
|
'expression': function expression() {
|
|
return checkExpression(pos) && getExpression();
|
|
},
|
|
'extend': function extend() {
|
|
return checkExtend(pos) && getExtend();
|
|
},
|
|
'function': function _function() {
|
|
return checkFunction(pos) && getFunction();
|
|
},
|
|
'global': function global() {
|
|
return checkGlobal(pos) && getGlobal();
|
|
},
|
|
'ident': function ident() {
|
|
return checkIdent(pos) && getIdent();
|
|
},
|
|
'important': function important() {
|
|
return checkImportant(pos) && getImportant();
|
|
},
|
|
'include': function include() {
|
|
return checkInclude(pos) && getInclude();
|
|
},
|
|
'interpolation': function interpolation() {
|
|
return checkInterpolation(pos) && getInterpolation();
|
|
},
|
|
'loop': function loop() {
|
|
return checkLoop(pos) && getLoop();
|
|
},
|
|
'mixin': function mixin() {
|
|
return checkMixin(pos) && getMixin();
|
|
},
|
|
'namespace': function namespace() {
|
|
return checkNamespace(pos) && getNamespace();
|
|
},
|
|
'number': function number() {
|
|
return checkNumber(pos) && getNumber();
|
|
},
|
|
'operator': function operator() {
|
|
return checkOperator(pos) && getOperator();
|
|
},
|
|
'optional': function optional() {
|
|
return checkOptional(pos) && getOptional();
|
|
},
|
|
'parentheses': function parentheses() {
|
|
return checkParentheses(pos) && getParentheses();
|
|
},
|
|
'parentselector': function parentselector() {
|
|
return checkParentSelector(pos) && getParentSelector();
|
|
},
|
|
'percentage': function percentage() {
|
|
return checkPercentage(pos) && getPercentage();
|
|
},
|
|
'placeholder': function placeholder() {
|
|
return checkPlaceholder(pos) && getPlaceholder();
|
|
},
|
|
'progid': function progid() {
|
|
return checkProgid(pos) && getProgid();
|
|
},
|
|
'property': function property() {
|
|
return checkProperty(pos) && getProperty();
|
|
},
|
|
'propertyDelim': function propertyDelim() {
|
|
return checkPropertyDelim(pos) && getPropertyDelim();
|
|
},
|
|
'pseudoc': function pseudoc() {
|
|
return checkPseudoc(pos) && getPseudoc();
|
|
},
|
|
'pseudoe': function pseudoe() {
|
|
return checkPseudoe(pos) && getPseudoe();
|
|
},
|
|
'ruleset': function ruleset() {
|
|
return checkRuleset(pos) && getRuleset();
|
|
},
|
|
's': function s() {
|
|
return checkS(pos) && getS();
|
|
},
|
|
'selector': function selector() {
|
|
return checkSelector(pos) && getSelector();
|
|
},
|
|
'shash': function shash() {
|
|
return checkShash(pos) && getShash();
|
|
},
|
|
'string': function string() {
|
|
return checkString(pos) && getString();
|
|
},
|
|
'stylesheet': function stylesheet() {
|
|
return checkStylesheet(pos) && getStylesheet();
|
|
},
|
|
'typeSelector': function typeSelector() {
|
|
return checkTypeSelector(pos) && getTypeSelector();
|
|
},
|
|
'unary': function unary() {
|
|
return checkUnary(pos) && getUnary();
|
|
},
|
|
'unicodeRange': function unicodeRange() {
|
|
return checkUnicodeRange(pos) && getUnicodeRange();
|
|
},
|
|
'universalSelector': function universalSelector() {
|
|
return checkUniversalSelector(pos) && getUniversalSelector();
|
|
},
|
|
'urange': function urange() {
|
|
return checkUrange(pos) && getUrange();
|
|
},
|
|
'uri': function uri() {
|
|
return checkUri(pos) && getUri();
|
|
},
|
|
'value': function value() {
|
|
return checkValue(pos) && getValue();
|
|
},
|
|
'variable': function variable() {
|
|
return checkVariable(pos) && getVariable();
|
|
},
|
|
'variableslist': function variableslist() {
|
|
return checkVariablesList(pos) && getVariablesList();
|
|
},
|
|
'vhash': function vhash() {
|
|
return checkVhash(pos) && getVhash();
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Stop parsing and display error
|
|
* @param {Number=} i Token's index number
|
|
*/
|
|
function throwError(i) {
|
|
var ln = tokens[i].ln;
|
|
|
|
throw { line: ln, syntax: 'scss' };
|
|
}
|
|
|
|
/**
|
|
* @param {Number} start
|
|
* @param {Number} finish
|
|
* @returns {String}
|
|
*/
|
|
function joinValues(start, finish) {
|
|
var s = '';
|
|
|
|
for (var i = start; i < finish + 1; i++) {
|
|
s += tokens[i].value;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* @param {Number} start
|
|
* @param {Number} num
|
|
* @returns {String}
|
|
*/
|
|
function joinValues2(start, num) {
|
|
if (start + num - 1 >= tokensLength) return;
|
|
|
|
var s = '';
|
|
|
|
for (var i = 0; i < num; i++) {
|
|
s += tokens[start + i].value;
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
function getLastPosition(content, line, column, colOffset) {
|
|
return typeof content === 'string' ? getLastPositionForString(content, line, column, colOffset) : getLastPositionForArray(content, line, column, colOffset);
|
|
}
|
|
|
|
function getLastPositionForString(content, line, column, colOffset) {
|
|
var position = [];
|
|
|
|
if (!content) {
|
|
position = [line, column];
|
|
if (colOffset) position[1] += colOffset - 1;
|
|
return position;
|
|
}
|
|
|
|
var lastLinebreak = content.lastIndexOf('\n');
|
|
var endsWithLinebreak = lastLinebreak === content.length - 1;
|
|
var splitContent = content.split('\n');
|
|
var linebreaksCount = splitContent.length - 1;
|
|
var prevLinebreak = linebreaksCount === 0 || linebreaksCount === 1 ? -1 : content.length - splitContent[linebreaksCount - 1].length - 2;
|
|
|
|
// Line:
|
|
var offset = endsWithLinebreak ? linebreaksCount - 1 : linebreaksCount;
|
|
position[0] = line + offset;
|
|
|
|
// Column:
|
|
if (endsWithLinebreak) {
|
|
offset = prevLinebreak !== -1 ? content.length - prevLinebreak : content.length - 1;
|
|
} else {
|
|
offset = linebreaksCount !== 0 ? content.length - lastLinebreak - column - 1 : content.length - 1;
|
|
}
|
|
position[1] = column + offset;
|
|
|
|
if (!colOffset) return position;
|
|
|
|
if (endsWithLinebreak) {
|
|
position[0]++;
|
|
position[1] = colOffset;
|
|
} else {
|
|
position[1] += colOffset;
|
|
}
|
|
|
|
return position;
|
|
}
|
|
|
|
function getLastPositionForArray(content, line, column, colOffset) {
|
|
var position = void 0;
|
|
|
|
if (content.length === 0) {
|
|
position = [line, column];
|
|
} else {
|
|
var c = content[content.length - 1];
|
|
if (c.hasOwnProperty('end')) {
|
|
position = [c.end.line, c.end.column];
|
|
} else {
|
|
position = getLastPosition(c.content, line, column);
|
|
}
|
|
}
|
|
|
|
if (!colOffset) return position;
|
|
|
|
if (tokens[pos - 1] && tokens[pos - 1].type !== 'Newline') {
|
|
position[1] += colOffset;
|
|
} else {
|
|
position[0]++;
|
|
position[1] = 1;
|
|
}
|
|
|
|
return position;
|
|
}
|
|
|
|
function newNode(type, content, line, column, end) {
|
|
if (!end) end = getLastPosition(content, line, column);
|
|
return new Node({
|
|
type: type,
|
|
content: content,
|
|
start: {
|
|
line: line,
|
|
column: column
|
|
},
|
|
end: {
|
|
line: end[0],
|
|
column: end[1]
|
|
},
|
|
syntax: 'scss'
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkAny(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkBrackets(i)) tokens[i].any_child = 1;else if (l = checkParentheses(i)) tokens[i].any_child = 2;else if (l = checkString(i)) tokens[i].any_child = 3;else if (l = checkVariablesList(i)) tokens[i].any_child = 4;else if (l = checkVariable(i)) tokens[i].any_child = 5;else if (l = checkPlaceholder(i)) tokens[i].any_child = 6;else if (l = checkPercentage(i)) tokens[i].any_child = 7;else if (l = checkDimension(i)) tokens[i].any_child = 8;else if (l = checkUnicodeRange(i)) tokens[i].any_child = 9;else if (l = checkNumber(i)) tokens[i].any_child = 10;else if (l = checkUri(i)) tokens[i].any_child = 11;else if (l = checkExpression(i)) tokens[i].any_child = 12;else if (l = checkFunctionsList(i)) tokens[i].any_child = 13;else if (l = checkFunction(i)) tokens[i].any_child = 14;else if (l = checkInterpolation(i)) tokens[i].any_child = 15;else if (l = checkIdent(i)) tokens[i].any_child = 16;else if (l = checkClass(i)) tokens[i].any_child = 17;else if (l = checkUnary(i)) tokens[i].any_child = 18;else if (l = checkParentSelector(i)) tokens[i].any_child = 19;else if (l = checkImportant(i)) tokens[i].any_child = 20;else if (l = checkGlobal(i)) tokens[i].any_child = 21;else if (l = checkDefault(i)) tokens[i].any_child = 22;else if (l = checkOptional(i)) tokens[i].any_child = 23;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {!Node}
|
|
*/
|
|
function getAny() {
|
|
var childType = tokens[pos].any_child;
|
|
|
|
if (childType === 1) return getBrackets();
|
|
if (childType === 2) return getParentheses();
|
|
if (childType === 3) return getString();
|
|
if (childType === 4) return getVariablesList();
|
|
if (childType === 5) return getVariable();
|
|
if (childType === 6) return getPlaceholder();
|
|
if (childType === 7) return getPercentage();
|
|
if (childType === 8) return getDimension();
|
|
if (childType === 9) return getUnicodeRange();
|
|
if (childType === 10) return getNumber();
|
|
if (childType === 11) return getUri();
|
|
if (childType === 12) return getExpression();
|
|
if (childType === 13) return getFunctionsList();
|
|
if (childType === 14) return getFunction();
|
|
if (childType === 15) return getInterpolation();
|
|
if (childType === 16) return getIdent();
|
|
if (childType === 17) return getClass();
|
|
if (childType === 18) return getUnary();
|
|
if (childType === 19) return getParentSelector();
|
|
if (childType === 20) return getImportant();
|
|
if (childType === 21) return getGlobal();
|
|
if (childType === 22) return getDefault();
|
|
if (childType === 23) return getOptional();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of mixin's arguments.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of arguments
|
|
*/
|
|
function checkArguments(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
while (i < tokens[start].right) {
|
|
if (l = checkArgument(i)) i += l;else return 0;
|
|
}
|
|
|
|
return tokens[start].right - start + 1;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getArguments() {
|
|
var type = NodeType.ArgumentsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var body = void 0;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
while (pos < tokensLength && tokens[pos].type !== TokenType.RightParenthesis) {
|
|
if (checkSingleValueDeclaration(pos)) {
|
|
content.push(getSingleValueDeclaration());
|
|
} else if (checkArgument(pos)) {
|
|
body = getArgument();
|
|
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
|
|
} else if (checkClass(pos)) content.push(getClass());else throwError(pos);
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is valid to be part of arguments list
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of argument
|
|
*/
|
|
function checkArgument(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkBrackets(i)) tokens[i].argument_child = 1;else if (l = checkParentheses(i)) tokens[i].argument_child = 2;else if (l = checkSingleValueDeclaration(i)) tokens[i].argument_child = 3;else if (l = checkFunctionsList(i)) tokens[i].argument_child = 4;else if (l = checkFunction(i)) tokens[i].argument_child = 5;else if (l = checkVariablesList(i)) tokens[i].argument_child = 6;else if (l = checkVariable(i)) tokens[i].argument_child = 7;else if (l = checkSC(i)) tokens[i].argument_child = 8;else if (l = checkDelim(i)) tokens[i].argument_child = 9;else if (l = checkDeclDelim(i)) tokens[i].argument_child = 10;else if (l = checkString(i)) tokens[i].argument_child = 11;else if (l = checkPercentage(i)) tokens[i].argument_child = 12;else if (l = checkDimension(i)) tokens[i].argument_child = 13;else if (l = checkNumber(i)) tokens[i].argument_child = 14;else if (l = checkUri(i)) tokens[i].argument_child = 15;else if (l = checkInterpolation(i)) tokens[i].argument_child = 16;else if (l = checkIdent(i)) tokens[i].argument_child = 17;else if (l = checkVhash(i)) tokens[i].argument_child = 18;else if (l = checkCustomProperty(i)) tokens[i].argument_child = 19;else if (l = checkOperator(i)) tokens[i].argument_child = 20;else if (l = checkUnary(i)) tokens[i].argument_child = 21;else if (l = checkParentSelector(i)) tokens[i].argument_child = 22;else if (l = checkImportant(i)) tokens[i].argument_child = 23;else if (l = checkGlobal(i)) tokens[i].argument_child = 24;else if (l = checkDefault(i)) tokens[i].argument_child = 25;else if (l = checkOptional(i)) tokens[i].argument_child = 26;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} Node that is part of arguments list
|
|
*/
|
|
function getArgument() {
|
|
var childType = tokens[pos].argument_child;
|
|
|
|
if (childType === 1) return getBrackets();
|
|
if (childType === 2) return getParentheses();
|
|
if (childType === 3) return getSingleValueDeclaration();
|
|
if (childType === 4) return getFunctionsList();
|
|
if (childType === 5) return getFunction();
|
|
if (childType === 6) return getVariablesList();
|
|
if (childType === 7) return getVariable();
|
|
if (childType === 8) return getSC();
|
|
if (childType === 9) return getDelim();
|
|
if (childType === 10) return getDeclDelim();
|
|
if (childType === 11) return getString();
|
|
if (childType === 12) return getPercentage();
|
|
if (childType === 13) return getDimension();
|
|
if (childType === 14) return getNumber();
|
|
if (childType === 15) return getUri();
|
|
if (childType === 16) return getInterpolation();
|
|
if (childType === 17) return getIdent();
|
|
if (childType === 18) return getVhash();
|
|
if (childType === 19) return getCustomProperty();
|
|
if (childType === 20) return getOperator();
|
|
if (childType === 21) return getUnary();
|
|
if (childType === 22) return getParentSelector();
|
|
if (childType === 23) return getImportant();
|
|
if (childType === 24) return getGlobal();
|
|
if (childType === 25) return getDefault();
|
|
if (childType === 26) return getOptional();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an @-word (e.g. `@import`, `@include`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkAtkeyword(i) {
|
|
var l = void 0;
|
|
|
|
// Check that token is `@`:
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.CommercialAt) return 0;
|
|
|
|
return (l = checkIdentOrInterpolation(i)) ? l + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with @-word
|
|
* @return {Node}
|
|
*/
|
|
function getAtkeyword() {
|
|
var type = NodeType.AtkeywordType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `@`.
|
|
pos++;
|
|
|
|
var content = getIdentOrInterpolation();
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a part of an @-rule
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of @-rule
|
|
*/
|
|
function checkAtrule(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// If token already has a record of being part of an @-rule,
|
|
// return the @-rule's length:
|
|
if (tokens[i].atrule_l !== undefined) return tokens[i].atrule_l;
|
|
|
|
// If token is part of an @-rule, save the rule's type to token.
|
|
// @keyframes:
|
|
if (l = checkKeyframesRule(i)) tokens[i].atrule_type = 4;
|
|
// @-rule with ruleset:
|
|
else if (l = checkAtruler(i)) tokens[i].atrule_type = 1;
|
|
// Block @-rule:
|
|
else if (l = checkAtruleb(i)) tokens[i].atrule_type = 2;
|
|
// Single-line @-rule:
|
|
else if (l = checkAtrules(i)) tokens[i].atrule_type = 3;else return 0;
|
|
|
|
// If token is part of an @-rule, save the rule's length to token:
|
|
tokens[i].atrule_l = l;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* Get node with @-rule
|
|
* @returns {Array}
|
|
*/
|
|
function getAtrule() {
|
|
var childType = tokens[pos].atrule_type;
|
|
|
|
if (childType === 1) return getAtruler(); // @-rule with ruleset
|
|
if (childType === 2) return getAtruleb(); // Block @-rule
|
|
if (childType === 3) return getAtrules(); // Single-line @-rule
|
|
if (childType === 4) return getKeyframesRule();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a block @-rule
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the @-rule
|
|
*/
|
|
function checkAtruleb(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a block @-rule
|
|
* @returns {Array} `['atruleb', ['atkeyword', x], y, ['block', z]]`
|
|
*/
|
|
function getAtruleb() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an @-rule with ruleset
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the @-rule
|
|
*/
|
|
function checkAtruler(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
|
|
|
|
if (l = checkAtrulers(i)) i += l;
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with an @-rule with ruleset
|
|
* @returns {Array} ['atruler', ['atkeyword', x], y, z]
|
|
*/
|
|
function getAtruler() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets(), getAtrulers());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkAtrulers(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkSC(i)) tokens[i].atrulers_child = 1;else if (l = checkAtrule(i)) tokens[i].atrulers_child = 2;else if (l = checkRuleset(i)) tokens[i].atrulers_child = 3;else break;
|
|
i += l;
|
|
}
|
|
|
|
if (i < tokensLength) tokens[i].atrulers_end = 1;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `['atrulers', x]`
|
|
*/
|
|
function getAtrulers() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `{`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC());
|
|
|
|
while (pos < tokensLength && !tokens[pos].atrulers_end) {
|
|
var childType = tokens[pos].atrulers_child;
|
|
if (childType === 1) content = content.concat(getSC());else if (childType === 2) content.push(getAtrule());else if (childType === 3) content.push(getRuleset());else break;
|
|
}
|
|
|
|
content = content.concat(getSC());
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `}`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkAtrules(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (l = checkTsets(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `['atrules', ['atkeyword', x], y]`
|
|
*/
|
|
function getAtrules() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getTsets());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a block (e.g. `{...}`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the block
|
|
*/
|
|
function checkBlock(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket ? tokens[i].right - i + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a block
|
|
* @returns {Array} `['block', x]`
|
|
*/
|
|
function getBlock() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = tokens[pos].right;
|
|
var content = [];
|
|
|
|
// Skip `{`.
|
|
pos++;
|
|
|
|
while (pos < end) {
|
|
if (checkBlockdecl(pos)) content = content.concat(getBlockdecl());else throwError(pos);
|
|
}
|
|
|
|
var end_ = getLastPosition(content, line, column, 1);
|
|
pos = end + 1;
|
|
|
|
return newNode(type, content, line, column, end_);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a declaration (property-value pair)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the declaration
|
|
*/
|
|
function checkBlockdecl(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkBlockdecl1(i)) tokens[i].bd_type = 1;else if (l = checkBlockdecl2(i)) tokens[i].bd_type = 2;else if (l = checkBlockdecl3(i)) tokens[i].bd_type = 3;else if (l = checkBlockdecl4(i)) tokens[i].bd_type = 4;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getBlockdecl() {
|
|
var childType = tokens[pos].bd_type;
|
|
|
|
if (childType === 1) return getBlockdecl1();
|
|
if (childType === 2) return getBlockdecl2();
|
|
if (childType === 3) return getBlockdecl3();
|
|
if (childType === 4) return getBlockdecl4();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkBlockdecl1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkConditionalStatement(i)) tokens[i].bd_kind = 1;else if (l = checkInclude(i)) tokens[i].bd_kind = 2;else if (l = checkExtend(i)) tokens[i].bd_kind = 4;else if (l = checkLoop(i)) tokens[i].bd_kind = 3;else if (l = checkAtrule(i)) tokens[i].bd_kind = 6;else if (l = checkRuleset(i)) tokens[i].bd_kind = 7;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 5;else return 0;
|
|
|
|
i += l;
|
|
|
|
if (i < tokensLength && (l = checkDeclDelim(i))) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getBlockdecl1() {
|
|
var sc = getSC();
|
|
var content = void 0;
|
|
|
|
switch (tokens[pos].bd_kind) {
|
|
case 1:
|
|
content = getConditionalStatement();
|
|
break;
|
|
case 2:
|
|
content = getInclude();
|
|
break;
|
|
case 3:
|
|
content = getLoop();
|
|
break;
|
|
case 4:
|
|
content = getExtend();
|
|
break;
|
|
case 5:
|
|
content = getDeclaration();
|
|
break;
|
|
case 6:
|
|
content = getAtrule();
|
|
break;
|
|
case 7:
|
|
content = getRuleset();
|
|
break;
|
|
}
|
|
|
|
return sc.concat(content, getSC(), getDeclDelim(), getSC());
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkBlockdecl2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkConditionalStatement(i)) tokens[i].bd_kind = 1;else if (l = checkInclude(i)) tokens[i].bd_kind = 2;else if (l = checkExtend(i)) tokens[i].bd_kind = 4;else if (l = checkMixin(i)) tokens[i].bd_kind = 8;else if (l = checkLoop(i)) tokens[i].bd_kind = 3;else if (l = checkAtrule(i)) tokens[i].bd_kind = 6;else if (l = checkRuleset(i)) tokens[i].bd_kind = 7;else if (l = checkDeclaration(i)) tokens[i].bd_kind = 5;else return 0;
|
|
|
|
i += l;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getBlockdecl2() {
|
|
var sc = getSC();
|
|
var content = void 0;
|
|
|
|
switch (tokens[pos].bd_kind) {
|
|
case 1:
|
|
content = getConditionalStatement();
|
|
break;
|
|
case 2:
|
|
content = getInclude();
|
|
break;
|
|
case 3:
|
|
content = getLoop();
|
|
break;
|
|
case 4:
|
|
content = getExtend();
|
|
break;
|
|
case 5:
|
|
content = getDeclaration();
|
|
break;
|
|
case 6:
|
|
content = getAtrule();
|
|
break;
|
|
case 7:
|
|
content = getRuleset();
|
|
break;
|
|
case 8:
|
|
content = getMixin();
|
|
break;
|
|
}
|
|
|
|
return sc.concat(content, getSC());
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkBlockdecl3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkDeclDelim(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `[s0, ['declDelim'], s1]` where `s0` and `s1` are
|
|
* are optional whitespaces.
|
|
*/
|
|
function getBlockdecl3() {
|
|
return [].concat(getSC(), getDeclDelim(), getSC());
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkBlockdecl4(i) {
|
|
return checkSC(i);
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getBlockdecl4() {
|
|
return getSC();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of text inside square brackets, e.g. `[1]`
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkBrackets(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
|
|
// Skip `[`.
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
if (i < tokens[start].right) {
|
|
var l = checkTsets(i);
|
|
if (l) i += l;else return 0;
|
|
}
|
|
|
|
// Skip `]`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with text inside parentheses or square brackets (e.g. `(1)`)
|
|
* @return {Node}
|
|
*/
|
|
function getBrackets() {
|
|
var type = NodeType.BracketsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var right = token.right;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
if (pos < right) {
|
|
content = getTsets();
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a class selector (e.g. `.abc`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the class selector
|
|
*/
|
|
function checkClass(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].class_l) return tokens[i].class_l;
|
|
|
|
// Skip `.`.
|
|
if (tokens[i].type === TokenType.FullStop) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkIdentOrInterpolation(i)) {
|
|
tokens[start].class_l = l + 1;
|
|
i += l;
|
|
} else break;
|
|
}
|
|
|
|
tokens[start].classEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a class selector
|
|
* @returns {Array} `['class', ['ident', x]]` where x is a class's
|
|
* identifier (without `.`, e.g. `abc`).
|
|
*/
|
|
function getClass() {
|
|
var type = NodeType.ClassType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = token.classEnd;
|
|
var content = [];
|
|
|
|
// Skip `.`
|
|
pos++;
|
|
|
|
while (pos < end) {
|
|
if (checkIdentOrInterpolation(pos)) {
|
|
content = content.concat(getIdentOrInterpolation());
|
|
} else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkCombinator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkCombinator1(i)) tokens[i].combinatorType = 1;else if (l = checkCombinator2(i)) tokens[i].combinatorType = 2;else if (l = checkCombinator3(i)) tokens[i].combinatorType = 3;else if (l = checkCombinator4(i)) tokens[i].combinatorType = 4;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getCombinator() {
|
|
var type = tokens[pos].combinatorType;
|
|
if (type === 1) return getCombinator1();
|
|
if (type === 2) return getCombinator2();
|
|
if (type === 3) return getCombinator3();
|
|
if (type === 4) return getCombinator4();
|
|
}
|
|
|
|
/**
|
|
* (1) `>>>`
|
|
*
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkCombinator1(i) {
|
|
if (i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign && i < tokensLength && tokens[i++].type === TokenType.GreaterThanSign) return 3;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator1() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '>>>';
|
|
|
|
// Skip combinator
|
|
pos += 3;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `||`
|
|
* (2) `>>`
|
|
*
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkCombinator2(i) {
|
|
if (i + 1 >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.VerticalLine && tokens[i + 1].type === TokenType.VerticalLine) return 2;
|
|
|
|
if (tokens[i].type === TokenType.GreaterThanSign && tokens[i + 1].type === TokenType.GreaterThanSign) return 2;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator2() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '' + token.value + tokens[pos + 1].value;
|
|
|
|
// Skip combinator
|
|
pos += 2;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `>`
|
|
* (2) `+`
|
|
* (3) `~`
|
|
*
|
|
* @param {Number} i
|
|
* @return {Number}
|
|
*/
|
|
function checkCombinator3(i) {
|
|
var type = tokens[i].type;
|
|
if (type === TokenType.PlusSign || type === TokenType.GreaterThanSign || type === TokenType.Tilde) return 1;else return 0;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator3() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
// Skip combinator
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `/panda/`
|
|
*/
|
|
function checkCombinator4(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (tokens[i].type === TokenType.Solidus) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getCombinator4() {
|
|
var type = NodeType.CombinatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `/`.
|
|
pos++;
|
|
|
|
var ident = getIdent();
|
|
|
|
// Skip `/`.
|
|
pos++;
|
|
|
|
var content = '/' + ident.content + '/';
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a multiline comment.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is a multiline comment, otherwise `0`
|
|
*/
|
|
function checkCommentML(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.CommentML ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a multiline comment
|
|
* @returns {Array} `['commentML', x]` where `x`
|
|
* is the comment's text (without `/*` and `* /`).
|
|
*/
|
|
function getCommentML() {
|
|
var type = NodeType.CommentMLType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos].value.substring(2);
|
|
var l = content.length;
|
|
|
|
if (content.charAt(l - 2) === '*' && content.charAt(l - 1) === '/') content = content.substring(0, l - 2);
|
|
|
|
var end = getLastPosition(content, line, column, 2);
|
|
if (end[0] === line) end[1] += 2;
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a single-line comment.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is a single-line comment, otherwise `0`
|
|
*/
|
|
function checkCommentSL(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.CommentSL ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a single-line comment.
|
|
* @returns {Array} `['commentSL', x]` where `x` is comment's message
|
|
* (without `//`)
|
|
*/
|
|
function getCommentSL() {
|
|
var type = NodeType.CommentSLType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos++].value.substring(2);
|
|
var end = getLastPosition(content, line, column + 2);
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a condition
|
|
* (e.g. `@if ...`, `@else if ...` or `@else ...`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the condition
|
|
*/
|
|
function checkCondition(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
var _i = void 0;
|
|
var s = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (['if', 'else'].indexOf(tokens[start + 1].value) < 0) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkBlock(i)) break;
|
|
|
|
s = checkSC(i);
|
|
_i = i + s;
|
|
|
|
if (l = _checkCondition(_i)) i += l + s;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function _checkCondition(i) {
|
|
return checkVariable(i) || checkNumber(i) || checkInterpolation(i) || checkIdent(i) || checkOperator(i) || checkCombinator(i) || checkString(i);
|
|
}
|
|
|
|
/**
|
|
* Get node with a condition.
|
|
* @returns {Array} `['condition', x]`
|
|
*/
|
|
function getCondition() {
|
|
var type = NodeType.ConditionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var s = void 0;
|
|
var _pos = void 0;
|
|
|
|
content.push(getAtkeyword());
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkBlock(pos)) break;
|
|
|
|
s = checkSC(pos);
|
|
_pos = pos + s;
|
|
|
|
if (!_checkCondition(_pos)) break;
|
|
|
|
if (s) content = content.concat(getSC());
|
|
content.push(_getCondition());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function _getCondition() {
|
|
if (checkVariable(pos)) return getVariable();
|
|
if (checkNumber(pos)) return getNumber();
|
|
if (checkInterpolation(pos)) return getInterpolation();
|
|
if (checkIdent(pos)) return getIdent();
|
|
if (checkOperator(pos)) return getOperator();
|
|
if (checkCombinator(pos)) return getCombinator();
|
|
if (checkString(pos)) return getString();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a conditional statement
|
|
* (e.g. `@if ... {} @else if ... {} @else ... {}`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the condition
|
|
*/
|
|
function checkConditionalStatement(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkCondition(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a condition.
|
|
* @returns {Array} `['condition', x]`
|
|
*/
|
|
function getConditionalStatement() {
|
|
var type = NodeType.ConditionalStatementType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getCondition(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a declaration (property-value pair)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the declaration
|
|
*/
|
|
function checkDeclaration(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkProperty(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkPropertyDelim(i)) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkValue(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a declaration
|
|
* @returns {Array} `['declaration', ['property', x], ['propertyDelim'],
|
|
* ['value', y]]`
|
|
*/
|
|
function getDeclaration() {
|
|
var type = NodeType.DeclarationType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getProperty(), getSC(), getPropertyDelim(), getSC(), getValue());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the declaration
|
|
*/
|
|
function checkSingleValueDeclaration(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkProperty(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkPropertyDelim(i)) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkSingleValue(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a declaration
|
|
* @returns {Array} `['declaration', ['property', x], ['propertyDelim'],
|
|
* ['value', y]]`
|
|
*/
|
|
function getSingleValueDeclaration() {
|
|
var type = NodeType.DeclarationType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getProperty(), getSC(), getPropertyDelim(), getSC(), getSingleValue());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a semicolon
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is a semicolon, otherwise `0`
|
|
*/
|
|
function checkDeclDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Semicolon ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a semicolon
|
|
* @returns {Array} `['declDelim']`
|
|
*/
|
|
function getDeclDelim() {
|
|
var type = NodeType.DeclDelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ';';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token if part of `!default` word.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the `!default` word
|
|
*/
|
|
function checkDefault(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].value === 'default') {
|
|
tokens[start].defaultEnd = i;
|
|
return i - start + 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get node with a `!default` word
|
|
* @returns {Array} `['default', sc]` where `sc` is optional whitespace
|
|
*/
|
|
function getDefault() {
|
|
var type = NodeType.DefaultType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, token.defaultEnd);
|
|
|
|
pos = token.defaultEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a comma
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is a comma, otherwise `0`
|
|
*/
|
|
function checkDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Comma ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a comma
|
|
* @returns {Array} `['delim']`
|
|
*/
|
|
function getDelim() {
|
|
var type = NodeType.DelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ',';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number with dimension unit (e.g. `10px`)
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkDimension(i) {
|
|
var ln = checkNumber(i);
|
|
var li = void 0;
|
|
|
|
if (i >= tokensLength || !ln || i + ln >= tokensLength) return 0;
|
|
|
|
return (li = checkUnit(i + ln)) ? ln + li : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node of a number with dimension unit
|
|
* @return {Node}
|
|
*/
|
|
function getDimension() {
|
|
var type = NodeType.DimensionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getNumber(), getUnit()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkExpression(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength || tokens[i++].value !== 'expression' || i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) {
|
|
return 0;
|
|
}
|
|
|
|
return tokens[i].right - start + 1;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getExpression() {
|
|
var type = NodeType.ExpressionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
pos++;
|
|
|
|
var content = joinValues(pos + 1, tokens[pos].right - 1);
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
if (end[0] === line) end[1] += 11;
|
|
pos = tokens[pos].right + 1;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkExtend(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
|
|
if (l = checkExtend1(i)) tokens[i].extend_child = 1;else if (l = checkExtend2(i)) tokens[i].extend_child = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getExtend() {
|
|
var childType = tokens[pos].extend_child;
|
|
|
|
if (childType === 1) return getExtend1();
|
|
if (childType === 2) return getExtend2();
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of an extend with `!optional` flag.
|
|
* @param {Number} i
|
|
*/
|
|
function checkExtend1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'extend') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkOptional(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getExtend1() {
|
|
var type = NodeType.ExtendType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getSelectorsGroup(), getSC(), getOptional());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Checks if token is part of an extend without `!optional` flag.
|
|
* @param {Number} i
|
|
*/
|
|
function checkExtend2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'extend') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getExtend2() {
|
|
var type = NodeType.ExtendType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getSelectorsGroup());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkFunction(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i < tokensLength && tokens[i].type === TokenType.LeftParenthesis ? tokens[i].right - start + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getFunction() {
|
|
var type = NodeType.FunctionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getIdentOrInterpolation(), getArguments());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a functions list (e.g. `function(value)...`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkFunctionsList(i) {
|
|
var d = 0; // Number of dots
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkFunction(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength && tokens[i].type === TokenType.FullStop) {
|
|
d++;
|
|
i++;
|
|
}
|
|
|
|
return d === 3 ? l + d : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a functions list
|
|
* @returns {Array}
|
|
*/
|
|
function getFunctionsList() {
|
|
var type = NodeType.FunctionsListType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getFunction()];
|
|
var end = getLastPosition(content, line, column, 3);
|
|
|
|
// Skip `...`.
|
|
pos += 3;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of `!global` word
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkGlobal(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].value === 'global') {
|
|
tokens[start].globalEnd = i;
|
|
return i - start + 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get node with `!global` word
|
|
*/
|
|
function getGlobal() {
|
|
var type = NodeType.GlobalType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, token.globalEnd);
|
|
|
|
pos = token.globalEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an identifier
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the identifier
|
|
*/
|
|
function checkIdent(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Check if token is part of a negative number
|
|
if (tokens[i].type === TokenType.HyphenMinus && tokens[i + 1].type === TokenType.DecimalNumber) return 0;
|
|
|
|
if (tokens[i].type === TokenType.HyphenMinus) i++;
|
|
|
|
if (checkInterpolation(i)) {
|
|
tokens[start].ident_last = i - 1;
|
|
return i - start;
|
|
}
|
|
|
|
if (tokens[i].type === TokenType.LowLine || tokens[i].type === TokenType.Identifier) i++;else return 0;
|
|
|
|
for (; i < tokensLength; i++) {
|
|
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
|
|
}
|
|
|
|
tokens[start].ident_last = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with an identifier
|
|
* @returns {Array} `['ident', x]` where `x` is identifier's name
|
|
*/
|
|
function getIdent() {
|
|
var type = NodeType.IdentType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, tokens[pos].ident_last);
|
|
|
|
pos = tokens[pos].ident_last + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the identifier
|
|
*/
|
|
function checkPartialIdent(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
for (; i < tokensLength; i++) {
|
|
if (tokens[i].type !== TokenType.HyphenMinus && tokens[i].type !== TokenType.LowLine && tokens[i].type !== TokenType.Identifier && tokens[i].type !== TokenType.DecimalNumber) break;
|
|
}
|
|
|
|
tokens[start].ident_last = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function checkIdentOrInterpolation(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
var prevIsInterpolation = false;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkInterpolation(i)) {
|
|
tokens[i].ii_type = 1;
|
|
i += l;
|
|
prevIsInterpolation = true;
|
|
} else if (l = checkIdent(i)) {
|
|
tokens[i].ii_type = 2;
|
|
i += l;
|
|
prevIsInterpolation = false;
|
|
} else if (prevIsInterpolation && (l = checkPartialIdent(i))) {
|
|
tokens[i].ii_type = 3;
|
|
i += l;
|
|
prevIsInterpolation = false;
|
|
} else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getIdentOrInterpolation() {
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
var tokenType = tokens[pos].ii_type;
|
|
|
|
if (tokenType === 1) {
|
|
content.push(getInterpolation());
|
|
} else if (tokenType === 2 || tokenType === 3) {
|
|
content.push(getIdent());
|
|
} else break;
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of `!important` word
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkImportant(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].value === 'important') {
|
|
tokens[start].importantEnd = i;
|
|
return i - start + 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get node with `!important` word
|
|
* @returns {Array} `['important', sc]` where `sc` is optional whitespace
|
|
*/
|
|
function getImportant() {
|
|
var type = NodeType.ImportantType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, token.importantEnd);
|
|
|
|
pos = token.importantEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin (`@include` or `@extend`
|
|
* directive).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the included mixin
|
|
*/
|
|
function checkInclude(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkInclude1(i)) tokens[i].include_type = 1;else if (l = checkInclude2(i)) tokens[i].include_type = 2;else if (l = checkInclude3(i)) tokens[i].include_type = 3;else if (l = checkInclude4(i)) tokens[i].include_type = 4;else if (l = checkInclude5(i)) tokens[i].include_type = 5;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin
|
|
* @returns {Array} `['include', x]`
|
|
*/
|
|
function getInclude() {
|
|
var type = tokens[pos].include_type;
|
|
|
|
if (type === 1) return getInclude1();
|
|
if (type === 2) return getInclude2();
|
|
if (type === 3) return getInclude3();
|
|
if (type === 4) return getInclude4();
|
|
if (type === 5) return getInclude5();
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin with keyfames selector like
|
|
* `@include nani(foo) { 0% {}}`
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the include
|
|
*/
|
|
function checkInclude1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'include') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin with keyfames selector like
|
|
* `@include nani(foo) { 0% {}}`
|
|
* @returns {Array} `['include', ['atkeyword', x], sc, ['selector', y], sc,
|
|
* ['arguments', z], sc, ['block', q], sc` where `x` is `include` or
|
|
* `extend`, `y` is mixin's identifier (selector), `z` are arguments
|
|
* passed to the mixin, `q` is block passed to the mixin containing a
|
|
* ruleset > selector > keyframesSelector, and `sc` are optional
|
|
* whitespaces
|
|
*/
|
|
function getInclude1() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getKeyframesBlocks());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin like `@include nani(foo) {...}`
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the include
|
|
*/
|
|
function checkInclude2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'include') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin like `@include nani(foo) {...}`
|
|
* @returns {Array} `['include', ['atkeyword', x], sc, ['selector', y], sc,
|
|
* ['arguments', z], sc, ['block', q], sc` where `x` is `include` or
|
|
* `extend`, `y` is mixin's identifier (selector), `z` are arguments
|
|
* passed to the mixin, `q` is block passed to the mixin and `sc`
|
|
* are optional whitespaces
|
|
*/
|
|
function getInclude2() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin like `@include nani(foo)`
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the include
|
|
*/
|
|
function checkInclude3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'include') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with included mixin like `@include nani(foo)`
|
|
* @returns {Array} `['include', ['atkeyword', x], sc, ['selector', y], sc,
|
|
* ['arguments', z], sc]` where `x` is `include` or `extend`, `y` is
|
|
* mixin's identifier (selector), `z` are arguments passed to the
|
|
* mixin and `sc` are optional whitespaces
|
|
*/
|
|
function getInclude3() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getArguments());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an included mixin with a content block passed
|
|
* as an argument (e.g. `@include nani {...}`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the mixin
|
|
*/
|
|
function checkInclude4(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'include') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with an included mixin with a content block passed
|
|
* as an argument (e.g. `@include nani {...}`)
|
|
* @returns {Array} `['include', x]`
|
|
*/
|
|
function getInclude4() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkInclude5(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (tokens[start + 1].value !== 'include') return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `['include', x]`
|
|
*/
|
|
function getInclude5() {
|
|
var type = NodeType.IncludeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC(), getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of an interpolated variable (e.g. `#{$nani}`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkInterpolation(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type !== TokenType.NumberSign || !tokens[i + 1] || tokens[i + 1].type !== TokenType.LeftCurlyBracket) return 0;
|
|
|
|
i += 2;
|
|
|
|
while (tokens[i].type !== TokenType.RightCurlyBracket) {
|
|
if (l = checkArgument(i)) i += l;else return 0;
|
|
}
|
|
|
|
return tokens[i].type === TokenType.RightCurlyBracket ? i - start + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an interpolated variable
|
|
* @returns {Array} `['interpolation', x]`
|
|
*/
|
|
function getInterpolation() {
|
|
var type = NodeType.InterpolationType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `#{`:
|
|
pos += 2;
|
|
|
|
while (pos < tokensLength && tokens[pos].type !== TokenType.RightCurlyBracket) {
|
|
var body = getArgument();
|
|
if (typeof body.content === 'string') content.push(body);else content = content.concat(body);
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `}`:
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check a single keyframe block - `5% {}`
|
|
* @param {Number} i
|
|
* @returns {Number}
|
|
*/
|
|
function checkKeyframesBlock(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkKeyframesSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a single keyframe block - `5% {}`
|
|
* @returns {Node}
|
|
*/
|
|
function getKeyframesBlock() {
|
|
var type = NodeType.RulesetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getKeyframesSelectorsGroup(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check all keyframe blocks - `5% {} 100% {}`
|
|
* @param {Number} i
|
|
* @returns {Number}
|
|
*/
|
|
function checkKeyframesBlocks(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.LeftCurlyBracket) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlock(i)) i += l;
|
|
|
|
while (tokens[i].type !== TokenType.RightCurlyBracket) {
|
|
if (l = checkSC(i)) i += l;else if (l = checkKeyframesBlock(i)) i += l;else if (l = checkAtrule(i)) {
|
|
i += l;
|
|
if (l = checkSC(i)) i += l;
|
|
if (l = checkDeclDelim(i)) i += l;
|
|
} else break;
|
|
}
|
|
|
|
if (i < tokensLength && tokens[i].type === TokenType.RightCurlyBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get all keyframe blocks - `5% {} 100% {}`
|
|
* @returns {Node}
|
|
*/
|
|
function getKeyframesBlocks() {
|
|
var type = NodeType.BlockType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var keyframesBlocksEnd = token.right;
|
|
var content = [];
|
|
|
|
// Skip `{`.
|
|
pos++;
|
|
|
|
while (pos < keyframesBlocksEnd) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkKeyframesBlock(pos)) content.push(getKeyframesBlock());else if (checkAtrule(pos)) {
|
|
content.push(getAtrule()); // @content
|
|
if (checkSC(pos)) content = content.concat(getSC());
|
|
if (checkDeclDelim(pos)) content.push(getDeclDelim());
|
|
} else break;
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `}`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a @keyframes rule.
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the @keyframes rule
|
|
*/
|
|
function checkKeyframesRule(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
var atruleName = joinValues2(i - l, l);
|
|
if (atruleName.toLowerCase().indexOf('keyframes') === -1) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i) || checkPseudoc(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkKeyframesBlocks(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Node}
|
|
*/
|
|
function getKeyframesRule() {
|
|
var type = NodeType.AtruleType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC());
|
|
|
|
if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());else if (checkPseudoc(pos)) {
|
|
content = content.concat(getPseudoc());
|
|
}
|
|
|
|
content = content.concat(getSC(), getKeyframesBlocks());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check a single keyframe selector - `5%`, `from` etc
|
|
* @param {Number} i
|
|
* @returns {Number}
|
|
*/
|
|
function checkKeyframesSelector(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) {
|
|
// Valid selectors are only `from` and `to`.
|
|
var selector = joinValues2(i, l);
|
|
if (selector !== 'from' && selector !== 'to') return 0;
|
|
|
|
i += l;
|
|
tokens[start].keyframesSelectorType = 1;
|
|
} else if (l = checkPercentage(i)) {
|
|
i += l;
|
|
tokens[start].keyframesSelectorType = 2;
|
|
} else if (l = checkInterpolation(i)) {
|
|
i += l;
|
|
tokens[start].keyframesSelectorType = 3;
|
|
} else {
|
|
return 0;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a single keyframe selector
|
|
* @returns {Node}
|
|
*/
|
|
function getKeyframesSelector() {
|
|
var keyframesSelectorType = NodeType.KeyframesSelectorType;
|
|
var selectorType = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (token.keyframesSelectorType === 1) {
|
|
content.push(getIdent());
|
|
} else if (token.keyframesSelectorType === 2) {
|
|
content.push(getPercentage());
|
|
} else if (token.keyframesSelectorType === 3) {
|
|
content.push(getInterpolation());
|
|
}
|
|
|
|
var keyframesSelector = newNode(keyframesSelectorType, content, line, column);
|
|
|
|
return newNode(selectorType, [keyframesSelector], line, column);
|
|
}
|
|
|
|
/**
|
|
* Check the keyframe's selector groups
|
|
* @param {Number} i
|
|
* @returns {Number}
|
|
*/
|
|
function checkKeyframesSelectorsGroup(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkKeyframesSelector(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
var comma = checkDelim(i + spaceBefore);
|
|
if (!comma) break;
|
|
|
|
var spaceAfter = checkSC(i + spaceBefore + comma);
|
|
if (l = checkKeyframesSelector(i + spaceBefore + comma + spaceAfter)) {
|
|
i += spaceBefore + comma + spaceAfter + l;
|
|
} else break;
|
|
}
|
|
|
|
tokens[start].selectorsGroupEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get the keyframe's selector groups
|
|
* @returns {Array} An array of keyframe selectors
|
|
*/
|
|
function getKeyframesSelectorsGroup() {
|
|
var selectorsGroup = [];
|
|
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
|
|
|
|
selectorsGroup.push(getKeyframesSelector());
|
|
|
|
while (pos < selectorsGroupEnd) {
|
|
selectorsGroup = selectorsGroup.concat(getSC(), getDelim(), getSC(), getKeyframesSelector());
|
|
}
|
|
|
|
return selectorsGroup;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a loop.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the loop
|
|
*/
|
|
function checkLoop(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkAtkeyword(i)) i += l;else return 0;
|
|
|
|
if (['for', 'each', 'while'].indexOf(tokens[start + 1].value) < 0) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkBlock(i)) {
|
|
i += l;
|
|
break;
|
|
} else if (l = checkVariable(i) || checkNumber(i) || checkInterpolation(i) || checkIdent(i) || checkSC(i) || checkOperator(i) || checkCombinator(i) || checkString(i)) i += l;else return 0;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a loop.
|
|
* @returns {Array} `['loop', x]`
|
|
*/
|
|
function getLoop() {
|
|
var type = NodeType.LoopType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(getAtkeyword());
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkBlock(pos)) {
|
|
content.push(getBlock());
|
|
break;
|
|
} else if (checkVariable(pos)) content.push(getVariable());else if (checkNumber(pos)) content.push(getNumber());else if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkIdent(pos)) content.push(getIdent());else if (checkOperator(pos)) content.push(getOperator());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkSC(pos)) content = content.concat(getSC());else if (checkString(pos)) content.push(getString());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a mixin
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the mixin
|
|
*/
|
|
function checkMixin(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if ((l = checkAtkeyword(i)) && tokens[i + 1].value === 'mixin') i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkArguments(i)) i += l;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a mixin
|
|
* @returns {Array} `['mixin', x]`
|
|
*/
|
|
function getMixin() {
|
|
var type = NodeType.MixinType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getAtkeyword(), getSC());
|
|
|
|
if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());
|
|
|
|
content = content.concat(getSC());
|
|
|
|
if (checkArguments(pos)) content.push(getArguments());
|
|
|
|
content = content.concat(getSC());
|
|
|
|
if (checkBlock(pos)) content.push(getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a namespace sign (`|`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is `|`, `0` if not
|
|
*/
|
|
function checkNamespace(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.VerticalLine ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a namespace sign
|
|
* @returns {Array} `['namespace']`
|
|
*/
|
|
function getNamespace() {
|
|
var type = NodeType.NamespaceType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '|';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkNmName2(i) {
|
|
if (tokens[i].type === TokenType.Identifier) return 1;else if (tokens[i].type !== TokenType.DecimalNumber) return 0;
|
|
|
|
i++;
|
|
|
|
return i < tokensLength && tokens[i].type === TokenType.Identifier ? 2 : 1;
|
|
}
|
|
|
|
/**
|
|
* @returns {String}
|
|
*/
|
|
function getNmName2() {
|
|
var s = tokens[pos].value;
|
|
|
|
if (tokens[pos++].type === TokenType.DecimalNumber && pos < tokensLength && tokens[pos].type === TokenType.Identifier) s += tokens[pos++].value;
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of number
|
|
*/
|
|
function checkNumber(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].number_l) return tokens[i].number_l;
|
|
|
|
// `10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && (!tokens[i + 1] || tokens[i + 1] && tokens[i + 1].type !== TokenType.FullStop)) {
|
|
tokens[i].number_l = 1;
|
|
return 1;
|
|
}
|
|
|
|
// `10.`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && (!tokens[i + 2] || tokens[i + 2].type !== TokenType.DecimalNumber)) {
|
|
tokens[i].number_l = 2;
|
|
return 2;
|
|
}
|
|
|
|
// `.10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.FullStop && tokens[i + 1].type === TokenType.DecimalNumber) {
|
|
tokens[i].number_l = 2;
|
|
return 2;
|
|
}
|
|
|
|
// `10.10`:
|
|
if (i < tokensLength && tokens[i].type === TokenType.DecimalNumber && tokens[i + 1] && tokens[i + 1].type === TokenType.FullStop && tokens[i + 2] && tokens[i + 2].type === TokenType.DecimalNumber) {
|
|
tokens[i].number_l = 3;
|
|
return 3;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with number
|
|
* @returns {Array} `['number', x]` where `x` is a number converted
|
|
* to string.
|
|
*/
|
|
function getNumber() {
|
|
var type = NodeType.NumberType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var l = tokens[pos].number_l;
|
|
var content = '';
|
|
|
|
for (var j = 0; j < l; j++) {
|
|
content += tokens[pos + j].value;
|
|
}
|
|
|
|
pos += l;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is an operator (`/`, `%`, `,`, `:` or `=`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is an operator, otherwise `0`
|
|
*/
|
|
function checkOperator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
switch (tokens[i].type) {
|
|
case TokenType.Solidus:
|
|
case TokenType.PercentSign:
|
|
case TokenType.Comma:
|
|
case TokenType.Colon:
|
|
case TokenType.EqualsSign:
|
|
case TokenType.EqualitySign:
|
|
case TokenType.InequalitySign:
|
|
case TokenType.LessThanSign:
|
|
case TokenType.GreaterThanSign:
|
|
case TokenType.Asterisk:
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an operator
|
|
* @returns {Array} `['operator', x]` where `x` is an operator converted
|
|
* to string.
|
|
*/
|
|
function getOperator() {
|
|
var type = NodeType.OperatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of `!optional` word
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkOptional(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i++].type !== TokenType.ExclamationMark) return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].value === 'optional') {
|
|
tokens[start].optionalEnd = i;
|
|
return i - start + 1;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get node with `!optional` word
|
|
*/
|
|
function getOptional() {
|
|
var type = NodeType.OptionalType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, token.optionalEnd);
|
|
|
|
pos = token.optionalEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of text inside parentheses, e.g. `(1)`
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkParentheses(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var right = tokens[i].right;
|
|
var l = void 0;
|
|
|
|
// Skip `(`.
|
|
if (tokens[i].type === TokenType.LeftParenthesis) i++;else return 0;
|
|
|
|
if (i < right) {
|
|
if (l = checkTsets(i)) i += l;else return 0;
|
|
}
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with text inside parentheses, e.g. `(1)`
|
|
* @return {Node}
|
|
*/
|
|
function getParentheses() {
|
|
var type = NodeType.ParenthesesType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var right = token.right;
|
|
var content = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
if (pos < right) {
|
|
content = getTsets();
|
|
}
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a parent selector, e.g. `&`
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkParentSelector(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Ampersand ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a parent selector
|
|
* @return {Node}
|
|
*/
|
|
function getParentSelector() {
|
|
var type = NodeType.ParentSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '&';
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a parent selector extension, e.g. `&--foo-bar`
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the parent selector extension
|
|
*/
|
|
function checkParentSelectorExtension(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get parent selector extension node
|
|
* @return {Node}
|
|
*/
|
|
function getParentSelectorExtension() {
|
|
var type = NodeType.ParentSelectorExtensionType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkIdentOrInterpolation(pos)) {
|
|
content = content.concat(getIdentOrInterpolation());
|
|
} else if (checkPartialIdent(pos)) {
|
|
content.push(getIdent());
|
|
} else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a parent selector with an extension or not
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of the parent selector and extension if applicable
|
|
*/
|
|
function checkParentSelectorWithExtension(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkParentSelector(i)) i += l;else return 0;
|
|
|
|
if (l = checkParentSelectorExtension(i)) i += l;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get parent selector node and extension node if applicable
|
|
* @return {Array}
|
|
*/
|
|
function getParentSelectorWithExtension() {
|
|
var content = [getParentSelector()];
|
|
|
|
if (checkParentSelectorExtension(pos)) content.push(getParentSelectorExtension());
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a number or an interpolation with a percent sign
|
|
* (e.g. `10%`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkPercentage(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkNumberOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Skip `%`.
|
|
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a percentage node that contains either a number or an interpolation
|
|
* @returns {Object} The percentage node
|
|
*/
|
|
function getPercentage() {
|
|
var type = NodeType.PercentageType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = getNumberOrInterpolation();
|
|
var end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `%`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a number or an interpolation
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkNumberOrInterpolation(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkInterpolation(i) || checkNumber(i)) i += l;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a number and/or interpolation node
|
|
* @returns {Array} An array containing a single or multiple nodes
|
|
*/
|
|
function getNumberOrInterpolation() {
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkNumber(pos)) content.push(getNumber());else break;
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a placeholder selector (e.g. `%abc`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the selector
|
|
*/
|
|
function checkPlaceholder(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[start].placeholder_l) return tokens[start].placeholder_l;
|
|
|
|
// Skip `%`.
|
|
if (tokens[i].type === TokenType.PercentSign) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) {
|
|
i += l;
|
|
tokens[start].placeholder_l = i - start;
|
|
} else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a placeholder selector
|
|
* @returns {Array} `['placeholder', ['ident', x]]` where x is a placeholder's
|
|
* identifier (without `%`, e.g. `abc`).
|
|
*/
|
|
function getPlaceholder() {
|
|
var type = NodeType.PlaceholderType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `%`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkProgid(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (joinValues2(i, 6) === 'progid:DXImageTransform.Microsoft.') i += 6;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.LeftParenthesis) {
|
|
tokens[start].progid_end = tokens[i].right;
|
|
i = tokens[i].right + 1;
|
|
} else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getProgid() {
|
|
var type = NodeType.ProgidType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var progid_end = token.progid_end;
|
|
var content = joinValues(pos, progid_end);
|
|
|
|
pos = progid_end + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a property
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the property
|
|
*/
|
|
function checkProperty(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkProperty1(i)) tokens[start].propertyType = 1;else if (l = checkProperty2(i)) tokens[start].propertyType = 2;else if (l = checkProperty3(i)) tokens[start].propertyType = 3;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* Get node with a property
|
|
* @return {Node}
|
|
*/
|
|
function getProperty() {
|
|
var type = tokens[pos].propertyType;
|
|
|
|
if (type === 1) return getProperty1();
|
|
if (type === 2) return getProperty2();
|
|
if (type === 3) return getProperty3();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a property
|
|
* (1) `foo`
|
|
* (2) `#{$foo}`
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the property
|
|
*/
|
|
function checkProperty1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a property
|
|
* @returns {Array}
|
|
*/
|
|
function getProperty1() {
|
|
var type = NodeType.PropertyType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = getIdentOrInterpolation();
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a custom property
|
|
* (1) `--foo-bar`
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the property
|
|
*/
|
|
function checkProperty2(i) {
|
|
return checkCustomProperty(i);
|
|
}
|
|
|
|
/**
|
|
* Get node with a custom property
|
|
* @return {Node}
|
|
*/
|
|
function getProperty2() {
|
|
return getCustomProperty();
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a property
|
|
* (1) `$foo`
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the property
|
|
*/
|
|
function checkProperty3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkVariable(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a property
|
|
* @returns {Array} `['property', x]`
|
|
*/
|
|
function getProperty3() {
|
|
var type = NodeType.PropertyType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getVariable()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a custom property
|
|
* @param {Number} i Token's index number
|
|
* @return {Number} Length of the property
|
|
*/
|
|
function checkCustomProperty(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type !== TokenType.HyphenMinus || tokens[i + 1] && tokens[i + 1].type !== TokenType.HyphenMinus) return 0;
|
|
|
|
// Skip `--`
|
|
i += 2;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a custom property
|
|
* @return {Node}
|
|
*/
|
|
function getCustomProperty() {
|
|
var type = NodeType.CustomPropertyType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `--`
|
|
pos += 2;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a colon
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is a colon, otherwise `0`
|
|
*/
|
|
function checkPropertyDelim(i) {
|
|
return i < tokensLength && tokens[i].type === TokenType.Colon ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a colon
|
|
* @returns {Array} `['propertyDelim']`
|
|
*/
|
|
function getPropertyDelim() {
|
|
var type = NodeType.PropertyDelimType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = ':';
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkPseudo(i) {
|
|
return checkPseudoe(i) || checkPseudoc(i);
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getPseudo() {
|
|
if (checkPseudoe(pos)) return getPseudoe();
|
|
if (checkPseudoc(pos)) return getPseudoc();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkPseudoe(i) {
|
|
var l = void 0;
|
|
|
|
// Check `::`
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.Colon || i >= tokensLength || tokens[i + 1].type !== TokenType.Colon) return 0;
|
|
|
|
if (l = checkPseudoElement1(i)) tokens[i].pseudoElementType = 1;else if (l = checkPseudoElement2(i)) tokens[i].pseudoElementType = 2;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Node}
|
|
*/
|
|
function getPseudoe() {
|
|
var childType = tokens[pos].pseudoElementType;
|
|
if (childType === 1) return getPseudoElement1();
|
|
if (childType === 2) return getPseudoElement2();
|
|
}
|
|
|
|
/**
|
|
* (1) `::slotted(selector)`
|
|
* (2) `::slotted(selector, selector)`
|
|
*/
|
|
function checkPseudoElement1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `::`.
|
|
i += 2;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* (1) `::slotted(selector)`
|
|
* (2) `::slotted(selector, selector)`
|
|
*/
|
|
function getPseudoElement1() {
|
|
var type = NodeType.PseudoeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `::`.
|
|
pos += 2;
|
|
|
|
content.push(getIdent());
|
|
|
|
{
|
|
var _type = NodeType.ArgumentsType;
|
|
var _token = tokens[pos];
|
|
var _line = _token.ln;
|
|
var _column = _token.col;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
|
|
|
|
var end = getLastPosition(selectorContent, _line, _column, 1);
|
|
var args = newNode(_type, selectorContent, _line, _column, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkPseudoElement2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `::`.
|
|
i += 2;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Node}
|
|
*/
|
|
function getPseudoElement2() {
|
|
var type = NodeType.PseudoeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `::`.
|
|
pos += 2;
|
|
|
|
var content = getIdentOrInterpolation();
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkPseudoc(i) {
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.Colon) return 0;
|
|
|
|
if (l = checkPseudoClass3(i)) tokens[i].pseudoClassType = 3;else if (l = checkPseudoClass4(i)) tokens[i].pseudoClassType = 4;else if (l = checkPseudoClass5(i)) tokens[i].pseudoClassType = 5;else if (l = checkPseudoClass1(i)) tokens[i].pseudoClassType = 1;else if (l = checkPseudoClass2(i)) tokens[i].pseudoClassType = 2;else if (l = checkPseudoClass6(i)) tokens[i].pseudoClassType = 6;else return 0;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getPseudoc() {
|
|
var childType = tokens[pos].pseudoClassType;
|
|
if (childType === 1) return getPseudoClass1();
|
|
if (childType === 2) return getPseudoClass2();
|
|
if (childType === 3) return getPseudoClass3();
|
|
if (childType === 4) return getPseudoClass4();
|
|
if (childType === 5) return getPseudoClass5();
|
|
if (childType === 6) return getPseudoClass6();
|
|
}
|
|
|
|
/**
|
|
* (-) `:not(panda)`
|
|
*/
|
|
function checkPseudoClass1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* (-) `:not(panda)`
|
|
*/
|
|
function getPseudoClass1() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
{
|
|
var _type2 = NodeType.ArgumentsType;
|
|
var _token2 = tokens[pos];
|
|
var _line2 = _token2.ln;
|
|
var _column2 = _token2.col;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var selectorContent = [].concat(getSC(), getSelectorsGroup(), getSC());
|
|
|
|
var end = getLastPosition(selectorContent, _line2, _column2, 1);
|
|
var args = newNode(_type2, selectorContent, _line2, _column2, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `:nth-child(odd)`
|
|
* (2) `:nth-child(even)`
|
|
* (3) `:lang(de-DE)`
|
|
*/
|
|
function checkPseudoClass2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass2() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
var value = [].concat(getSC(), getIdentOrInterpolation(), getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(-3n + 2)`
|
|
*/
|
|
function checkPseudoClass3(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
|
|
if (l = checkNumberOrInterpolation(i)) i += l;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].value === 'n') i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.PlusSign || tokens[i].type === TokenType.HyphenMinus) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkNumberOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass3() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkNumberOrInterpolation(pos)) value = value.concat(getNumberOrInterpolation());
|
|
|
|
{
|
|
var _token3 = tokens[pos];
|
|
|
|
if (_token3.value === 'n') {
|
|
var _l = _token3.ln;
|
|
var _c = _token3.col;
|
|
var _content = _token3.value;
|
|
var ident = newNode(NodeType.IdentType, _content, _l, _c);
|
|
value.push(ident);
|
|
pos++;
|
|
}
|
|
}
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkNumberOrInterpolation(pos)) value = value.concat(getNumberOrInterpolation());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(-3n)`
|
|
*/
|
|
function checkPseudoClass4(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
|
|
if (l = checkInterpolation(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;
|
|
|
|
if (tokens[i].value === 'n') i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass4() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkInterpolation(pos)) value.push(getInterpolation());
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
if (checkIdent(pos)) value.push(getIdent());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:nth-child(+8)`
|
|
*/
|
|
function checkPseudoClass5(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
if (tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
var right = tokens[i].right;
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkUnary(i)) i += l;
|
|
if (tokens[i].type === TokenType.DecimalNumber) i++;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (i !== right) return 0;
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass5() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
var l = tokens[pos].ln;
|
|
var c = tokens[pos].col;
|
|
var value = [];
|
|
|
|
// Skip `(`.
|
|
pos++;
|
|
|
|
value = value.concat(getSC());
|
|
|
|
if (checkUnary(pos)) value.push(getUnary());
|
|
if (checkNumber(pos)) value.push(getNumber());
|
|
|
|
value = value.concat(getSC());
|
|
|
|
var end = getLastPosition(value, l, c, 1);
|
|
var args = newNode(NodeType.ArgumentsType, value, l, c, end);
|
|
content.push(args);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (-) `:checked`
|
|
*/
|
|
function checkPseudoClass6(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
// Skip `:`.
|
|
i++;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getPseudoClass6() {
|
|
var type = NodeType.PseudocType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `:`.
|
|
pos++;
|
|
|
|
var content = getIdentOrInterpolation();
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkRuleset(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkSelectorsGroup(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkBlock(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getRuleset() {
|
|
var type = NodeType.RulesetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [].concat(getSelectorsGroup(), getSC(), getBlock());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is marked as a space (if it's a space or a tab
|
|
* or a line break).
|
|
* @param {Number} i
|
|
* @returns {Number} Number of spaces in a row starting with the given token.
|
|
*/
|
|
function checkS(i) {
|
|
return i < tokensLength && tokens[i].ws ? tokens[i].ws_last - i + 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with spaces
|
|
* @returns {Array} `['s', x]` where `x` is a string containing spaces
|
|
*/
|
|
function getS() {
|
|
var type = NodeType.SType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = joinValues(pos, tokens[pos].ws_last);
|
|
|
|
pos = tokens[pos].ws_last + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a space or a comment.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Number of similar (space or comment) tokens
|
|
* in a row starting with the given token.
|
|
*/
|
|
function checkSC(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
var lsc = 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkS(i)) tokens[i].sc_child = 1;else if (l = checkCommentML(i)) tokens[i].sc_child = 2;else if (l = checkCommentSL(i)) tokens[i].sc_child = 3;else break;
|
|
|
|
i += l;
|
|
lsc += l;
|
|
}
|
|
|
|
return lsc || 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with spaces and comments
|
|
* @returns {Array} Array containing nodes with spaces (if there are any)
|
|
* and nodes with comments (if there are any):
|
|
* `[['s', x]*, ['comment', y]*]` where `x` is a string of spaces
|
|
* and `y` is a comment's text (without `/*` and `* /`).
|
|
*/
|
|
function getSC() {
|
|
var sc = [];
|
|
|
|
if (pos >= tokensLength) return sc;
|
|
|
|
while (pos < tokensLength) {
|
|
var childType = tokens[pos].sc_child;
|
|
|
|
if (childType === 1) sc.push(getS());else if (childType === 2) sc.push(getCommentML());else if (childType === 3) sc.push(getCommentSL());else break;
|
|
}
|
|
|
|
return sc;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside a simple
|
|
* selector
|
|
* @param {number} i Token's index number
|
|
* @return {number}
|
|
*/
|
|
function checkShash(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
|
|
|
|
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkIdentOrInterpolation(i) || checkPartialIdent(i)) i += l;else break;
|
|
}
|
|
|
|
tokens[start].shashEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a hexadecimal number (e.g. `#fff`) inside a simple selector
|
|
* @returns {Node}
|
|
*/
|
|
function getShash() {
|
|
var type = NodeType.ShashType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = token.shashEnd;
|
|
var content = [];
|
|
|
|
// Skip `#`.
|
|
pos++;
|
|
|
|
while (pos < end) {
|
|
if (checkIdentOrInterpolation(pos)) {
|
|
content = content.concat(getIdentOrInterpolation());
|
|
} else if (checkPartialIdent(pos)) {
|
|
content.push(getIdent());
|
|
} else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a string (text wrapped in quotes)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is part of a string, `0` if not
|
|
*/
|
|
function checkString(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.StringSQ || tokens[i].type === TokenType.StringDQ) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get string's node
|
|
* @returns {Array} `['string', x]` where `x` is a string (including
|
|
* quotes).
|
|
*/
|
|
function getString() {
|
|
var type = NodeType.StringType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Validate stylesheet: it should consist of any number (0 or more) of
|
|
* rulesets (sets of rules with selectors), @-rules, whitespaces or
|
|
* comments.
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkStylesheet(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkSC(i)) tokens[i].stylesheet_child = 1;else if (l = checkRuleset(i)) tokens[i].stylesheet_child = 2;else if (l = checkInclude(i)) tokens[i].stylesheet_child = 3;else if (l = checkExtend(i)) tokens[i].stylesheet_child = 4;else if (l = checkMixin(i)) tokens[i].stylesheet_child = 5;else if (l = checkLoop(i)) tokens[i].stylesheet_child = 6;else if (l = checkConditionalStatement(i)) tokens[i].stylesheet_child = 7;else if (l = checkAtrule(i)) tokens[i].stylesheet_child = 8;else if (l = checkDeclaration(i)) tokens[i].stylesheet_child = 9;else if (l = checkDeclDelim(i)) tokens[i].stylesheet_child = 10;else throwError(i);
|
|
|
|
i += l;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array} `['stylesheet', x]` where `x` is all stylesheet's
|
|
* nodes.
|
|
*/
|
|
function getStylesheet() {
|
|
var type = NodeType.StylesheetType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
var childType = tokens[pos].stylesheet_child;
|
|
|
|
if (childType === 1) content = content.concat(getSC());
|
|
if (childType === 2) content.push(getRuleset());
|
|
if (childType === 3) content.push(getInclude());
|
|
if (childType === 4) content.push(getExtend());
|
|
if (childType === 5) content.push(getMixin());
|
|
if (childType === 6) content.push(getLoop());
|
|
if (childType === 7) content.push(getConditionalStatement());
|
|
if (childType === 8) content.push(getAtrule());
|
|
if (childType === 9) content.push(getDeclaration());
|
|
if (childType === 10) content.push(getDeclDelim());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkTset(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkVhash(i)) tokens[i].tset_child = 1;else if (l = checkOperator(i)) tokens[i].tset_child = 2;else if (l = checkAny(i)) tokens[i].tset_child = 3;else if (l = checkSC(i)) tokens[i].tset_child = 4;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getTset() {
|
|
var childType = tokens[pos].tset_child;
|
|
|
|
if (childType === 1) return getVhash();
|
|
if (childType === 2) return getOperator();
|
|
if (childType === 3) return getAny();
|
|
if (childType === 4) return getSC();
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkTsets(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (l = checkTset(i)) {
|
|
i += l;
|
|
}
|
|
|
|
tokens[start].tsets_end = i;
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getTsets() {
|
|
var content = [];
|
|
var t = void 0;
|
|
|
|
if (pos >= tokensLength) return content;
|
|
|
|
var end = tokens[pos].tsets_end;
|
|
while (pos < end) {
|
|
t = getTset();
|
|
if (typeof t.content === 'string') content.push(t);else content = content.concat(t);
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is an unary (arithmetical) sign (`+` or `-`)
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} `1` if token is an unary sign, `0` if not
|
|
*/
|
|
function checkUnary(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type === TokenType.HyphenMinus || tokens[i].type === TokenType.PlusSign) {
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with an unary (arithmetical) sign (`+` or `-`)
|
|
* @returns {Array} `['unary', x]` where `x` is an unary sign
|
|
* converted to string.
|
|
*/
|
|
function getUnary() {
|
|
var type = NodeType.OperatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a unicode range (single or multiple <urange> nodes)
|
|
* @param {number} i Token's index
|
|
* @return {number} Unicode range node's length
|
|
*/
|
|
function checkUnicodeRange(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkUrange(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
var comma = checkDelim(i + spaceBefore);
|
|
if (!comma) break;
|
|
|
|
var spaceAfter = checkSC(i + spaceBefore + comma);
|
|
if (l = checkUrange(i + spaceBefore + comma + spaceAfter)) {
|
|
i += spaceBefore + comma + spaceAfter + l;
|
|
} else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a unicode range node
|
|
* @return {Node}
|
|
*/
|
|
function getUnicodeRange() {
|
|
var type = NodeType.UnicodeRangeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
while (pos < tokensLength) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkDelim(pos)) content.push(getDelim());else if (checkUrange(pos)) content.push(getUrange());else break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is unit
|
|
* @param {Number} i Token's index number
|
|
* @return {Number}
|
|
*/
|
|
function checkUnit(i) {
|
|
var units = ['em', 'ex', 'ch', 'rem', 'vh', 'vw', 'vmin', 'vmax', 'px', 'mm', 'q', 'cm', 'in', 'pt', 'pc', 'deg', 'grad', 'rad', 'turn', 's', 'ms', 'Hz', 'kHz', 'dpi', 'dpcm', 'dppx'];
|
|
|
|
return units.indexOf(tokens[i].value) !== -1 ? 1 : 0;
|
|
}
|
|
|
|
/**
|
|
* Get unit node of type ident
|
|
* @return {Node} An ident node containing the unit value
|
|
*/
|
|
function getUnit() {
|
|
var type = NodeType.IdentType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = token.value;
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is a u-range (part of a unicode-range)
|
|
* (1) `U+416`
|
|
* (2) `U+400-4ff`
|
|
* (3) `U+4??`
|
|
* @param {number} i Token's index
|
|
* @return {number} Urange node's length
|
|
*/
|
|
function checkUrange(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Check for unicode prefix (u+ or U+)
|
|
if (tokens[i].value === 'U' || tokens[i].value === 'u') i += 1;else return 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].value === '+') i += 1;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = _checkUnicodeWildcard(i)) i += l;else break;
|
|
}
|
|
|
|
tokens[start].urangeEnd = i - 1;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a u-range node (part of a unicode-range)
|
|
* @return {Node}
|
|
*/
|
|
function getUrange() {
|
|
var startPos = pos;
|
|
var type = NodeType.UrangeType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content = joinValues(startPos, tokens[startPos].urangeEnd);
|
|
pos = tokens[startPos].urangeEnd + 1;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check for unicode wildcard characters `?`
|
|
* @param {number} i Token's index
|
|
* @return {number} Wildcard length
|
|
*/
|
|
function _checkUnicodeWildcard(i) {
|
|
var start = i;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (tokens[i].type === TokenType.QuestionMark) i += 1;else break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of URI, e.g. `url('/css/styles.css')`
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of URI
|
|
*/
|
|
function checkUri(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength || tokens[i].value !== 'url') return 0;
|
|
|
|
// Skip `url`.
|
|
i++;
|
|
|
|
if (i >= tokensLength || tokens[i].type !== TokenType.LeftParenthesis) return 0;
|
|
|
|
// Store the opening parenthesis token as we will reference it's `right`
|
|
// property to determine when the parentheses close
|
|
var leftParenthesis = tokens[i];
|
|
|
|
// Skip `(`.
|
|
i++;
|
|
|
|
// Determine the type of URI
|
|
while (i < leftParenthesis.right) {
|
|
if (l = checkUri1(i)) {
|
|
i += l;
|
|
tokens[start].uriType = 1; // Raw based URI (without quotes)
|
|
} else if (l = checkUri2(i)) {
|
|
i += l;
|
|
tokens[start].uriType = 2; // Non-raw based URI (with quotes)
|
|
} else return 0;
|
|
}
|
|
|
|
// Skip `)`.
|
|
i++;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get specific type of URI node
|
|
* @return {Node} Specific type of URI node
|
|
*/
|
|
function getUri() {
|
|
var startPos = pos;
|
|
var type = NodeType.UriType;
|
|
var token = tokens[startPos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var end = void 0;
|
|
|
|
var uriType = tokens[startPos].uriType;
|
|
|
|
// Skip `url` and `(`.
|
|
pos += 2;
|
|
|
|
if (uriType === 1) content = content.concat(getUri1());else if (uriType === 2) content = content.concat(getUri2());else end = getLastPosition(content, line, column, 4);
|
|
|
|
if (!end) end = getLastPosition(content, line, column, 1);
|
|
|
|
// Skip `)`.
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token type is valid URI character
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of raw node
|
|
*/
|
|
function checkUriRawCharacters(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else if (l = checkNumber(i)) i += l;else {
|
|
switch (tokens[i].type) {
|
|
case TokenType.ExclamationMark:
|
|
case TokenType.NumberSign:
|
|
case TokenType.DollarSign:
|
|
case TokenType.PercentSign:
|
|
case TokenType.Ampersand:
|
|
case TokenType.Asterisk:
|
|
case TokenType.PlusSign:
|
|
case TokenType.Comma:
|
|
case TokenType.HyphenMinus:
|
|
case TokenType.FullStop:
|
|
case TokenType.Solidus:
|
|
case TokenType.Colon:
|
|
case TokenType.Semicolon:
|
|
case TokenType.LessThanSign:
|
|
case TokenType.EqualsSign:
|
|
case TokenType.GreaterThanSign:
|
|
case TokenType.QuotationMark:
|
|
case TokenType.CommercialAt:
|
|
case TokenType.LeftSquareBracket:
|
|
case TokenType.RightSquareBracket:
|
|
case TokenType.CircumflexAccent:
|
|
case TokenType.LowLine:
|
|
case TokenType.LeftCurlyBracket:
|
|
case TokenType.VerticalLine:
|
|
case TokenType.RightCurlyBracket:
|
|
case TokenType.Tilde:
|
|
i += 1;
|
|
break;
|
|
|
|
default:
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Check if content of URI can be contained within a raw node
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of raw node
|
|
*/
|
|
function checkUriRaw(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (checkInterpolation(i) || checkVariable(i)) break;else if (l = checkUriRawCharacters(i)) i += l;else break;
|
|
}
|
|
|
|
tokens[start].uri_raw_end = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a raw node
|
|
* @return {Node}
|
|
*/
|
|
function getUriRaw() {
|
|
var startPos = pos;
|
|
var type = NodeType.RawType;
|
|
var token = tokens[startPos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var l = void 0;
|
|
|
|
while (pos < tokens[startPos].uri_raw_end) {
|
|
if (checkInterpolation(pos) || checkVariable(pos)) break;else if (l = checkUriRawCharacters(pos)) pos += l;else break;
|
|
}
|
|
|
|
content = joinValues(startPos, pos - 1);
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check for a raw (without quotes) URI
|
|
* (1) http://foo.com/bar.png
|
|
* (2) http://foo.com/#{$bar}.png
|
|
* (3) #{$foo}/bar.png
|
|
* (4) #{$foo}
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of URI node
|
|
*/
|
|
function checkUri1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkInterpolation(i) || checkUriRaw(i)) i += l;else break;
|
|
}
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
// Check that we are at the end of the uri
|
|
if (i < tokens[start - 1].right) return 0;
|
|
|
|
tokens[start].uri_end = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a raw (without quotes) URI
|
|
node
|
|
* @return {Array}
|
|
*/
|
|
function getUri1() {
|
|
var startPos = pos;
|
|
var content = [];
|
|
|
|
if (checkSC(pos)) content = content.concat(getSC());
|
|
|
|
while (pos < tokens[startPos].uri_end) {
|
|
if (checkInterpolation(pos)) content.push(getInterpolation());else if (checkUriRaw(pos)) content.push(getUriRaw());else break;
|
|
}
|
|
|
|
if (checkSC(pos)) content = content.concat(getSC());
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check for a non-raw (with quotes) URI
|
|
* (1) 'http://foo.com/bar.png'
|
|
* (2) 'http://foo.com/'#{$bar}.png
|
|
* (3) #{$foo}'/bar.png'
|
|
* @param {number} i Token's index number
|
|
* @return {number} Length of URI node
|
|
*/
|
|
function checkUri2(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (l = checkSC(i)) i += l;else if (l = checkString(i)) i += l;else if (l = checkFunction(i)) i += l;else if (l = checkUnary(i)) i += l;else if (l = checkIdentOrInterpolation(i)) i += l;else if (l = checkVariable(i)) i += l;else break;
|
|
}
|
|
|
|
// Check that we are at the end of the uri
|
|
if (i < tokens[start - 1].right) return 0;
|
|
|
|
tokens[start].uri_end = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get a non-raw (with quotes) URI node
|
|
* @return {Array}
|
|
*/
|
|
function getUri2() {
|
|
var startPos = pos;
|
|
var content = [];
|
|
|
|
while (pos < tokens[startPos].uri_end) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkUnary(pos)) content.push(getUnary());else if (_checkValue(pos)) content.push(_getValue());else break;
|
|
}
|
|
|
|
return content;
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a value
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the value
|
|
*/
|
|
function checkValue(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
var s = void 0;
|
|
var _i = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (checkDeclDelim(i)) break;
|
|
|
|
s = checkSC(i);
|
|
_i = i + s;
|
|
|
|
if (l = _checkValue(_i)) i += l + s;
|
|
if (!l || checkBlock(i - l)) break;
|
|
}
|
|
|
|
tokens[start].value_end = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getValue() {
|
|
var type = NodeType.ValueType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var end = tokens[pos].value_end;
|
|
var content = [];
|
|
var _pos = void 0;
|
|
var s = void 0;
|
|
|
|
while (pos < end) {
|
|
s = checkSC(pos);
|
|
_pos = pos + s;
|
|
|
|
if (checkDeclDelim(_pos)) break;
|
|
|
|
if (!_checkValue(_pos)) break;
|
|
|
|
if (s) content = content.concat(getSC());
|
|
content.push(_getValue());
|
|
|
|
if (checkBlock(_pos)) break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function _checkValue(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkInterpolation(i)) tokens[i].value_child = 1;else if (l = checkVariable(i)) tokens[i].value_child = 2;else if (l = checkVhash(i)) tokens[i].value_child = 3;else if (l = checkBlock(i)) tokens[i].value_child = 4;else if (l = checkAtkeyword(i)) tokens[i].value_child = 5;else if (l = checkOperator(i)) tokens[i].value_child = 6;else if (l = checkImportant(i)) tokens[i].value_child = 7;else if (l = checkGlobal(i)) tokens[i].value_child = 8;else if (l = checkDefault(i)) tokens[i].value_child = 9;else if (l = checkProgid(i)) tokens[i].value_child = 10;else if (l = checkAny(i)) tokens[i].value_child = 11;else if (l = checkParentSelector(i)) tokens[i].value_child = 12;
|
|
|
|
return l;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function _getValue() {
|
|
var childType = tokens[pos].value_child;
|
|
if (childType === 1) return getInterpolation();
|
|
if (childType === 2) return getVariable();
|
|
if (childType === 3) return getVhash();
|
|
if (childType === 4) return getBlock();
|
|
if (childType === 5) return getAtkeyword();
|
|
if (childType === 6) return getOperator();
|
|
if (childType === 7) return getImportant();
|
|
if (childType === 8) return getGlobal();
|
|
if (childType === 9) return getDefault();
|
|
if (childType === 10) return getProgid();
|
|
if (childType === 11) return getAny();
|
|
if (childType === 12) return getParentSelector();
|
|
}
|
|
|
|
/**
|
|
* @param {number} i Token's index number
|
|
* @returns {number} Length of the value
|
|
*/
|
|
function checkSingleValue(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
var s = void 0;
|
|
var _i = void 0;
|
|
|
|
while (i < tokensLength) {
|
|
if (checkDeclDelim(i) || checkDelim(i)) break;
|
|
|
|
s = checkSC(i);
|
|
_i = i + s;
|
|
|
|
if (l = _checkValue(_i)) i += l + s;
|
|
if (!l || checkBlock(i - l)) break;
|
|
}
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @returns {Array}
|
|
*/
|
|
function getSingleValue() {
|
|
var type = NodeType.ValueType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var _pos = void 0;
|
|
var s = void 0;
|
|
|
|
while (pos < tokensLength) {
|
|
s = checkSC(pos);
|
|
_pos = pos + s;
|
|
|
|
if (checkDeclDelim(_pos) || checkDelim(_pos)) break;
|
|
|
|
if (!_checkValue(_pos)) break;
|
|
|
|
if (s) content = content.concat(getSC());
|
|
content.push(_getValue());
|
|
|
|
if (checkBlock(_pos)) break;
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a variable
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number} Length of the variable
|
|
*/
|
|
function checkVariable(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Skip `$`.
|
|
if (tokens[i].type === TokenType.DollarSign) i++;else return 0;
|
|
|
|
if (l = checkIdent(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a variable
|
|
* @returns {Array} `['variable', ['ident', x]]` where `x` is
|
|
* a variable name.
|
|
*/
|
|
function getVariable() {
|
|
var type = NodeType.VariableType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `$`.
|
|
pos++;
|
|
|
|
var content = [getIdent()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a variables list (e.g. `$values...`).
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkVariablesList(i) {
|
|
var d = 0; // Number of dots
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (l = checkVariable(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength && tokens[i].type === TokenType.FullStop) {
|
|
d++;
|
|
i++;
|
|
}
|
|
|
|
return d === 3 ? l + d : 0;
|
|
}
|
|
|
|
/**
|
|
* Get node with a variables list
|
|
* @returns {Array} `['variableslist', ['variable', ['ident', x]]]` where
|
|
* `x` is a variable name.
|
|
*/
|
|
function getVariablesList() {
|
|
var type = NodeType.VariablesListType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getVariable()];
|
|
var end = getLastPosition(content, line, column, 3);
|
|
|
|
// Skip `...`.
|
|
pos += 3;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a hexadecimal number (e.g. `#fff`) inside
|
|
* some value
|
|
* @param {Number} i Token's index number
|
|
* @returns {Number}
|
|
*/
|
|
function checkVhash(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (i >= tokensLength) return 0;
|
|
|
|
// Skip `#`.
|
|
if (tokens[i].type === TokenType.NumberSign) i++;else return 0;
|
|
|
|
if (l = checkNmName2(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get node with a hexadecimal number (e.g. `#fff`) inside some value
|
|
* @returns {Array} `['vhash', x]` where `x` is a hexadecimal number
|
|
* converted to string (without `#`, e.g. `'fff'`).
|
|
*/
|
|
function getVhash() {
|
|
var type = NodeType.VhashType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
|
|
// Skip `#`.
|
|
pos++;
|
|
|
|
var content = getNmName2();
|
|
var end = getLastPosition(content, line, column + 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkSelectorsGroup(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
var selectorCounter = 0;
|
|
var delimCounter = 0;
|
|
|
|
if (l = checkSelector(i)) {
|
|
i += l;
|
|
selectorCounter++;
|
|
} else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var tempStart = i;
|
|
var tempIndex = i;
|
|
var tempLength = void 0;
|
|
|
|
var spaceBefore = checkSC(tempIndex);
|
|
|
|
if (tempLength = checkDelim(tempIndex + spaceBefore)) {
|
|
tempIndex += spaceBefore + tempLength;
|
|
delimCounter++;
|
|
|
|
if (tempLength = checkSC(tempIndex)) tempIndex += tempLength;
|
|
if (tempLength = checkSelector(tempIndex)) {
|
|
tempIndex += tempLength;
|
|
selectorCounter++;
|
|
}
|
|
} else break;
|
|
|
|
i += tempIndex - tempStart;
|
|
}
|
|
|
|
tokens[start].selectorsGroupEnd = i;
|
|
tokens[start].selectorsGroupSelectorCount = selectorCounter;
|
|
tokens[start].selectorsGroupDelimCount = delimCounter;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getSelectorsGroup() {
|
|
var selectorsGroup = [];
|
|
var selectorCounter = 0;
|
|
var delimCounter = 0;
|
|
|
|
var selectorsGroupEnd = tokens[pos].selectorsGroupEnd;
|
|
var selectorCount = tokens[pos].selectorsGroupSelectorCount;
|
|
var delimCount = tokens[pos].selectorsGroupDelimCount;
|
|
|
|
selectorsGroup.push(getSelector());
|
|
selectorCounter++;
|
|
|
|
while (pos < selectorsGroupEnd) {
|
|
if (delimCounter < delimCount) {
|
|
selectorsGroup = selectorsGroup.concat(getSC());
|
|
selectorsGroup = selectorsGroup.concat(getDelim());
|
|
delimCounter++;
|
|
|
|
selectorsGroup = selectorsGroup.concat(getSC());
|
|
|
|
if (selectorCounter < selectorCount) {
|
|
selectorsGroup = selectorsGroup.concat(getSelector());
|
|
selectorCounter++;
|
|
}
|
|
}
|
|
}
|
|
|
|
return selectorsGroup;
|
|
}
|
|
|
|
function checkSelector(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkSelector1(i)) tokens[i].selectorType = 1;else if (l = checkSelector2(i)) tokens[i].selectorType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getSelector() {
|
|
var selectorType = tokens[pos].selectorType;
|
|
if (selectorType === 1) return getSelector1();else return getSelector2();
|
|
}
|
|
|
|
/**
|
|
* Checks for selector which starts with a compound selector.
|
|
*/
|
|
function checkSelector1(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkCompoundSelector(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var space = checkSC(i);
|
|
var comma = checkCombinator(i + space);
|
|
if (!space && !comma) break;
|
|
|
|
if (comma) {
|
|
i += space + comma;
|
|
space = checkSC(i);
|
|
}
|
|
|
|
if (l = checkCompoundSelector(i + space)) i += space + l;else break;
|
|
}
|
|
|
|
tokens[start].selectorEnd = i;
|
|
return i - start;
|
|
}
|
|
|
|
function getSelector1() {
|
|
var type = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var selectorEnd = token.selectorEnd;
|
|
var content = getCompoundSelector();
|
|
|
|
while (pos < selectorEnd) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* Checks for a selector that starts with a combinator.
|
|
*/
|
|
function checkSelector2(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkCombinator(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var spaceBefore = checkSC(i);
|
|
if (l = checkCompoundSelector(i + spaceBefore)) i += spaceBefore + l;else break;
|
|
|
|
var spaceAfter = checkSC(i);
|
|
var comma = checkCombinator(i + spaceAfter);
|
|
if (!spaceAfter && !comma) break;
|
|
if (comma) {
|
|
i += spaceAfter + comma;
|
|
}
|
|
}
|
|
|
|
tokens[start].selectorEnd = i;
|
|
return i - start;
|
|
}
|
|
|
|
function getSelector2() {
|
|
var type = NodeType.SelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var selectorEnd = token.selectorEnd;
|
|
var content = [getCombinator()];
|
|
|
|
while (pos < selectorEnd) {
|
|
if (checkSC(pos)) content = content.concat(getSC());else if (checkCombinator(pos)) content.push(getCombinator());else if (checkCompoundSelector(pos)) content = content.concat(getCompoundSelector());
|
|
}
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkCompoundSelector(i) {
|
|
var l = void 0;
|
|
|
|
if (l = checkCompoundSelector1(i)) {
|
|
tokens[i].compoundSelectorType = 1;
|
|
} else if (l = checkCompoundSelector2(i)) {
|
|
tokens[i].compoundSelectorType = 2;
|
|
}
|
|
|
|
return l;
|
|
}
|
|
|
|
function getCompoundSelector() {
|
|
var type = tokens[pos].compoundSelectorType;
|
|
if (type === 1) return getCompoundSelector1();
|
|
if (type === 2) return getCompoundSelector2();
|
|
}
|
|
|
|
/**
|
|
* Check for compound selectors that start with either a type selector,
|
|
* placeholder or parent selector with extension
|
|
* (1) `foo.bar`
|
|
* (2) `foo[attr=val]`
|
|
* (3) `foo:first-of-type`
|
|
* (4) `foo%bar`
|
|
* @param {number} i Token's index
|
|
* @return {number} Compound selector's length
|
|
*/
|
|
function checkCompoundSelector1(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkUniversalSelector(i) || checkTypeSelector(i) || checkPlaceholder(i) || checkParentSelectorWithExtension(i)) i += l;else return 0;
|
|
|
|
while (i < tokensLength) {
|
|
var _l2 = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i) || checkPlaceholder(i) || checkInterpolation(i);
|
|
|
|
if (_l2) i += _l2;else break;
|
|
}
|
|
|
|
tokens[start].compoundSelectorEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array} An array of nodes that make up the compound selector
|
|
*/
|
|
function getCompoundSelector1() {
|
|
var sequence = [];
|
|
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
|
|
|
|
if (checkUniversalSelector(pos)) sequence.push(getUniversalSelector());else if (checkTypeSelector(pos)) sequence.push(getTypeSelector());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkParentSelectorWithExtension(pos)) sequence = sequence.concat(getParentSelectorWithExtension());
|
|
|
|
while (pos < compoundSelectorEnd) {
|
|
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkInterpolation(pos)) sequence.push(getInterpolation());else break;
|
|
}
|
|
|
|
return sequence;
|
|
}
|
|
|
|
/**
|
|
* Check for all other compound selectors
|
|
* (1) `.foo.bar`
|
|
* (2) `.foo[attr=val]`
|
|
* (3) `.foo:first-of-type`
|
|
* (4) `.foo%bar`
|
|
* (5) `.foo#{$bar}`
|
|
* @param {number} i Token's index
|
|
* @return {number} Compound selector's length
|
|
*/
|
|
function checkCompoundSelector2(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
|
|
while (i < tokensLength) {
|
|
var l = checkShash(i) || checkClass(i) || checkAttributeSelector(i) || checkPseudo(i) || checkPlaceholder(i) || checkInterpolation(i);
|
|
|
|
if (l) i += l;else break;
|
|
}
|
|
|
|
tokens[start].compoundSelectorEnd = i;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* @return {Array} An array of nodes that make up the compound selector
|
|
*/
|
|
function getCompoundSelector2() {
|
|
var sequence = [];
|
|
var compoundSelectorEnd = tokens[pos].compoundSelectorEnd;
|
|
|
|
while (pos < compoundSelectorEnd) {
|
|
if (checkShash(pos)) sequence.push(getShash());else if (checkClass(pos)) sequence.push(getClass());else if (checkAttributeSelector(pos)) sequence.push(getAttributeSelector());else if (checkPseudo(pos)) sequence.push(getPseudo());else if (checkPlaceholder(pos)) sequence.push(getPlaceholder());else if (checkInterpolation(pos)) sequence.push(getInterpolation());else break;
|
|
}
|
|
|
|
return sequence;
|
|
}
|
|
|
|
function checkUniversalSelector(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.Asterisk) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getUniversalSelector() {
|
|
var type = NodeType.UniversalSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
var end = void 0;
|
|
|
|
if (checkNamePrefix(pos)) {
|
|
content.push(getNamePrefix());
|
|
end = getLastPosition(content, line, column, 1);
|
|
}
|
|
|
|
pos++;
|
|
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* Check if token is part of a type selector
|
|
* @param {number} i Token's index
|
|
* @return {number} Type selector's length
|
|
*/
|
|
function checkTypeSelector(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
/**
|
|
* Get type selector node
|
|
* @return {Node}
|
|
*/
|
|
function getTypeSelector() {
|
|
var type = NodeType.TypeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkNamePrefix(pos)) content.push(getNamePrefix());
|
|
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeSelector(i) {
|
|
var l = void 0;
|
|
if (l = checkAttributeSelector1(i)) tokens[i].attributeSelectorType = 1;else if (l = checkAttributeSelector2(i)) tokens[i].attributeSelectorType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getAttributeSelector() {
|
|
var type = tokens[pos].attributeSelectorType;
|
|
if (type === 1) return getAttributeSelector1();else return getAttributeSelector2();
|
|
}
|
|
|
|
/**
|
|
* (1) `[panda=nani]`
|
|
* (2) `[panda='nani']`
|
|
* (3) `[panda='nani' i]`
|
|
*
|
|
*/
|
|
function checkAttributeSelector1(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeName(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeMatch(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeValue(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeFlags(i)) {
|
|
i += l;
|
|
if (l = checkSC(i)) i += l;
|
|
}
|
|
|
|
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeSelector1() {
|
|
var type = NodeType.AttributeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC(), getAttributeName(), getSC(), getAttributeMatch(), getSC(), getAttributeValue(), getSC());
|
|
|
|
if (checkAttributeFlags(pos)) {
|
|
content.push(getAttributeFlags());
|
|
content = content.concat(getSC());
|
|
}
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
/**
|
|
* (1) `[panda]`
|
|
*/
|
|
function checkAttributeSelector2(i) {
|
|
var start = i;
|
|
|
|
if (tokens[i].type === TokenType.LeftSquareBracket) i++;else return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (l = checkAttributeName(i)) i += l;else return 0;
|
|
|
|
if (l = checkSC(i)) i += l;
|
|
|
|
if (tokens[i].type === TokenType.RightSquareBracket) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeSelector2() {
|
|
var type = NodeType.AttributeSelectorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
// Skip `[`.
|
|
pos++;
|
|
|
|
content = content.concat(getSC(), getAttributeName(), getSC());
|
|
|
|
// Skip `]`.
|
|
pos++;
|
|
|
|
var end = getLastPosition(content, line, column, 1);
|
|
return newNode(type, content, line, column, end);
|
|
}
|
|
|
|
function checkAttributeName(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamePrefix(i)) i += l;
|
|
|
|
if (l = checkIdentOrInterpolation(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeName() {
|
|
var type = NodeType.AttributeNameType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkNamePrefix(pos)) content.push(getNamePrefix());
|
|
content = content.concat(getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeMatch(i) {
|
|
var l = void 0;
|
|
if (l = checkAttributeMatch1(i)) tokens[i].attributeMatchType = 1;else if (l = checkAttributeMatch2(i)) tokens[i].attributeMatchType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getAttributeMatch() {
|
|
var type = tokens[pos].attributeMatchType;
|
|
if (type === 1) return getAttributeMatch1();else return getAttributeMatch2();
|
|
}
|
|
|
|
function checkAttributeMatch1(i) {
|
|
var start = i;
|
|
|
|
var type = tokens[i].type;
|
|
if (type === TokenType.Tilde || type === TokenType.VerticalLine || type === TokenType.CircumflexAccent || type === TokenType.DollarSign || type === TokenType.Asterisk) i++;else return 0;
|
|
|
|
if (tokens[i].type === TokenType.EqualsSign) i++;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getAttributeMatch1() {
|
|
var type = NodeType.AttributeMatchType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = tokens[pos].value + tokens[pos + 1].value;
|
|
pos += 2;
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeMatch2(i) {
|
|
if (tokens[i].type === TokenType.EqualsSign) return 1;else return 0;
|
|
}
|
|
|
|
function getAttributeMatch2() {
|
|
var type = NodeType.AttributeMatchType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '=';
|
|
|
|
pos++;
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeValue(i) {
|
|
return checkString(i) || checkIdentOrInterpolation(i);
|
|
}
|
|
|
|
function getAttributeValue() {
|
|
var type = NodeType.AttributeValueType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (checkString(pos)) content.push(getString());else content = content.concat(getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkAttributeFlags(i) {
|
|
return checkIdentOrInterpolation(i);
|
|
}
|
|
|
|
function getAttributeFlags() {
|
|
var type = NodeType.AttributeFlagsType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = getIdentOrInterpolation();
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
function checkNamePrefix(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
if (l = checkNamePrefix1(i)) tokens[i].namePrefixType = 1;else if (l = checkNamePrefix2(i)) tokens[i].namePrefixType = 2;
|
|
|
|
return l;
|
|
}
|
|
|
|
function getNamePrefix() {
|
|
var type = tokens[pos].namePrefixType;
|
|
if (type === 1) return getNamePrefix1();else return getNamePrefix2();
|
|
}
|
|
|
|
/**
|
|
* (1) `panda|`
|
|
* (2) `panda<comment>|`
|
|
*/
|
|
function checkNamePrefix1(i) {
|
|
var start = i;
|
|
var l = void 0;
|
|
|
|
if (l = checkNamespacePrefix(i)) i += l;else return 0;
|
|
|
|
if (l = checkCommentML(i)) i += l;
|
|
|
|
if (l = checkNamespaceSeparator(i)) i += l;else return 0;
|
|
|
|
return i - start;
|
|
}
|
|
|
|
function getNamePrefix1() {
|
|
var type = NodeType.NamePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
content.push(getNamespacePrefix());
|
|
|
|
if (checkCommentML(pos)) content.push(getCommentML());
|
|
|
|
content.push(getNamespaceSeparator());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `|`
|
|
*/
|
|
function checkNamePrefix2(i) {
|
|
return checkNamespaceSeparator(i);
|
|
}
|
|
|
|
function getNamePrefix2() {
|
|
var type = NodeType.NamePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [getNamespaceSeparator()];
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `*`
|
|
* (2) `panda`
|
|
*/
|
|
function checkNamespacePrefix(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
var l = void 0;
|
|
|
|
if (tokens[i].type === TokenType.Asterisk) return 1;else if (l = checkIdentOrInterpolation(i)) return l;else return 0;
|
|
}
|
|
|
|
function getNamespacePrefix() {
|
|
var type = NodeType.NamespacePrefixType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = [];
|
|
|
|
if (token.type === TokenType.Asterisk) {
|
|
var asteriskNode = newNode(NodeType.IdentType, '*', line, column);
|
|
content.push(asteriskNode);
|
|
pos++;
|
|
} else if (checkIdentOrInterpolation(pos)) content = content.concat(getIdentOrInterpolation());
|
|
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
/**
|
|
* (1) `|`
|
|
*/
|
|
function checkNamespaceSeparator(i) {
|
|
if (i >= tokensLength) return 0;
|
|
|
|
if (tokens[i].type !== TokenType.VerticalLine) return 0;
|
|
|
|
// Return false if `|=` - [attr|=value]
|
|
if (tokens[i + 1] && tokens[i + 1].type === TokenType.EqualsSign) return 0;
|
|
|
|
return 1;
|
|
}
|
|
|
|
function getNamespaceSeparator() {
|
|
var type = NodeType.NamespaceSeparatorType;
|
|
var token = tokens[pos];
|
|
var line = token.ln;
|
|
var column = token.col;
|
|
var content = '|';
|
|
|
|
pos++;
|
|
return newNode(type, content, line, column);
|
|
}
|
|
|
|
module.exports = function (_tokens, context) {
|
|
tokens = _tokens;
|
|
tokensLength = tokens.length;
|
|
pos = 0;
|
|
|
|
return contexts[context]();
|
|
};
|
|
|
|
/***/ }),
|
|
/* 28 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
module.exports = function (css, tabSize) {
|
|
var TokenType = __webpack_require__(13);
|
|
|
|
var tokens = [];
|
|
var urlMode = false;
|
|
var c = void 0; // Current character
|
|
var cn = void 0; // Next character
|
|
var pos = 0;
|
|
var tn = 0;
|
|
var ln = 1;
|
|
var col = 1;
|
|
|
|
var Punctuation = {
|
|
' ': TokenType.Space,
|
|
'\n': TokenType.Newline,
|
|
'\r': TokenType.Newline,
|
|
'\t': TokenType.Tab,
|
|
'!': TokenType.ExclamationMark,
|
|
'"': TokenType.QuotationMark,
|
|
'#': TokenType.NumberSign,
|
|
'$': TokenType.DollarSign,
|
|
'%': TokenType.PercentSign,
|
|
'&': TokenType.Ampersand,
|
|
'\'': TokenType.Apostrophe,
|
|
'(': TokenType.LeftParenthesis,
|
|
')': TokenType.RightParenthesis,
|
|
'*': TokenType.Asterisk,
|
|
'+': TokenType.PlusSign,
|
|
',': TokenType.Comma,
|
|
'-': TokenType.HyphenMinus,
|
|
'.': TokenType.FullStop,
|
|
'/': TokenType.Solidus,
|
|
':': TokenType.Colon,
|
|
';': TokenType.Semicolon,
|
|
'<': TokenType.LessThanSign,
|
|
'=': TokenType.EqualsSign,
|
|
'==': TokenType.EqualitySign,
|
|
'!=': TokenType.InequalitySign,
|
|
'>': TokenType.GreaterThanSign,
|
|
'?': TokenType.QuestionMark,
|
|
'@': TokenType.CommercialAt,
|
|
'[': TokenType.LeftSquareBracket,
|
|
']': TokenType.RightSquareBracket,
|
|
'^': TokenType.CircumflexAccent,
|
|
'_': TokenType.LowLine,
|
|
'{': TokenType.LeftCurlyBracket,
|
|
'|': TokenType.VerticalLine,
|
|
'}': TokenType.RightCurlyBracket,
|
|
'~': TokenType.Tilde,
|
|
'`': TokenType.Backtick
|
|
};
|
|
|
|
/**
|
|
* Add a token to the token list
|
|
* @param {string} type
|
|
* @param {string} value
|
|
*/
|
|
function pushToken(type, value, column) {
|
|
tokens.push({
|
|
tn: tn++,
|
|
ln: ln,
|
|
col: column,
|
|
type: type,
|
|
value: value
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Check if a character is a decimal digit
|
|
* @param {string} c Character
|
|
* @returns {boolean}
|
|
*/
|
|
function isDecimalDigit(c) {
|
|
return '0123456789'.indexOf(c) >= 0;
|
|
}
|
|
|
|
/**
|
|
* Parse spaces
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseSpaces(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a non-space character:
|
|
for (; pos < css.length; pos++) {
|
|
if (css.charAt(pos) !== ' ') break;
|
|
}
|
|
|
|
// Add a substring containing only spaces to tokens:
|
|
pushToken(TokenType.Space, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse a string within quotes
|
|
* @param {string} css Unparsed part of CSS string
|
|
* @param {string} q Quote (either `'` or `"`)
|
|
*/
|
|
function parseString(css, q) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a matching quote:
|
|
for (pos++; pos < css.length; pos++) {
|
|
// Skip escaped quotes:
|
|
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) === q) break;
|
|
}
|
|
|
|
// Add the string (including quotes) to tokens:
|
|
var type = q === '"' ? TokenType.StringDQ : TokenType.StringSQ;
|
|
pushToken(type, css.substring(start, pos + 1), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse numbers
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseDecimalNumber(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet a character that's not a digit:
|
|
for (; pos < css.length; pos++) {
|
|
if (!isDecimalDigit(css.charAt(pos))) break;
|
|
}
|
|
|
|
// Add the number to tokens:
|
|
pushToken(TokenType.DecimalNumber, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse identifier
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseIdentifier(css) {
|
|
var start = pos;
|
|
|
|
// Skip all opening slashes:
|
|
while (css.charAt(pos) === '/') {
|
|
pos++;
|
|
} // Read the string until we meet a punctuation mark:
|
|
for (; pos < css.length; pos++) {
|
|
// Skip all '\':
|
|
if (css.charAt(pos) === '\\') pos++;else if (css.charAt(pos) in Punctuation) break;
|
|
}
|
|
|
|
var ident = css.substring(start, pos--);
|
|
|
|
// Enter url mode if parsed substring is `url`:
|
|
if (!urlMode && ident === 'url' && css.charAt(pos + 1) === '(') {
|
|
urlMode = true;
|
|
}
|
|
|
|
// Add identifier to tokens:
|
|
pushToken(TokenType.Identifier, ident, col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Parse equality sign
|
|
*/
|
|
function parseEquality() {
|
|
pushToken(TokenType.EqualitySign, '==', col);
|
|
pos++;
|
|
col++;
|
|
}
|
|
|
|
/**
|
|
* Parse inequality sign
|
|
*/
|
|
function parseInequality() {
|
|
pushToken(TokenType.InequalitySign, '!=', col);
|
|
pos++;
|
|
col++;
|
|
}
|
|
|
|
/**
|
|
* Parse a multiline comment
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseMLComment(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet `*/`.
|
|
// Since we already know first 2 characters (`/*`), start reading
|
|
// from `pos + 2`:
|
|
for (pos += 2; pos < css.length; pos++) {
|
|
if (css.charAt(pos) === '*' && css.charAt(pos + 1) === '/') {
|
|
pos++;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Add full comment (including `/*` and `*/`) to the list of tokens:
|
|
var comment = css.substring(start, pos + 1);
|
|
pushToken(TokenType.CommentML, comment, col);
|
|
|
|
var newlines = comment.split('\n');
|
|
if (newlines.length > 1) {
|
|
ln += newlines.length - 1;
|
|
col = newlines[newlines.length - 1].length;
|
|
} else {
|
|
col += pos - start;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Parse a single line comment
|
|
* @param {string} css Unparsed part of CSS string
|
|
*/
|
|
function parseSLComment(css) {
|
|
var start = pos;
|
|
|
|
// Read the string until we meet line break.
|
|
// Since we already know first 2 characters (`//`), start reading
|
|
// from `pos + 2`:
|
|
for (pos += 2; pos < css.length; pos++) {
|
|
if (css.charAt(pos) === '\n' || css.charAt(pos) === '\r') {
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Add comment (including `//` and line break) to the list of tokens:
|
|
pushToken(TokenType.CommentSL, css.substring(start, pos--), col);
|
|
col += pos - start;
|
|
}
|
|
|
|
/**
|
|
* Convert a CSS string to a list of tokens
|
|
* @param {string} css CSS string
|
|
* @returns {Array} List of tokens
|
|
* @private
|
|
*/
|
|
function getTokens(css) {
|
|
// Parse string, character by character:
|
|
for (pos = 0; pos < css.length; col++, pos++) {
|
|
c = css.charAt(pos);
|
|
cn = css.charAt(pos + 1);
|
|
|
|
// If we meet `/*`, it's a start of a multiline comment.
|
|
// Parse following characters as a multiline comment:
|
|
if (c === '/' && cn === '*') {
|
|
parseMLComment(css);
|
|
}
|
|
|
|
// If we meet `//` and it is not a part of url:
|
|
else if (!urlMode && c === '/' && cn === '/') {
|
|
// If we're currently inside a block, treat `//` as a start
|
|
// of identifier. Else treat `//` as a start of a single-line
|
|
// comment:
|
|
parseSLComment(css);
|
|
}
|
|
|
|
// If current character is a double or single quote, it's a start
|
|
// of a string:
|
|
else if (c === '"' || c === "'") {
|
|
parseString(css, c);
|
|
}
|
|
|
|
// If current character is a space:
|
|
else if (c === ' ') {
|
|
parseSpaces(css);
|
|
}
|
|
|
|
// If current character is `=`, it must be combined with next `=`
|
|
else if (c === '=' && cn === '=') {
|
|
parseEquality(css);
|
|
}
|
|
|
|
// If we meet `!=`, this must be inequality
|
|
else if (c === '!' && cn === '=') {
|
|
parseInequality(css);
|
|
}
|
|
|
|
// If current character is a punctuation mark:
|
|
else if (c in Punctuation) {
|
|
// Check for CRLF here or just LF
|
|
if (c === '\r' && cn === '\n' || c === '\n') {
|
|
// If \r we know the next character is \n due to statement above
|
|
// so we push a CRLF token type to the token list and importantly
|
|
// skip the next character so as not to double count newlines or
|
|
// columns etc
|
|
if (c === '\r') {
|
|
pushToken(TokenType.Newline, '\r\n', col);
|
|
pos++; // If CRLF skip the next character and push crlf token
|
|
} else if (c === '\n') {
|
|
// If just a LF newline and not part of CRLF newline we can just
|
|
// push punctuation as usual
|
|
pushToken(Punctuation[c], c, col);
|
|
}
|
|
|
|
ln++; // Go to next line
|
|
col = 0; // Reset the column count
|
|
} else if (c !== '\r' && c !== '\n') {
|
|
// Handle all other punctuation and add to list of tokens
|
|
pushToken(Punctuation[c], c, col);
|
|
} // Go to next line
|
|
if (c === ')') urlMode = false; // Exit url mode
|
|
else if (c === '\t' && tabSize > 1) col += tabSize - 1;
|
|
}
|
|
|
|
// If current character is a decimal digit:
|
|
else if (isDecimalDigit(c)) {
|
|
parseDecimalNumber(css);
|
|
}
|
|
|
|
// If current character is anything else:
|
|
else {
|
|
parseIdentifier(css);
|
|
}
|
|
}
|
|
|
|
return tokens;
|
|
}
|
|
|
|
return getTokens(css);
|
|
};
|
|
|
|
/***/ }),
|
|
/* 29 */
|
|
/***/ (function(module, exports, __webpack_require__) {
|
|
|
|
'use strict';
|
|
|
|
var Node = __webpack_require__(1);
|
|
var NodeTypes = __webpack_require__(15);
|
|
|
|
module.exports = function () {
|
|
return new Node({
|
|
type: NodeTypes.StylesheetType,
|
|
content: [],
|
|
start: [0, 0],
|
|
end: [0, 0]
|
|
});
|
|
};
|
|
|
|
/***/ })
|
|
/******/ ])
|
|
});
|
|
; |