blob: 89a0ccdaca97529c4c15c4387eca74d650c55cc2 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
|
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.`
);
}
const node: typeof scopePath.node = {
...scopePath.node,
body: emptyBody,
params: [],
};
scopePath.replaceWith(node);
} else {
path.replaceWith(emptyFragment);
}
}
|