saepequia
Version:
A simple, maximally extensible, dependency minimized framework for building modern Ethereum dApps
51 lines (41 loc) • 1.6 kB
text/typescript
import { types as t } from '@babel/core';
import type { NodePath } from '@babel/traverse';
import type { Function, JSXElement as JSXElementNode } from '@babel/types';
function getFunctionName(path: NodePath<Function>): string | null {
if (path.isClassMethod() && t.isIdentifier(path.node.key)) {
return path.node.key.name;
}
return null;
}
export default function JSXElement(path: NodePath<JSXElementNode>) {
// JSX can be safely replaced on an empty fragment because it is unnecessary for styles
const emptyFragment = t.jsxFragment(
t.jsxOpeningFragment(),
t.jsxClosingFragment(),
[]
);
// We can do even more
// If that JSX is a result of a function, we can replace the function body.
const scopePath = path.scope.path;
if (scopePath.isFunction()) {
const emptyBody = t.blockStatement([t.returnStatement(emptyFragment)]);
// Is it not just a function, but a method `render`?
if (getFunctionName(scopePath) === 'render') {
const decl = scopePath.findParent((p) => p.isClassDeclaration());
// Replace the whole component
if (decl?.isClassDeclaration()) {
decl.replaceWith(t.functionDeclaration(decl.node.id, [], emptyBody));
return;
}
}
const body = scopePath.get('body');
if (Array.isArray(body)) {
throw new Error(
`A body of a function is expected to be a single element but an array was returned. It's possible if JS syntax has been changed since that code was written.`
);
}
body.replaceWith(emptyBody);
} else {
path.replaceWith(emptyFragment);
}
}