2015-03-23 10:55:49 -07:00
|
|
|
/**
|
|
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
|
|
* All rights reserved.
|
|
|
|
*
|
|
|
|
* This source code is licensed under the BSD-style license found in the
|
|
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
|
|
*/
|
2015-03-10 13:55:54 -07:00
|
|
|
|
|
|
|
/*global exports:true*/
|
|
|
|
/*jslint node:true*/
|
2015-03-26 11:14:54 -07:00
|
|
|
'use strict';
|
2015-03-10 13:55:54 -07:00
|
|
|
|
2016-09-14 14:16:35 -07:00
|
|
|
var Syntax = require('./syntax');
|
2015-03-10 13:55:54 -07:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Executes visitor on the object and its children (recursively).
|
|
|
|
* While traversing the tree, a scope chain is built and passed to the visitor.
|
|
|
|
*
|
|
|
|
* If the visitor returns false, the object's children are not traversed.
|
|
|
|
*
|
|
|
|
* @param {object} object
|
|
|
|
* @param {function} visitor
|
|
|
|
* @param {?array} scopeChain
|
|
|
|
*/
|
|
|
|
function traverse(object, visitor, scopeChain) {
|
|
|
|
scopeChain = scopeChain || [{}];
|
|
|
|
|
|
|
|
var scope = scopeChain[0];
|
|
|
|
|
|
|
|
switch (object.type) {
|
|
|
|
case Syntax.VariableDeclaration:
|
|
|
|
object.declarations.forEach(function(decl) {
|
|
|
|
scope[decl.id.name] = decl.init;
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case Syntax.ClassDeclaration:
|
|
|
|
scope[object.id.name] = object;
|
|
|
|
break;
|
|
|
|
case Syntax.FunctionDeclaration:
|
|
|
|
// A function declaration creates a symbol in the current scope
|
|
|
|
scope[object.id.name] = object;
|
|
|
|
/* falls through */
|
|
|
|
case Syntax.FunctionExpression:
|
|
|
|
case Syntax.Program:
|
|
|
|
scopeChain = [{}].concat(scopeChain);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (object.type === Syntax.FunctionExpression ||
|
|
|
|
object.type === Syntax.FunctionDeclaration) {
|
|
|
|
// add parameters to the new scope
|
|
|
|
object.params.forEach(function(param) {
|
|
|
|
// since the value of the parameters are unknown during parsing time
|
|
|
|
// we set the value to `undefined`.
|
|
|
|
scopeChain[0][param.name] = undefined;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
if (object.type) {
|
|
|
|
if (visitor.call(null, object, scopeChain) === false) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (var key in object) {
|
|
|
|
if (object.hasOwnProperty(key)) {
|
|
|
|
var child = object[key];
|
|
|
|
if (typeof child === 'object' && child !== null) {
|
|
|
|
traverse(child, visitor, scopeChain);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Executes visitor on the object and its children, but only traverses into
|
|
|
|
* children which can be statically analyzed and don't depend on runtime
|
|
|
|
* information.
|
|
|
|
*
|
|
|
|
* @param {object} object
|
|
|
|
* @param {function} visitor
|
|
|
|
* @param {?array} scopeChain
|
|
|
|
*/
|
|
|
|
function traverseFlat(object, visitor, scopeChain) {
|
|
|
|
traverse(object, function(node, scopeChain) {
|
|
|
|
switch (node.type) {
|
|
|
|
case Syntax.FunctionDeclaration:
|
|
|
|
case Syntax.FunctionExpression:
|
|
|
|
case Syntax.IfStatement:
|
|
|
|
case Syntax.WithStatement:
|
|
|
|
case Syntax.SwitchStatement:
|
|
|
|
case Syntax.TryStatement:
|
|
|
|
case Syntax.WhileStatement:
|
|
|
|
case Syntax.DoWhileStatement:
|
|
|
|
case Syntax.ForStatement:
|
|
|
|
case Syntax.ForInStatement:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return visitor(node, scopeChain);
|
|
|
|
}, scopeChain);
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = traverseFlat;
|