Files
babel/packages/babel-helper-simple-access/src/index.ts
Huáng Jùnliàng 614b486780 Use named imports for babel types (#13685)
* migrate to named babel types imports

* perf: transform babel types import to destructuring

* fix merge errors

* apply plugin to itself
2021-08-18 10:28:40 -04:00

140 lines
3.5 KiB
TypeScript

import {
LOGICAL_OPERATORS,
assignmentExpression,
binaryExpression,
cloneNode,
identifier,
logicalExpression,
numericLiteral,
sequenceExpression,
unaryExpression,
} from "@babel/types";
import type { NodePath } from "@babel/traverse";
export default function simplifyAccess(path: NodePath, bindingNames) {
path.traverse(simpleAssignmentVisitor, {
scope: path.scope,
bindingNames,
seen: new WeakSet(),
});
}
const simpleAssignmentVisitor = {
UpdateExpression: {
exit(path) {
const { scope, bindingNames } = this;
const arg = path.get("argument");
if (!arg.isIdentifier()) return;
const localName = arg.node.name;
if (!bindingNames.has(localName)) return;
// redeclared in this scope
if (scope.getBinding(localName) !== path.scope.getBinding(localName)) {
return;
}
if (
path.parentPath.isExpressionStatement() &&
!path.isCompletionRecord()
) {
// ++i => (i += 1);
const operator = path.node.operator == "++" ? "+=" : "-=";
path.replaceWith(
assignmentExpression(operator, arg.node, numericLiteral(1)),
);
} else if (path.node.prefix) {
// ++i => (i = (+i) + 1);
path.replaceWith(
assignmentExpression(
"=",
identifier(localName),
binaryExpression(
path.node.operator[0],
unaryExpression("+", arg.node),
numericLiteral(1),
),
),
);
} else {
const old = path.scope.generateUidIdentifierBasedOnNode(
arg.node,
"old",
);
const varName = old.name;
path.scope.push({ id: old });
const binary = binaryExpression(
path.node.operator[0],
identifier(varName),
numericLiteral(1),
);
// i++ => (_old = (+i), i = _old + 1, _old)
path.replaceWith(
sequenceExpression([
assignmentExpression(
"=",
identifier(varName),
unaryExpression("+", arg.node),
),
assignmentExpression("=", cloneNode(arg.node), binary),
identifier(varName),
]),
);
}
},
},
AssignmentExpression: {
exit(path) {
const { scope, seen, bindingNames } = this;
if (path.node.operator === "=") return;
if (seen.has(path.node)) return;
seen.add(path.node);
const left = path.get("left");
if (!left.isIdentifier()) return;
// Simple update-assign foo += 1;
// => exports.foo = (foo += 1);
const localName = left.node.name;
if (!bindingNames.has(localName)) return;
// redeclared in this scope
if (scope.getBinding(localName) !== path.scope.getBinding(localName)) {
return;
}
const operator = path.node.operator.slice(0, -1);
if (LOGICAL_OPERATORS.includes(operator)) {
// &&, ||, ??
// (foo &&= bar) => (foo && foo = bar)
path.replaceWith(
logicalExpression(
operator,
path.node.left,
assignmentExpression(
"=",
cloneNode(path.node.left),
path.node.right,
),
),
);
} else {
// (foo += bar) => (foo = foo + bar)
path.node.right = binaryExpression(
operator,
cloneNode(path.node.left),
path.node.right,
);
path.node.operator = "=";
}
},
},
};