index.js 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. "use strict";
  2. Object.defineProperty(exports, "__esModule", {
  3. value: true
  4. });
  5. exports.default = void 0;
  6. var _helperPluginUtils = require("@babel/helper-plugin-utils");
  7. var _core = require("@babel/core");
  8. var _noHelperImplementation = _interopRequireDefault(require("./no-helper-implementation"));
  9. function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
  10. var _default = (0, _helperPluginUtils.declare)((api, options) => {
  11. api.assertVersion(7);
  12. const {
  13. loose,
  14. assumeArray,
  15. allowArrayLike
  16. } = options;
  17. if (loose === true && assumeArray === true) {
  18. throw new Error(`The loose and assumeArray options cannot be used together in @babel/plugin-transform-for-of`);
  19. }
  20. if (assumeArray === true && allowArrayLike === true) {
  21. throw new Error(`The assumeArray and allowArrayLike options cannot be used together in @babel/plugin-transform-for-of`);
  22. }
  23. if (allowArrayLike && /^7\.\d\./.test(api.version)) {
  24. throw new Error(`The allowArrayLike is only supported when using @babel/core@^7.10.0`);
  25. }
  26. if (assumeArray) {
  27. return {
  28. name: "transform-for-of",
  29. visitor: {
  30. ForOfStatement(path) {
  31. const {
  32. scope
  33. } = path;
  34. const {
  35. left,
  36. right,
  37. await: isAwait
  38. } = path.node;
  39. if (isAwait) {
  40. return;
  41. }
  42. const i = scope.generateUidIdentifier("i");
  43. let array = scope.maybeGenerateMemoised(right, true);
  44. const inits = [_core.types.variableDeclarator(i, _core.types.numericLiteral(0))];
  45. if (array) {
  46. inits.push(_core.types.variableDeclarator(array, right));
  47. } else {
  48. array = right;
  49. }
  50. const item = _core.types.memberExpression(_core.types.cloneNode(array), _core.types.cloneNode(i), true);
  51. let assignment;
  52. if (_core.types.isVariableDeclaration(left)) {
  53. assignment = left;
  54. assignment.declarations[0].init = item;
  55. } else {
  56. assignment = _core.types.expressionStatement(_core.types.assignmentExpression("=", left, item));
  57. }
  58. let blockBody;
  59. const body = path.get("body");
  60. if (body.isBlockStatement() && Object.keys(path.getBindingIdentifiers()).some(id => body.scope.hasOwnBinding(id))) {
  61. blockBody = _core.types.blockStatement([assignment, body.node]);
  62. } else {
  63. blockBody = _core.types.toBlock(body.node);
  64. blockBody.body.unshift(assignment);
  65. }
  66. path.replaceWith(_core.types.forStatement(_core.types.variableDeclaration("let", inits), _core.types.binaryExpression("<", _core.types.cloneNode(i), _core.types.memberExpression(_core.types.cloneNode(array), _core.types.identifier("length"))), _core.types.updateExpression("++", _core.types.cloneNode(i)), blockBody));
  67. }
  68. }
  69. };
  70. }
  71. const buildForOfArray = (0, _core.template)(`
  72. for (var KEY = 0, NAME = ARR; KEY < NAME.length; KEY++) BODY;
  73. `);
  74. const buildForOfLoose = _core.template.statements(`
  75. for (var ITERATOR_HELPER = CREATE_ITERATOR_HELPER(OBJECT, ALLOW_ARRAY_LIKE), STEP_KEY;
  76. !(STEP_KEY = ITERATOR_HELPER()).done;) BODY;
  77. `);
  78. const buildForOf = _core.template.statements(`
  79. var ITERATOR_HELPER = CREATE_ITERATOR_HELPER(OBJECT, ALLOW_ARRAY_LIKE), STEP_KEY;
  80. try {
  81. for (ITERATOR_HELPER.s(); !(STEP_KEY = ITERATOR_HELPER.n()).done;) BODY;
  82. } catch (err) {
  83. ITERATOR_HELPER.e(err);
  84. } finally {
  85. ITERATOR_HELPER.f();
  86. }
  87. `);
  88. const builder = loose ? {
  89. build: buildForOfLoose,
  90. helper: "createForOfIteratorHelperLoose",
  91. getContainer: nodes => nodes
  92. } : {
  93. build: buildForOf,
  94. helper: "createForOfIteratorHelper",
  95. getContainer: nodes => nodes[1].block.body
  96. };
  97. function _ForOfStatementArray(path) {
  98. const {
  99. node,
  100. scope
  101. } = path;
  102. const right = scope.generateUidIdentifierBasedOnNode(node.right, "arr");
  103. const iterationKey = scope.generateUidIdentifier("i");
  104. const loop = buildForOfArray({
  105. BODY: node.body,
  106. KEY: iterationKey,
  107. NAME: right,
  108. ARR: node.right
  109. });
  110. _core.types.inherits(loop, node);
  111. _core.types.ensureBlock(loop);
  112. const iterationValue = _core.types.memberExpression(_core.types.cloneNode(right), _core.types.cloneNode(iterationKey), true);
  113. const left = node.left;
  114. if (_core.types.isVariableDeclaration(left)) {
  115. left.declarations[0].init = iterationValue;
  116. loop.body.body.unshift(left);
  117. } else {
  118. loop.body.body.unshift(_core.types.expressionStatement(_core.types.assignmentExpression("=", left, iterationValue)));
  119. }
  120. return loop;
  121. }
  122. return {
  123. name: "transform-for-of",
  124. visitor: {
  125. ForOfStatement(path, state) {
  126. const right = path.get("right");
  127. if (right.isArrayExpression() || right.isGenericType("Array") || _core.types.isArrayTypeAnnotation(right.getTypeAnnotation())) {
  128. path.replaceWith(_ForOfStatementArray(path));
  129. return;
  130. }
  131. if (!state.availableHelper(builder.helper)) {
  132. (0, _noHelperImplementation.default)(loose, path, state);
  133. return;
  134. }
  135. const {
  136. node,
  137. parent,
  138. scope
  139. } = path;
  140. const left = node.left;
  141. let declar;
  142. const stepKey = scope.generateUid("step");
  143. const stepValue = _core.types.memberExpression(_core.types.identifier(stepKey), _core.types.identifier("value"));
  144. if (_core.types.isVariableDeclaration(left)) {
  145. declar = _core.types.variableDeclaration(left.kind, [_core.types.variableDeclarator(left.declarations[0].id, stepValue)]);
  146. } else {
  147. declar = _core.types.expressionStatement(_core.types.assignmentExpression("=", left, stepValue));
  148. }
  149. path.ensureBlock();
  150. node.body.body.unshift(declar);
  151. const nodes = builder.build({
  152. CREATE_ITERATOR_HELPER: state.addHelper(builder.helper),
  153. ITERATOR_HELPER: scope.generateUidIdentifier("iterator"),
  154. ALLOW_ARRAY_LIKE: allowArrayLike ? _core.types.booleanLiteral(true) : null,
  155. STEP_KEY: _core.types.identifier(stepKey),
  156. OBJECT: node.right,
  157. BODY: node.body
  158. });
  159. const container = builder.getContainer(nodes);
  160. _core.types.inherits(container[0], node);
  161. _core.types.inherits(container[0].body, node.body);
  162. if (_core.types.isLabeledStatement(parent)) {
  163. container[0] = _core.types.labeledStatement(parent.label, container[0]);
  164. path.parentPath.replaceWithMultiple(nodes);
  165. path.remove();
  166. } else {
  167. path.replaceWithMultiple(nodes);
  168. }
  169. }
  170. }
  171. };
  172. });
  173. exports.default = _default;