summaryrefslogtreecommitdiff
path: root/@linaria/packages/babel/src/evaluators/visitors/JSXElement.ts
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);
  }
}