| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492 | 'use strict';var isObject = require('isobject');var define = require('define-property');var utils = require('snapdragon-util');var ownNames;/** * Create a new AST `Node` with the given `val` and `type`. * * ```js * var node = new Node('*', 'Star'); * var node = new Node({type: 'star', val: '*'}); * ``` * @name Node * @param {String|Object} `val` Pass a matched substring, or an object to merge onto the node. * @param {String} `type` The node type to use when `val` is a string. * @return {Object} node instance * @api public */function Node(val, type, parent) {  if (typeof type !== 'string') {    parent = type;    type = null;  }  define(this, 'parent', parent);  define(this, 'isNode', true);  define(this, 'expect', null);  if (typeof type !== 'string' && isObject(val)) {    lazyKeys();    var keys = Object.keys(val);    for (var i = 0; i < keys.length; i++) {      var key = keys[i];      if (ownNames.indexOf(key) === -1) {        this[key] = val[key];      }    }  } else {    this.type = type;    this.val = val;  }}/** * Returns true if the given value is a node. * * ```js * var Node = require('snapdragon-node'); * var node = new Node({type: 'foo'}); * console.log(Node.isNode(node)); //=> true * console.log(Node.isNode({})); //=> false * ``` * @param {Object} `node` * @returns {Boolean} * @api public */Node.isNode = function(node) {  return utils.isNode(node);};/** * Define a non-enumberable property on the node instance. * Useful for adding properties that shouldn't be extended * or visible during debugging. * * ```js * var node = new Node(); * node.define('foo', 'something non-enumerable'); * ``` * @param {String} `name` * @param {any} `val` * @return {Object} returns the node instance * @api public */Node.prototype.define = function(name, val) {  define(this, name, val);  return this;};/** * Returns true if `node.val` is an empty string, or `node.nodes` does * not contain any non-empty text nodes. * * ```js * var node = new Node({type: 'text'}); * node.isEmpty(); //=> true * node.val = 'foo'; * node.isEmpty(); //=> false * ``` * @param {Function} `fn` (optional) Filter function that is called on `node` and/or child nodes. `isEmpty` will return false immediately when the filter function returns false on any nodes. * @return {Boolean} * @api public */Node.prototype.isEmpty = function(fn) {  return utils.isEmpty(this, fn);};/** * Given node `foo` and node `bar`, push node `bar` onto `foo.nodes`, and * set `foo` as `bar.parent`. * * ```js * var foo = new Node({type: 'foo'}); * var bar = new Node({type: 'bar'}); * foo.push(bar); * ``` * @param {Object} `node` * @return {Number} Returns the length of `node.nodes` * @api public */Node.prototype.push = function(node) {  assert(Node.isNode(node), 'expected node to be an instance of Node');  define(node, 'parent', this);  this.nodes = this.nodes || [];  return this.nodes.push(node);};/** * Given node `foo` and node `bar`, unshift node `bar` onto `foo.nodes`, and * set `foo` as `bar.parent`. * * ```js * var foo = new Node({type: 'foo'}); * var bar = new Node({type: 'bar'}); * foo.unshift(bar); * ``` * @param {Object} `node` * @return {Number} Returns the length of `node.nodes` * @api public */Node.prototype.unshift = function(node) {  assert(Node.isNode(node), 'expected node to be an instance of Node');  define(node, 'parent', this);  this.nodes = this.nodes || [];  return this.nodes.unshift(node);};/** * Pop a node from `node.nodes`. * * ```js * var node = new Node({type: 'foo'}); * node.push(new Node({type: 'a'})); * node.push(new Node({type: 'b'})); * node.push(new Node({type: 'c'})); * node.push(new Node({type: 'd'})); * console.log(node.nodes.length); * //=> 4 * node.pop(); * console.log(node.nodes.length); * //=> 3 * ``` * @return {Number} Returns the popped `node` * @api public */Node.prototype.pop = function() {  return this.nodes && this.nodes.pop();};/** * Shift a node from `node.nodes`. * * ```js * var node = new Node({type: 'foo'}); * node.push(new Node({type: 'a'})); * node.push(new Node({type: 'b'})); * node.push(new Node({type: 'c'})); * node.push(new Node({type: 'd'})); * console.log(node.nodes.length); * //=> 4 * node.shift(); * console.log(node.nodes.length); * //=> 3 * ``` * @return {Object} Returns the shifted `node` * @api public */Node.prototype.shift = function() {  return this.nodes && this.nodes.shift();};/** * Remove `node` from `node.nodes`. * * ```js * node.remove(childNode); * ``` * @param {Object} `node` * @return {Object} Returns the removed node. * @api public */Node.prototype.remove = function(node) {  assert(Node.isNode(node), 'expected node to be an instance of Node');  this.nodes = this.nodes || [];  var idx = node.index;  if (idx !== -1) {    node.index = -1;    return this.nodes.splice(idx, 1);  }  return null;};/** * Get the first child node from `node.nodes` that matches the given `type`. * If `type` is a number, the child node at that index is returned. * * ```js * var child = node.find(1); //<= index of the node to get * var child = node.find('foo'); //<= node.type of a child node * var child = node.find(/^(foo|bar)$/); //<= regex to match node.type * var child = node.find(['foo', 'bar']); //<= array of node.type(s) * ``` * @param {String} `type` * @return {Object} Returns a child node or undefined. * @api public */Node.prototype.find = function(type) {  return utils.findNode(this.nodes, type);};/** * Return true if the node is the given `type`. * * ```js * var node = new Node({type: 'bar'}); * cosole.log(node.isType('foo'));          // false * cosole.log(node.isType(/^(foo|bar)$/));  // true * cosole.log(node.isType(['foo', 'bar'])); // true * ``` * @param {String} `type` * @return {Boolean} * @api public */Node.prototype.isType = function(type) {  return utils.isType(this, type);};/** * Return true if the `node.nodes` has the given `type`. * * ```js * var foo = new Node({type: 'foo'}); * var bar = new Node({type: 'bar'}); * foo.push(bar); * * cosole.log(foo.hasType('qux'));          // false * cosole.log(foo.hasType(/^(qux|bar)$/));  // true * cosole.log(foo.hasType(['qux', 'bar'])); // true * ``` * @param {String} `type` * @return {Boolean} * @api public */Node.prototype.hasType = function(type) {  return utils.hasType(this, type);};/** * Get the siblings array, or `null` if it doesn't exist. * * ```js * var foo = new Node({type: 'foo'}); * var bar = new Node({type: 'bar'}); * var baz = new Node({type: 'baz'}); * foo.push(bar); * foo.push(baz); * * console.log(bar.siblings.length) // 2 * console.log(baz.siblings.length) // 2 * ``` * @return {Array} * @api public */Object.defineProperty(Node.prototype, 'siblings', {  set: function() {    throw new Error('node.siblings is a getter and cannot be defined');  },  get: function() {    return this.parent ? this.parent.nodes : null;  }});/** * Get the node's current index from `node.parent.nodes`. * This should always be correct, even when the parent adds nodes. * * ```js * var foo = new Node({type: 'foo'}); * var bar = new Node({type: 'bar'}); * var baz = new Node({type: 'baz'}); * var qux = new Node({type: 'qux'}); * foo.push(bar); * foo.push(baz); * foo.unshift(qux); * * console.log(bar.index) // 1 * console.log(baz.index) // 2 * console.log(qux.index) // 0 * ``` * @return {Number} * @api public */Object.defineProperty(Node.prototype, 'index', {  set: function(index) {    define(this, 'idx', index);  },  get: function() {    if (!Array.isArray(this.siblings)) {      return -1;    }    var tok = this.idx !== -1 ? this.siblings[this.idx] : null;    if (tok !== this) {      this.idx = this.siblings.indexOf(this);    }    return this.idx;  }});/** * Get the previous node from the siblings array or `null`. * * ```js * var foo = new Node({type: 'foo'}); * var bar = new Node({type: 'bar'}); * var baz = new Node({type: 'baz'}); * foo.push(bar); * foo.push(baz); * * console.log(baz.prev.type) // 'bar' * ``` * @return {Object} * @api public */Object.defineProperty(Node.prototype, 'prev', {  set: function() {    throw new Error('node.prev is a getter and cannot be defined');  },  get: function() {    if (Array.isArray(this.siblings)) {      return this.siblings[this.index - 1] || this.parent.prev;    }    return null;  }});/** * Get the siblings array, or `null` if it doesn't exist. * * ```js * var foo = new Node({type: 'foo'}); * var bar = new Node({type: 'bar'}); * var baz = new Node({type: 'baz'}); * foo.push(bar); * foo.push(baz); * * console.log(bar.siblings.length) // 2 * console.log(baz.siblings.length) // 2 * ``` * @return {Object} * @api public */Object.defineProperty(Node.prototype, 'next', {  set: function() {    throw new Error('node.next is a getter and cannot be defined');  },  get: function() {    if (Array.isArray(this.siblings)) {      return this.siblings[this.index + 1] || this.parent.next;    }    return null;  }});/** * Get the first node from `node.nodes`. * * ```js * var foo = new Node({type: 'foo'}); * var bar = new Node({type: 'bar'}); * var baz = new Node({type: 'baz'}); * var qux = new Node({type: 'qux'}); * foo.push(bar); * foo.push(baz); * foo.push(qux); * * console.log(foo.first.type) // 'bar' * ``` * @return {Object} The first node, or undefiend * @api public */Object.defineProperty(Node.prototype, 'first', {  get: function() {    return this.nodes ? this.nodes[0] : null;  }});/** * Get the last node from `node.nodes`. * * ```js * var foo = new Node({type: 'foo'}); * var bar = new Node({type: 'bar'}); * var baz = new Node({type: 'baz'}); * var qux = new Node({type: 'qux'}); * foo.push(bar); * foo.push(baz); * foo.push(qux); * * console.log(foo.last.type) // 'qux' * ``` * @return {Object} The last node, or undefiend * @api public */Object.defineProperty(Node.prototype, 'last', {  get: function() {    return this.nodes ? utils.last(this.nodes) : null;  }});/** * Get the last node from `node.nodes`. * * ```js * var foo = new Node({type: 'foo'}); * var bar = new Node({type: 'bar'}); * var baz = new Node({type: 'baz'}); * var qux = new Node({type: 'qux'}); * foo.push(bar); * foo.push(baz); * foo.push(qux); * * console.log(foo.last.type) // 'qux' * ``` * @return {Object} The last node, or undefiend * @api public */Object.defineProperty(Node.prototype, 'scope', {  get: function() {    if (this.isScope !== true) {      return this.parent ? this.parent.scope : this;    }    return this;  }});/** * Get own property names from Node prototype, but only the * first time `Node` is instantiated */function lazyKeys() {  if (!ownNames) {    ownNames = Object.getOwnPropertyNames(Node.prototype);  }}/** * Simplified assertion. Throws an error is `val` is falsey. */function assert(val, message) {  if (!val) throw new Error(message);}/** * Expose `Node` */exports = module.exports = Node;
 |