* migrate to named babel types imports * perf: transform babel types import to destructuring * fix merge errors * apply plugin to itself
140 lines
3.5 KiB
TypeScript
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 = "=";
|
|
}
|
|
},
|
|
},
|
|
};
|