Spaces:
Sleeping
Sleeping
| ; | |
| Object.defineProperty(exports, "__esModule", { | |
| value: true | |
| }); | |
| exports.default = void 0; | |
| var _helperPluginUtils = require("@babel/helper-plugin-utils"); | |
| var _core = require("@babel/core"); | |
| var _loop = require("./loop.js"); | |
| var _validation = require("./validation.js"); | |
| var _annexB_3_ = require("./annex-B_3_3.js"); | |
| var _default = exports.default = (0, _helperPluginUtils.declare)((api, opts) => { | |
| api.assertVersion(7); | |
| const { | |
| throwIfClosureRequired = false, | |
| tdz: tdzEnabled = false | |
| } = opts; | |
| if (typeof throwIfClosureRequired !== "boolean") { | |
| throw new Error(`.throwIfClosureRequired must be a boolean, or undefined`); | |
| } | |
| if (typeof tdzEnabled !== "boolean") { | |
| throw new Error(`.tdz must be a boolean, or undefined`); | |
| } | |
| return { | |
| name: "transform-block-scoping", | |
| visitor: _core.traverse.visitors.merge([_annexB_3_.annexB33FunctionsVisitor, { | |
| Loop(path, state) { | |
| const isForStatement = path.isForStatement(); | |
| const headPath = isForStatement ? path.get("init") : path.isForXStatement() ? path.get("left") : null; | |
| let needsBodyWrap = false; | |
| const markNeedsBodyWrap = () => { | |
| if (throwIfClosureRequired) { | |
| throw path.buildCodeFrameError("Compiling let/const in this block would add a closure " + "(throwIfClosureRequired)."); | |
| } | |
| needsBodyWrap = true; | |
| }; | |
| const body = path.get("body"); | |
| let bodyScope; | |
| if (body.isBlockStatement()) { | |
| bodyScope = body.scope; | |
| } | |
| const bindings = (0, _loop.getLoopBodyBindings)(path); | |
| for (const binding of bindings) { | |
| const { | |
| capturedInClosure | |
| } = (0, _loop.getUsageInBody)(binding, path); | |
| if (capturedInClosure) markNeedsBodyWrap(); | |
| } | |
| const captured = []; | |
| const updatedBindingsUsages = new Map(); | |
| if (headPath && isBlockScoped(headPath.node)) { | |
| const names = Object.keys(headPath.getBindingIdentifiers()); | |
| const headScope = headPath.scope; | |
| for (let name of names) { | |
| var _bodyScope; | |
| if ((_bodyScope = bodyScope) != null && _bodyScope.hasOwnBinding(name)) continue; | |
| let binding = headScope.getOwnBinding(name); | |
| if (!binding) { | |
| headScope.crawl(); | |
| binding = headScope.getOwnBinding(name); | |
| } | |
| const { | |
| usages, | |
| capturedInClosure, | |
| hasConstantViolations | |
| } = (0, _loop.getUsageInBody)(binding, path); | |
| if (headScope.parent.hasBinding(name) || headScope.parent.hasGlobal(name)) { | |
| const newName = headScope.generateUid(name); | |
| headScope.rename(name, newName); | |
| name = newName; | |
| } | |
| if (capturedInClosure) { | |
| markNeedsBodyWrap(); | |
| captured.push(name); | |
| } | |
| if (isForStatement && hasConstantViolations) { | |
| updatedBindingsUsages.set(name, usages); | |
| } | |
| } | |
| } | |
| if (needsBodyWrap) { | |
| const varPath = (0, _loop.wrapLoopBody)(path, captured, updatedBindingsUsages); | |
| if (headPath != null && headPath.isVariableDeclaration()) { | |
| transformBlockScopedVariable(headPath, state, tdzEnabled); | |
| } | |
| varPath.get("declarations.0.init").unwrapFunctionEnvironment(); | |
| } | |
| }, | |
| VariableDeclaration(path, state) { | |
| transformBlockScopedVariable(path, state, tdzEnabled); | |
| }, | |
| ClassDeclaration(path) { | |
| const { | |
| id | |
| } = path.node; | |
| if (!id) return; | |
| const { | |
| scope | |
| } = path.parentPath; | |
| if (!(0, _annexB_3_.isVarScope)(scope) && scope.parent.hasBinding(id.name, { | |
| noUids: true | |
| })) { | |
| path.scope.rename(id.name); | |
| } | |
| } | |
| }]) | |
| }; | |
| }); | |
| const conflictingFunctionsVisitor = { | |
| Scope(path, { | |
| names | |
| }) { | |
| for (const name of names) { | |
| const binding = path.scope.getOwnBinding(name); | |
| if (binding && binding.kind === "hoisted") { | |
| path.scope.rename(name); | |
| } | |
| } | |
| }, | |
| "Expression|Declaration"(path) { | |
| path.skip(); | |
| } | |
| }; | |
| function transformBlockScopedVariable(path, state, tdzEnabled) { | |
| if (!isBlockScoped(path.node)) return; | |
| const dynamicTDZNames = (0, _validation.validateUsage)(path, state, tdzEnabled); | |
| path.node.kind = "var"; | |
| const bindingNames = Object.keys(path.getBindingIdentifiers()); | |
| for (const name of bindingNames) { | |
| const binding = path.scope.getOwnBinding(name); | |
| if (!binding) continue; | |
| binding.kind = "var"; | |
| } | |
| if (isInLoop(path) && !(0, _loop.isVarInLoopHead)(path) || dynamicTDZNames.length > 0) { | |
| for (const decl of path.node.declarations) { | |
| var _decl$init; | |
| (_decl$init = decl.init) != null ? _decl$init : decl.init = path.scope.buildUndefinedNode(); | |
| } | |
| } | |
| const blockScope = path.scope; | |
| const varScope = blockScope.getFunctionParent() || blockScope.getProgramParent(); | |
| if (varScope !== blockScope) { | |
| for (const name of bindingNames) { | |
| let newName = name; | |
| if (blockScope.parent.hasBinding(name, { | |
| noUids: true | |
| }) || blockScope.parent.hasGlobal(name)) { | |
| newName = blockScope.generateUid(name); | |
| blockScope.rename(name, newName); | |
| } | |
| blockScope.moveBindingTo(newName, varScope); | |
| } | |
| } | |
| blockScope.path.traverse(conflictingFunctionsVisitor, { | |
| names: bindingNames | |
| }); | |
| for (const name of dynamicTDZNames) { | |
| path.scope.push({ | |
| id: _core.types.identifier(name), | |
| init: state.addHelper("temporalUndefined") | |
| }); | |
| } | |
| } | |
| function isLetOrConst(kind) { | |
| return kind === "let" || kind === "const"; | |
| } | |
| function isInLoop(path) { | |
| if (!path.parentPath) return false; | |
| if (path.parentPath.isLoop()) return true; | |
| if (path.parentPath.isFunctionParent()) return false; | |
| return isInLoop(path.parentPath); | |
| } | |
| function isBlockScoped(node) { | |
| if (!_core.types.isVariableDeclaration(node)) return false; | |
| if (node[_core.types.BLOCK_SCOPED_SYMBOL]) { | |
| return true; | |
| } | |
| if (!isLetOrConst(node.kind) && node.kind !== "using") { | |
| return false; | |
| } | |
| return true; | |
| } | |
| //# sourceMappingURL=index.js.map | |