web 3d图形渲染器
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

298 lines
9.6 KiB

  1. "use strict";
  2. Object.defineProperty(exports, "__esModule", {
  3. value: true
  4. });
  5. exports.default = rewriteLiveReferences;
  6. var t = _interopRequireWildcard(require("@babel/types"));
  7. var _template = _interopRequireDefault(require("@babel/template"));
  8. var _helperSimpleAccess = _interopRequireDefault(require("@babel/helper-simple-access"));
  9. function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
  10. function _getRequireWildcardCache() { if (typeof WeakMap !== "function") return null; var cache = new WeakMap(); _getRequireWildcardCache = function () { return cache; }; return cache; }
  11. function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } if (obj === null || typeof obj !== "object" && typeof obj !== "function") { return { default: obj }; } var cache = _getRequireWildcardCache(); if (cache && cache.has(obj)) { return cache.get(obj); } var newObj = {}; var hasPropertyDescriptor = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) { var desc = hasPropertyDescriptor ? Object.getOwnPropertyDescriptor(obj, key) : null; if (desc && (desc.get || desc.set)) { Object.defineProperty(newObj, key, desc); } else { newObj[key] = obj[key]; } } } newObj.default = obj; if (cache) { cache.set(obj, newObj); } return newObj; }
  12. const assert = require("assert");
  13. function rewriteLiveReferences(programPath, metadata) {
  14. const imported = new Map();
  15. const exported = new Map();
  16. const requeueInParent = path => {
  17. programPath.requeue(path);
  18. };
  19. for (const [source, data] of metadata.source) {
  20. for (const [localName, importName] of data.imports) {
  21. imported.set(localName, [source, importName, null]);
  22. }
  23. for (const localName of data.importsNamespace) {
  24. imported.set(localName, [source, null, localName]);
  25. }
  26. }
  27. for (const [local, data] of metadata.local) {
  28. let exportMeta = exported.get(local);
  29. if (!exportMeta) {
  30. exportMeta = [];
  31. exported.set(local, exportMeta);
  32. }
  33. exportMeta.push(...data.names);
  34. }
  35. const rewriteBindingInitVisitorState = {
  36. metadata,
  37. requeueInParent,
  38. scope: programPath.scope,
  39. exported
  40. };
  41. programPath.traverse(rewriteBindingInitVisitor, rewriteBindingInitVisitorState);
  42. (0, _helperSimpleAccess.default)(programPath, new Set([...Array.from(imported.keys()), ...Array.from(exported.keys())]));
  43. const rewriteReferencesVisitorState = {
  44. seen: new WeakSet(),
  45. metadata,
  46. requeueInParent,
  47. scope: programPath.scope,
  48. imported,
  49. exported,
  50. buildImportReference: ([source, importName, localName], identNode) => {
  51. const meta = metadata.source.get(source);
  52. if (localName) {
  53. if (meta.lazy) identNode = t.callExpression(identNode, []);
  54. return identNode;
  55. }
  56. let namespace = t.identifier(meta.name);
  57. if (meta.lazy) namespace = t.callExpression(namespace, []);
  58. const computed = metadata.stringSpecifiers.has(importName);
  59. return t.memberExpression(namespace, computed ? t.stringLiteral(importName) : t.identifier(importName), computed);
  60. }
  61. };
  62. programPath.traverse(rewriteReferencesVisitor, rewriteReferencesVisitorState);
  63. }
  64. const rewriteBindingInitVisitor = {
  65. Scope(path) {
  66. path.skip();
  67. },
  68. ClassDeclaration(path) {
  69. const {
  70. requeueInParent,
  71. exported,
  72. metadata
  73. } = this;
  74. const {
  75. id
  76. } = path.node;
  77. if (!id) throw new Error("Expected class to have a name");
  78. const localName = id.name;
  79. const exportNames = exported.get(localName) || [];
  80. if (exportNames.length > 0) {
  81. const statement = t.expressionStatement(buildBindingExportAssignmentExpression(metadata, exportNames, t.identifier(localName)));
  82. statement._blockHoist = path.node._blockHoist;
  83. requeueInParent(path.insertAfter(statement)[0]);
  84. }
  85. },
  86. VariableDeclaration(path) {
  87. const {
  88. requeueInParent,
  89. exported,
  90. metadata
  91. } = this;
  92. Object.keys(path.getOuterBindingIdentifiers()).forEach(localName => {
  93. const exportNames = exported.get(localName) || [];
  94. if (exportNames.length > 0) {
  95. const statement = t.expressionStatement(buildBindingExportAssignmentExpression(metadata, exportNames, t.identifier(localName)));
  96. statement._blockHoist = path.node._blockHoist;
  97. requeueInParent(path.insertAfter(statement)[0]);
  98. }
  99. });
  100. }
  101. };
  102. const buildBindingExportAssignmentExpression = (metadata, exportNames, localExpr) => {
  103. return (exportNames || []).reduce((expr, exportName) => {
  104. const {
  105. stringSpecifiers
  106. } = metadata;
  107. const computed = stringSpecifiers.has(exportName);
  108. return t.assignmentExpression("=", t.memberExpression(t.identifier(metadata.exportName), computed ? t.stringLiteral(exportName) : t.identifier(exportName), computed), expr);
  109. }, localExpr);
  110. };
  111. const buildImportThrow = localName => {
  112. return _template.default.expression.ast`
  113. (function() {
  114. throw new Error('"' + '${localName}' + '" is read-only.');
  115. })()
  116. `;
  117. };
  118. const rewriteReferencesVisitor = {
  119. ReferencedIdentifier(path) {
  120. const {
  121. seen,
  122. buildImportReference,
  123. scope,
  124. imported,
  125. requeueInParent
  126. } = this;
  127. if (seen.has(path.node)) return;
  128. seen.add(path.node);
  129. const localName = path.node.name;
  130. const importData = imported.get(localName);
  131. if (importData) {
  132. const localBinding = path.scope.getBinding(localName);
  133. const rootBinding = scope.getBinding(localName);
  134. if (rootBinding !== localBinding) return;
  135. const ref = buildImportReference(importData, path.node);
  136. ref.loc = path.node.loc;
  137. if ((path.parentPath.isCallExpression({
  138. callee: path.node
  139. }) || path.parentPath.isOptionalCallExpression({
  140. callee: path.node
  141. }) || path.parentPath.isTaggedTemplateExpression({
  142. tag: path.node
  143. })) && t.isMemberExpression(ref)) {
  144. path.replaceWith(t.sequenceExpression([t.numericLiteral(0), ref]));
  145. } else if (path.isJSXIdentifier() && t.isMemberExpression(ref)) {
  146. const {
  147. object,
  148. property
  149. } = ref;
  150. path.replaceWith(t.jsxMemberExpression(t.jsxIdentifier(object.name), t.jsxIdentifier(property.name)));
  151. } else {
  152. path.replaceWith(ref);
  153. }
  154. requeueInParent(path);
  155. path.skip();
  156. }
  157. },
  158. AssignmentExpression: {
  159. exit(path) {
  160. const {
  161. scope,
  162. seen,
  163. imported,
  164. exported,
  165. requeueInParent,
  166. buildImportReference
  167. } = this;
  168. if (seen.has(path.node)) return;
  169. seen.add(path.node);
  170. const left = path.get("left");
  171. if (left.isMemberExpression()) return;
  172. if (left.isIdentifier()) {
  173. const localName = left.node.name;
  174. if (scope.getBinding(localName) !== path.scope.getBinding(localName)) {
  175. return;
  176. }
  177. const exportedNames = exported.get(localName);
  178. const importData = imported.get(localName);
  179. if ((exportedNames == null ? void 0 : exportedNames.length) > 0 || importData) {
  180. assert(path.node.operator === "=", "Path was not simplified");
  181. const assignment = path.node;
  182. if (importData) {
  183. assignment.left = buildImportReference(importData, assignment.left);
  184. assignment.right = t.sequenceExpression([assignment.right, buildImportThrow(localName)]);
  185. }
  186. path.replaceWith(buildBindingExportAssignmentExpression(this.metadata, exportedNames, assignment));
  187. requeueInParent(path);
  188. }
  189. } else {
  190. const ids = left.getOuterBindingIdentifiers();
  191. const programScopeIds = Object.keys(ids).filter(localName => scope.getBinding(localName) === path.scope.getBinding(localName));
  192. const id = programScopeIds.find(localName => imported.has(localName));
  193. if (id) {
  194. path.node.right = t.sequenceExpression([path.node.right, buildImportThrow(id)]);
  195. }
  196. const items = [];
  197. programScopeIds.forEach(localName => {
  198. const exportedNames = exported.get(localName) || [];
  199. if (exportedNames.length > 0) {
  200. items.push(buildBindingExportAssignmentExpression(this.metadata, exportedNames, t.identifier(localName)));
  201. }
  202. });
  203. if (items.length > 0) {
  204. let node = t.sequenceExpression(items);
  205. if (path.parentPath.isExpressionStatement()) {
  206. node = t.expressionStatement(node);
  207. node._blockHoist = path.parentPath.node._blockHoist;
  208. }
  209. const statement = path.insertAfter(node)[0];
  210. requeueInParent(statement);
  211. }
  212. }
  213. }
  214. },
  215. "ForOfStatement|ForInStatement"(path) {
  216. const {
  217. scope,
  218. node
  219. } = path;
  220. const {
  221. left
  222. } = node;
  223. const {
  224. exported,
  225. scope: programScope
  226. } = this;
  227. if (!t.isVariableDeclaration(left)) {
  228. let didTransform = false;
  229. const bodyPath = path.get("body");
  230. const loopBodyScope = bodyPath.scope;
  231. for (const name of Object.keys(t.getOuterBindingIdentifiers(left))) {
  232. if (exported.get(name) && programScope.getBinding(name) === scope.getBinding(name)) {
  233. didTransform = true;
  234. if (loopBodyScope.hasOwnBinding(name)) {
  235. loopBodyScope.rename(name);
  236. }
  237. }
  238. }
  239. if (!didTransform) {
  240. return;
  241. }
  242. const newLoopId = scope.generateUidIdentifierBasedOnNode(left);
  243. bodyPath.unshiftContainer("body", t.expressionStatement(t.assignmentExpression("=", left, newLoopId)));
  244. path.get("left").replaceWith(t.variableDeclaration("let", [t.variableDeclarator(t.cloneNode(newLoopId))]));
  245. scope.registerDeclaration(path.get("left"));
  246. }
  247. }
  248. };