|  | /* vim: set sw=4 sts=4 : */ | 
|  | (function () { | 
|  |  | 
|  | var estraverse = require('estraverse'); | 
|  | var parser = require('./parser'); | 
|  |  | 
|  | var isArray = Array.isArray || function isArray(array) { | 
|  | return {}.toString.call(array) === '[object Array]'; | 
|  | }; | 
|  |  | 
|  | var LEFT_SIDE = {}; | 
|  | var RIGHT_SIDE = {}; | 
|  |  | 
|  | function esqueryModule() { | 
|  |  | 
|  | /** | 
|  | * Get the value of a property which may be multiple levels down in the object. | 
|  | */ | 
|  | function getPath(obj, key) { | 
|  | var i, keys = key.split("."); | 
|  | for (i = 0; i < keys.length; i++) { | 
|  | if (obj == null) { return obj; } | 
|  | obj = obj[keys[i]]; | 
|  | } | 
|  | return obj; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Determine whether `node` can be reached by following `path`, starting at `ancestor`. | 
|  | */ | 
|  | function inPath(node, ancestor, path) { | 
|  | var field, remainingPath, i; | 
|  | if (path.length === 0) { return node === ancestor; } | 
|  | if (ancestor == null) { return false; } | 
|  | field = ancestor[path[0]]; | 
|  | remainingPath = path.slice(1); | 
|  | if (isArray(field)) { | 
|  | for (i = 0, l = field.length; i < l; ++i) { | 
|  | if (inPath(node, field[i], remainingPath)) { return true; } | 
|  | } | 
|  | return false; | 
|  | } else { | 
|  | return inPath(node, field, remainingPath); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Given a `node` and its ancestors, determine if `node` is matched by `selector`. | 
|  | */ | 
|  | function matches(node, selector, ancestry) { | 
|  | var path, ancestor, i, l, p; | 
|  | if (!selector) { return true; } | 
|  | if (!node) { return false; } | 
|  | if (!ancestry) { ancestry = []; } | 
|  |  | 
|  | switch(selector.type) { | 
|  | case 'wildcard': | 
|  | return true; | 
|  |  | 
|  | case 'identifier': | 
|  | return selector.value.toLowerCase() === node.type.toLowerCase(); | 
|  |  | 
|  | case 'field': | 
|  | path = selector.name.split('.'); | 
|  | ancestor = ancestry[path.length - 1]; | 
|  | return inPath(node, ancestor, path); | 
|  |  | 
|  | case 'matches': | 
|  | for (i = 0, l = selector.selectors.length; i < l; ++i) { | 
|  | if (matches(node, selector.selectors[i], ancestry)) { return true; } | 
|  | } | 
|  | return false; | 
|  |  | 
|  | case 'compound': | 
|  | for (i = 0, l = selector.selectors.length; i < l; ++i) { | 
|  | if (!matches(node, selector.selectors[i], ancestry)) { return false; } | 
|  | } | 
|  | return true; | 
|  |  | 
|  | case 'not': | 
|  | for (i = 0, l = selector.selectors.length; i < l; ++i) { | 
|  | if (matches(node, selector.selectors[i], ancestry)) { return false; } | 
|  | } | 
|  | return true; | 
|  |  | 
|  | case 'has': | 
|  | var a, collector = []; | 
|  | for (i = 0, l = selector.selectors.length; i < l; ++i) { | 
|  | a = []; | 
|  | estraverse.traverse(node, { | 
|  | enter: function (node, parent) { | 
|  | if (parent != null) { a.unshift(parent); } | 
|  | if (matches(node, selector.selectors[i], a)) { | 
|  | collector.push(node); | 
|  | } | 
|  | }, | 
|  | leave: function () { a.shift(); } | 
|  | }); | 
|  | } | 
|  | return collector.length !== 0; | 
|  |  | 
|  | case 'child': | 
|  | if (matches(node, selector.right, ancestry)) { | 
|  | return matches(ancestry[0], selector.left, ancestry.slice(1)); | 
|  | } | 
|  | return false; | 
|  |  | 
|  | case 'descendant': | 
|  | if (matches(node, selector.right, ancestry)) { | 
|  | for (i = 0, l = ancestry.length; i < l; ++i) { | 
|  | if (matches(ancestry[i], selector.left, ancestry.slice(i + 1))) { | 
|  | return true; | 
|  | } | 
|  | } | 
|  | } | 
|  | return false; | 
|  |  | 
|  | case 'attribute': | 
|  | p = getPath(node, selector.name); | 
|  | switch (selector.operator) { | 
|  | case null: | 
|  | case void 0: | 
|  | return p != null; | 
|  | case '=': | 
|  | switch (selector.value.type) { | 
|  | case 'regexp': return typeof p === 'string' && selector.value.value.test(p); | 
|  | case 'literal': return '' + selector.value.value === '' + p; | 
|  | case 'type': return selector.value.value === typeof p; | 
|  | } | 
|  | case '!=': | 
|  | switch (selector.value.type) { | 
|  | case 'regexp': return !selector.value.value.test(p); | 
|  | case 'literal': return '' + selector.value.value !== '' + p; | 
|  | case 'type': return selector.value.value !== typeof p; | 
|  | } | 
|  | case '<=': return p <= selector.value.value; | 
|  | case '<': return p < selector.value.value; | 
|  | case '>': return p > selector.value.value; | 
|  | case '>=': return p >= selector.value.value; | 
|  | } | 
|  |  | 
|  | case 'sibling': | 
|  | return matches(node, selector.right, ancestry) && | 
|  | sibling(node, selector.left, ancestry, LEFT_SIDE) || | 
|  | selector.left.subject && | 
|  | matches(node, selector.left, ancestry) && | 
|  | sibling(node, selector.right, ancestry, RIGHT_SIDE); | 
|  |  | 
|  | case 'adjacent': | 
|  | return matches(node, selector.right, ancestry) && | 
|  | adjacent(node, selector.left, ancestry, LEFT_SIDE) || | 
|  | selector.right.subject && | 
|  | matches(node, selector.left, ancestry) && | 
|  | adjacent(node, selector.right, ancestry, RIGHT_SIDE); | 
|  |  | 
|  | case 'nth-child': | 
|  | return matches(node, selector.right, ancestry) && | 
|  | nthChild(node, ancestry, function (length) { | 
|  | return selector.index.value - 1; | 
|  | }); | 
|  |  | 
|  | case 'nth-last-child': | 
|  | return matches(node, selector.right, ancestry) && | 
|  | nthChild(node, ancestry, function (length) { | 
|  | return length - selector.index.value; | 
|  | }); | 
|  |  | 
|  | case 'class': | 
|  | if(!node.type) return false; | 
|  | switch(selector.name.toLowerCase()){ | 
|  | case 'statement': | 
|  | if(node.type.slice(-9) === 'Statement') return true; | 
|  | // fallthrough: interface Declaration <: Statement { } | 
|  | case 'declaration': | 
|  | return node.type.slice(-11) === 'Declaration'; | 
|  | case 'pattern': | 
|  | if(node.type.slice(-7) === 'Pattern') return true; | 
|  | // fallthrough: interface Expression <: Node, Pattern { } | 
|  | case 'expression': | 
|  | return node.type.slice(-10) === 'Expression' || | 
|  | node.type.slice(-7) === 'Literal' || | 
|  | ( | 
|  | node.type === 'Identifier' && | 
|  | (ancestry.length === 0 || ancestry[0].type !== 'MetaProperty') | 
|  | ) || | 
|  | node.type === 'MetaProperty'; | 
|  | case 'function': | 
|  | return node.type.slice(0, 8) === 'Function' || | 
|  | node.type === 'ArrowFunctionExpression'; | 
|  | } | 
|  | throw new Error('Unknown class name: ' + selector.name); | 
|  | } | 
|  |  | 
|  | throw new Error('Unknown selector type: ' + selector.type); | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Determines if the given node has a sibling that matches the given selector. | 
|  | */ | 
|  | function sibling(node, selector, ancestry, side) { | 
|  | var parent = ancestry[0], listProp, startIndex, keys, i, l, k, lowerBound, upperBound; | 
|  | if (!parent) { return false; } | 
|  | keys = estraverse.VisitorKeys[parent.type]; | 
|  | for (i = 0, l = keys.length; i < l; ++i) { | 
|  | listProp = parent[keys[i]]; | 
|  | if (isArray(listProp)) { | 
|  | startIndex = listProp.indexOf(node); | 
|  | if (startIndex < 0) { continue; } | 
|  | if (side === LEFT_SIDE) { | 
|  | lowerBound = 0; | 
|  | upperBound = startIndex; | 
|  | } else { | 
|  | lowerBound = startIndex + 1; | 
|  | upperBound = listProp.length; | 
|  | } | 
|  | for (k = lowerBound; k < upperBound; ++k) { | 
|  | if (matches(listProp[k], selector, ancestry)) { | 
|  | return true; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Determines if the given node has an asjacent sibling that matches the given selector. | 
|  | */ | 
|  | function adjacent(node, selector, ancestry, side) { | 
|  | var parent = ancestry[0], listProp, keys, i, l, idx; | 
|  | if (!parent) { return false; } | 
|  | keys = estraverse.VisitorKeys[parent.type]; | 
|  | for (i = 0, l = keys.length; i < l; ++i) { | 
|  | listProp = parent[keys[i]]; | 
|  | if (isArray(listProp)) { | 
|  | idx = listProp.indexOf(node); | 
|  | if (idx < 0) { continue; } | 
|  | if (side === LEFT_SIDE && idx > 0 && matches(listProp[idx - 1], selector, ancestry)) { | 
|  | return true; | 
|  | } | 
|  | if (side === RIGHT_SIDE && idx < listProp.length - 1 && matches(listProp[idx + 1], selector, ancestry)) { | 
|  | return true; | 
|  | } | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Determines if the given node is the nth child, determined by idxFn, which is given the containing list's length. | 
|  | */ | 
|  | function nthChild(node, ancestry, idxFn) { | 
|  | var parent = ancestry[0], listProp, keys, i, l, idx; | 
|  | if (!parent) { return false; } | 
|  | keys = estraverse.VisitorKeys[parent.type]; | 
|  | for (i = 0, l = keys.length; i < l; ++i) { | 
|  | listProp = parent[keys[i]]; | 
|  | if (isArray(listProp)) { | 
|  | idx = listProp.indexOf(node); | 
|  | if (idx >= 0 && idx === idxFn(listProp.length)) { return true; } | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * For each selector node marked as a subject, find the portion of the selector that the subject must match. | 
|  | */ | 
|  | function subjects(selector, ancestor) { | 
|  | var results, p; | 
|  | if (selector == null || typeof selector != 'object') { return []; } | 
|  | if (ancestor == null) { ancestor = selector; } | 
|  | results = selector.subject ? [ancestor] : []; | 
|  | for(p in selector) { | 
|  | if(!{}.hasOwnProperty.call(selector, p)) { continue; } | 
|  | [].push.apply(results, subjects(selector[p], p === 'left' ? selector[p] : ancestor)); | 
|  | } | 
|  | return results; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * From a JS AST and a selector AST, collect all JS AST nodes that match the selector. | 
|  | */ | 
|  | function match(ast, selector) { | 
|  | var ancestry = [], results = [], altSubjects, i, l, k, m; | 
|  | if (!selector) { return results; } | 
|  | altSubjects = subjects(selector); | 
|  | estraverse.traverse(ast, { | 
|  | enter: function (node, parent) { | 
|  | if (parent != null) { ancestry.unshift(parent); } | 
|  | if (matches(node, selector, ancestry)) { | 
|  | if (altSubjects.length) { | 
|  | for (i = 0, l = altSubjects.length; i < l; ++i) { | 
|  | if (matches(node, altSubjects[i], ancestry)) { results.push(node); } | 
|  | for (k = 0, m = ancestry.length; k < m; ++k) { | 
|  | if (matches(ancestry[k], altSubjects[i], ancestry.slice(k + 1))) { | 
|  | results.push(ancestry[k]); | 
|  | } | 
|  | } | 
|  | } | 
|  | } else { | 
|  | results.push(node); | 
|  | } | 
|  | } | 
|  | }, | 
|  | leave: function () { ancestry.shift(); } | 
|  | }); | 
|  | return results; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Parse a selector string and return its AST. | 
|  | */ | 
|  | function parse(selector) { | 
|  | return parser.parse(selector); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Query the code AST using the selector string. | 
|  | */ | 
|  | function query(ast, selector) { | 
|  | return match(ast, parse(selector)); | 
|  | } | 
|  |  | 
|  | query.parse = parse; | 
|  | query.match = match; | 
|  | query.matches = matches; | 
|  | return query.query = query; | 
|  | } | 
|  |  | 
|  |  | 
|  | if (typeof define === "function" && define.amd) { | 
|  | define(esqueryModule); | 
|  | } else if (typeof module !== 'undefined' && module.exports) { | 
|  | module.exports = esqueryModule(); | 
|  | } else { | 
|  | this.esquery = esqueryModule(); | 
|  | } | 
|  |  | 
|  | })(); |