UNPKG

stylus

Version:

Robust, expressive, and feature-rich CSS superset

251 lines (216 loc) 4.7 kB
/*! * Stylus - Node * Copyright (c) Automattic <developer.wordpress.com> * MIT Licensed */ /** * Module dependencies. */ var Evaluator = require('../visitor/evaluator') , utils = require('../utils') , nodes = require('./'); class CoercionError extends Error { /** * Initialize a new `CoercionError` with the given `msg`. * * @param {String} msg * @api private */ constructor(msg) { super(); this.name = 'CoercionError' this.message = msg if (Error.captureStackTrace) { Error.captureStackTrace(this, CoercionError); } } } module.exports = class Node { /** * Node constructor. * * @api public */ constructor() { this.lineno = nodes.lineno || 1; this.column = nodes.column || 1; this.filename = nodes.filename; } /** * Return this node. * * @return {Node} * @api public */ get first() { return this; } /** * Return hash. * * @return {String} * @api public */ get hash() { return this.val; } /** * Return node name. * * @return {String} * @api public */ get nodeName() { return this.constructor.name.toLowerCase(); } /** * Return this node. * * @return {Node} * @api public */ clone() { return this; } /** * Return a JSON representation of this node. * * @return {Object} * @api public */ toJSON() { return { lineno: this.lineno, column: this.column, filename: this.filename }; } /** * Nodes by default evaluate to themselves. * * @return {Node} * @api public */ eval() { return new Evaluator(this).evaluate(); } /** * Return true. * * @return {Boolean} * @api public */ toBoolean() { return nodes.true; } /** * Return the expression, or wrap this node in an expression. * * @return {Expression} * @api public */ toExpression() { if ('expression' == this.nodeName) return this; var expr = new nodes.Expression; expr.push(this); return expr; } /** * Return false if `op` is generally not coerced. * * @param {String} op * @return {Boolean} * @api private */ shouldCoerce(op) { switch (op) { case 'is a': case 'in': case '||': case '&&': return false; default: return true; } } /** * Operate on `right` with the given `op`. * * @param {String} op * @param {Node} right * @return {Node} * @api public */ operate(op, right) { switch (op) { case 'is a': if ('string' == right.first.nodeName) { return new nodes.Boolean(this.nodeName == right.val); } else { throw new Error('"is a" expects a string, got ' + right.toString()); } case '==': return new nodes.Boolean(this.hash == right.hash); case '!=': return new nodes.Boolean(this.hash != right.hash); case '>=': return new nodes.Boolean(this.hash >= right.hash); case '<=': return new nodes.Boolean(this.hash <= right.hash); case '>': return new nodes.Boolean(this.hash > right.hash); case '<': return new nodes.Boolean(this.hash < right.hash); case '||': return this.toBoolean().isTrue ? this : right; case 'in': var vals = utils.unwrap(right).nodes , len = vals && vals.length , hash = this.hash; if (!vals) throw new Error('"in" given invalid right-hand operand, expecting an expression'); // 'prop' in obj if (1 == len && 'object' == vals[0].nodeName) { return new nodes.Boolean(vals[0].has(this.hash)); } for (var i = 0; i < len; ++i) { if (hash == vals[i].hash) { return nodes.true; } } return nodes.false; case '&&': var a = this.toBoolean() , b = right.toBoolean(); return a.isTrue && b.isTrue ? right : a.isFalse ? this : right; default: if ('[]' == op) { var msg = 'cannot perform ' + this + '[' + right + ']'; } else { var msg = 'cannot perform' + ' ' + this + ' ' + op + ' ' + right; } throw new Error(msg); } } /** * Default coercion throws. * * @param {Node} other * @return {Node} * @api public */ coerce(other) { if (other.nodeName == this.nodeName) return other; throw new CoercionError('cannot coerce ' + other + ' to ' + this.nodeName); } };