251 lines
7.1 KiB
JavaScript
251 lines
7.1 KiB
JavaScript
import nameFunction from "@babel/helper-function-name";
|
|
import syntaxClassProperties from "@babel/plugin-syntax-class-properties";
|
|
import { template, types as t } from "@babel/core";
|
|
|
|
export default function(api, options) {
|
|
const { loose } = options;
|
|
|
|
const findBareSupers = {
|
|
Super(path) {
|
|
if (path.parentPath.isCallExpression({ callee: path.node })) {
|
|
this.push(path.parentPath);
|
|
}
|
|
},
|
|
};
|
|
|
|
const referenceVisitor = {
|
|
"TSTypeAnnotation|TypeAnnotation"(path) {
|
|
path.skip();
|
|
},
|
|
ReferencedIdentifier(path) {
|
|
if (this.scope.hasOwnBinding(path.node.name)) {
|
|
this.collision = true;
|
|
path.skip();
|
|
}
|
|
},
|
|
};
|
|
|
|
const ClassFieldDefinitionEvaluationTDZVisitor = {
|
|
Expression(path) {
|
|
if (path === this.shouldSkip) {
|
|
path.skip();
|
|
}
|
|
},
|
|
|
|
ReferencedIdentifier(path) {
|
|
if (this.classRef === path.scope.getBinding(path.node.name)) {
|
|
const classNameTDZError = this.file.addHelper("classNameTDZError");
|
|
const throwNode = t.callExpression(classNameTDZError, [
|
|
t.stringLiteral(path.node.name),
|
|
]);
|
|
|
|
path.replaceWith(t.sequenceExpression([throwNode, path.node]));
|
|
path.skip();
|
|
}
|
|
},
|
|
};
|
|
|
|
const buildClassPropertySpec = (ref, { key, value, computed }, scope) => {
|
|
return template.statement`
|
|
Object.defineProperty(REF, KEY, {
|
|
configurable: true,
|
|
enumerable: true,
|
|
writable: true,
|
|
value: VALUE
|
|
});
|
|
`({
|
|
REF: t.cloneNode(ref),
|
|
KEY: t.isIdentifier(key) && !computed ? t.stringLiteral(key.name) : key,
|
|
VALUE: value || scope.buildUndefinedNode(),
|
|
});
|
|
};
|
|
|
|
const buildClassPropertyLoose = (ref, { key, value, computed }, scope) => {
|
|
return template.statement`MEMBER = VALUE`({
|
|
MEMBER: t.memberExpression(
|
|
t.cloneNode(ref),
|
|
key,
|
|
computed || t.isLiteral(key),
|
|
),
|
|
VALUE: value || scope.buildUndefinedNode(),
|
|
});
|
|
};
|
|
|
|
const buildClassProperty = loose
|
|
? buildClassPropertyLoose
|
|
: buildClassPropertySpec;
|
|
|
|
return {
|
|
inherits: syntaxClassProperties,
|
|
|
|
visitor: {
|
|
Class(path) {
|
|
const isDerived = !!path.node.superClass;
|
|
let constructor;
|
|
const props = [];
|
|
const computedPaths = [];
|
|
const body = path.get("body");
|
|
|
|
for (const path of body.get("body")) {
|
|
if (path.node.computed) {
|
|
computedPaths.push(path);
|
|
}
|
|
|
|
if (path.isClassProperty()) {
|
|
props.push(path);
|
|
} else if (path.isClassMethod({ kind: "constructor" })) {
|
|
constructor = path;
|
|
}
|
|
}
|
|
|
|
if (!props.length) return;
|
|
|
|
let ref;
|
|
|
|
if (path.isClassExpression() || !path.node.id) {
|
|
nameFunction(path);
|
|
ref = path.scope.generateUidIdentifier("class");
|
|
} else {
|
|
// path.isClassDeclaration() && path.node.id
|
|
ref = path.node.id;
|
|
}
|
|
|
|
const computedNodes = [];
|
|
const staticNodes = [];
|
|
let instanceBody = [];
|
|
|
|
for (const computedPath of computedPaths) {
|
|
const computedNode = computedPath.node;
|
|
// Make sure computed property names are only evaluated once (upon class definition)
|
|
// and in the right order in combination with static properties
|
|
if (!computedPath.get("key").isConstantExpression()) {
|
|
computedPath.traverse(ClassFieldDefinitionEvaluationTDZVisitor, {
|
|
classRef: path.scope.getBinding(ref.name),
|
|
file: this.file,
|
|
shouldSkip: computedPath.get("value"),
|
|
});
|
|
const ident = path.scope.generateUidIdentifierBasedOnNode(
|
|
computedNode.key,
|
|
);
|
|
computedNodes.push(
|
|
t.variableDeclaration("var", [
|
|
t.variableDeclarator(ident, computedNode.key),
|
|
]),
|
|
);
|
|
computedNode.key = t.cloneNode(ident);
|
|
}
|
|
}
|
|
|
|
for (const prop of props) {
|
|
const propNode = prop.node;
|
|
if (propNode.decorators && propNode.decorators.length > 0) continue;
|
|
|
|
if (propNode.static) {
|
|
staticNodes.push(buildClassProperty(ref, propNode, path.scope));
|
|
} else {
|
|
instanceBody.push(
|
|
buildClassProperty(t.thisExpression(), propNode, path.scope),
|
|
);
|
|
}
|
|
}
|
|
|
|
const afterNodes = [...staticNodes];
|
|
|
|
if (instanceBody.length) {
|
|
if (!constructor) {
|
|
const newConstructor = t.classMethod(
|
|
"constructor",
|
|
t.identifier("constructor"),
|
|
[],
|
|
t.blockStatement([]),
|
|
);
|
|
if (isDerived) {
|
|
newConstructor.params = [t.restElement(t.identifier("args"))];
|
|
newConstructor.body.body.push(
|
|
t.returnStatement(
|
|
t.callExpression(t.super(), [
|
|
t.spreadElement(t.identifier("args")),
|
|
]),
|
|
),
|
|
);
|
|
}
|
|
[constructor] = body.unshiftContainer("body", newConstructor);
|
|
}
|
|
|
|
const collisionState = {
|
|
collision: false,
|
|
scope: constructor.scope,
|
|
};
|
|
|
|
for (const prop of props) {
|
|
prop.traverse(referenceVisitor, collisionState);
|
|
if (collisionState.collision) break;
|
|
}
|
|
|
|
if (collisionState.collision) {
|
|
const initialisePropsRef = path.scope.generateUidIdentifier(
|
|
"initialiseProps",
|
|
);
|
|
|
|
afterNodes.push(
|
|
t.variableDeclaration("var", [
|
|
t.variableDeclarator(
|
|
initialisePropsRef,
|
|
t.functionExpression(
|
|
null,
|
|
[],
|
|
t.blockStatement(instanceBody),
|
|
),
|
|
),
|
|
]),
|
|
);
|
|
|
|
instanceBody = [
|
|
t.expressionStatement(
|
|
t.callExpression(
|
|
t.memberExpression(
|
|
t.cloneNode(initialisePropsRef),
|
|
t.identifier("call"),
|
|
),
|
|
[t.thisExpression()],
|
|
),
|
|
),
|
|
];
|
|
}
|
|
|
|
//
|
|
|
|
if (isDerived) {
|
|
const bareSupers = [];
|
|
constructor.traverse(findBareSupers, bareSupers);
|
|
for (const bareSuper of bareSupers) {
|
|
bareSuper.insertAfter(instanceBody);
|
|
}
|
|
} else {
|
|
constructor.get("body").unshiftContainer("body", instanceBody);
|
|
}
|
|
}
|
|
|
|
for (const prop of props) {
|
|
prop.remove();
|
|
}
|
|
|
|
if (computedNodes.length === 0 && afterNodes.length === 0) return;
|
|
|
|
if (path.isClassExpression()) {
|
|
path.scope.push({ id: ref });
|
|
path.replaceWith(
|
|
t.assignmentExpression("=", t.cloneNode(ref), path.node),
|
|
);
|
|
} else if (!path.node.id) {
|
|
// Anonymous class declaration
|
|
path.node.id = ref;
|
|
}
|
|
|
|
path.insertBefore(computedNodes);
|
|
path.insertAfter(afterNodes);
|
|
},
|
|
},
|
|
};
|
|
}
|