import LooseTransformer from "./loose"; import VanillaTransformer from "./vanilla"; import annotateAsPure from "@babel/helper-annotate-as-pure"; import nameFunction from "@babel/helper-function-name"; import { types as t } from "@babel/core"; import globals from "globals"; const getBuiltinClasses = category => Object.keys(globals[category]).filter(name => /^[A-Z]/.test(name)); const builtinClasses = new Set([ ...getBuiltinClasses("builtin"), ...getBuiltinClasses("browser"), ]); export default function(api, options) { const { loose } = options; const Constructor = loose ? LooseTransformer : VanillaTransformer; // todo: investigate traversal requeueing const VISITED = Symbol(); return { visitor: { ExportDefaultDeclaration(path) { if (!path.get("declaration").isClassDeclaration()) return; const { node } = path; const ref = node.declaration.id || path.scope.generateUidIdentifier("class"); node.declaration.id = ref; // Split the class declaration and the export into two separate statements. path.replaceWith(node.declaration); path.insertAfter( t.exportNamedDeclaration(null, [ t.exportSpecifier(t.cloneNode(ref), t.identifier("default")), ]), ); }, ClassDeclaration(path) { const { node } = path; const ref = node.id || path.scope.generateUidIdentifier("class"); path.replaceWith( t.variableDeclaration("let", [ t.variableDeclarator(ref, t.toExpression(node)), ]), ); }, ClassExpression(path, state) { const { node } = path; if (node[VISITED]) return; const inferred = nameFunction(path); if (inferred && inferred !== node) { path.replaceWith(inferred); return; } node[VISITED] = true; path.replaceWith( new Constructor(path, state.file, builtinClasses).run(), ); if (path.isCallExpression()) { annotateAsPure(path); if (path.get("callee").isArrowFunctionExpression()) { path.get("callee").arrowFunctionToExpression(); } } }, }, }; }