@babel/helper-hoist-variables
- npm
- Yarn
- pnpm
- Bun
npm install @babel/helper-hoist-variables --save
yarn add @babel/helper-hoist-variables
pnpm add @babel/helper-hoist-variables
bun add @babel/helper-hoist-variables
API
type EmitFunction = (
  id: t.Identifier,
  idName: string,
  hasInit: boolean,
) => any;
declare export default hoistVariables(path: NodePath, emit: EmitFunction, kind: "var" | "let" = "var");
Usage
hoistVariables traverses a given path and replaces variable declarations (var x = 1) with assignment expressions (x = 1). It can also replace left hand side expressions for (x of []); like in for (var x of []);). The binding identifiers (x) from the declarations will be emitted via the EmitFunction callback. hoistVariables is only applied to declarations within the same scope of path, in other words, it will not hoist variables defined inside a function body nested inside path. If kind is "let" it will hoist let declarations.
Generally an EmitFunction will declare such variables in outer scope.
Example
JavaScript
import hoistVariables from "@babel/helper-hoist-variables";
// a naive plugin transforms
// `(function myEval(){ var x = 1; })()`
// to
// `var x;(function myEval(){ x = 1; })()`
module.exports = api => {
  return {
    visitor: {
      Function(path) {
        if (path.node.id.name === "myEval") {
          const outerScope = path.scope.parent;
          hoistVariables(
            path.get("body"),
            id => {
              // Use Scope#push to register a `var` binding in
              // the outer scope
              outerScope.push({ id });
            },
            "var"
          );
        }
      },
    },
  };
};
@babel/plugin-proposal-do-expressions also uses the same technique to hoist the var declarations in do blocks to the containing function scope.