import Binding from "../binding"; import * as t from "babel-types"; let renameVisitor = { ReferencedIdentifier({ node }, state) { if (node.name === state.oldName) { node.name = state.newName; } }, Scope(path, state) { if (!path.scope.bindingIdentifierEquals(state.oldName, state.binding.identifier)) { path.skip(); } }, "AssignmentExpression|Declaration"(path, state) { let ids = path.getBindingIdentifiers(); for (let name in ids) { if (name === state.oldName) ids[name].name = state.newName; } } }; export default class Renamer { constructor(binding: Binding, oldName: string, newName: string) { this.newName = newName; this.oldName = oldName; this.binding = binding; } oldName: string; newName: string; binding: Binding; maybeConvertFromExportDeclaration(parentDeclar) { let exportDeclar = parentDeclar.parentPath.isExportDeclaration() && parentDeclar.parentPath; if (!exportDeclar) return; // build specifiers that point back to this export declaration let isDefault = exportDeclar.isExportDefaultDeclaration(); let bindingIdentifiers = parentDeclar.getOuterBindingIdentifiers(); let specifiers = []; for (let name in bindingIdentifiers) { let localName = name === this.oldName ? this.newName : name; let exportedName = isDefault ? "default" : name; specifiers.push(t.exportSpecifier(t.identifier(localName), t.identifier(exportedName))); } let aliasDeclar = t.exportNamedDeclaration(null, specifiers); // hoist to the top if it's a function if (parentDeclar.isFunctionDeclaration()) { aliasDeclar._blockHoist = 3; } exportDeclar.insertAfter(aliasDeclar); exportDeclar.replaceWith(parentDeclar.node); } maybeConvertFromClassFunctionDeclaration(path) { return; // TODO // retain the `name` of a class/function declaration if (!path.isFunctionDeclaration() && !path.isClassDeclaration()) return; if (this.binding.kind !== "hoisted") return; path.node.id = t.identifier(this.oldName); path.node._blockHoist = 3; path.replaceWith(t.variableDeclaration("let", [ t.variableDeclarator(t.identifier(this.newName), t.toExpression(path.node)) ])); } maybeConvertFromClassFunctionExpression(path) { return; // TODO // retain the `name` of a class/function expression if (!path.isFunctionExpression() && !path.isClassExpression()) return; if (this.binding.kind !== "local") return; path.node.id = t.identifier(this.oldName); this.binding.scope.parent.push({ id: t.identifier(this.newName) }); path.replaceWith(t.assignmentExpression("=", t.identifier(this.newName), path.node)); } rename(block?) { let { binding, oldName, newName } = this; let { scope, path } = binding; let parentDeclar = path.find((path) => path.isDeclaration() || path.isFunctionExpression()); if (parentDeclar) { this.maybeConvertFromExportDeclaration(parentDeclar); } scope.traverse(block || scope.block, renameVisitor, this); if (!block) { scope.removeOwnBinding(oldName); scope.bindings[newName] = binding; this.binding.identifier.name = newName; } if (binding.type === "hoisted") { // https://github.com/babel/babel/issues/2435 // todo: hoist and convert function to a let } if (parentDeclar) { this.maybeConvertFromClassFunctionDeclaration(parentDeclar); this.maybeConvertFromClassFunctionExpression(parentDeclar); } } }