@mmapp/react-compiler 0.1.0-alpha.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (314) hide show
  1. package/README.md +107 -0
  2. package/compile-blueprint-chat.mjs +99 -0
  3. package/compile-blueprint-glass-console.mjs +98 -0
  4. package/compile-chat-defs.mjs +92 -0
  5. package/dist/babel/index.d.mts +3 -0
  6. package/dist/babel/index.d.ts +3 -0
  7. package/dist/babel/index.js +4851 -0
  8. package/dist/babel/index.mjs +7 -0
  9. package/dist/chunk-26U577GB.mjs +3465 -0
  10. package/dist/chunk-2FBDFAX6.mjs +2362 -0
  11. package/dist/chunk-2L4QSMXG.mjs +175 -0
  12. package/dist/chunk-2REDFOER.mjs +931 -0
  13. package/dist/chunk-46YKSHQR.mjs +175 -0
  14. package/dist/chunk-4XHK6FWL.mjs +2058 -0
  15. package/dist/chunk-5M7DKKBC.mjs +215 -0
  16. package/dist/chunk-5VNJ7C6N.mjs +154 -0
  17. package/dist/chunk-6CQOAAMV.mjs +1803 -0
  18. package/dist/chunk-6SEVAAVT.mjs +3516 -0
  19. package/dist/chunk-6YLR5ZDA.mjs +2829 -0
  20. package/dist/chunk-AOGY2GK6.mjs +3292 -0
  21. package/dist/chunk-AXXUXRNA.mjs +1434 -0
  22. package/dist/chunk-CHLVKMQW.mjs +175 -0
  23. package/dist/chunk-CKGOZAB7.mjs +939 -0
  24. package/dist/chunk-D34RAZUX.mjs +2223 -0
  25. package/dist/chunk-EQGA6A6D.mjs +121 -0
  26. package/dist/chunk-EY2CSXYA.mjs +822 -0
  27. package/dist/chunk-FIQ65CDR.mjs +925 -0
  28. package/dist/chunk-FOZXJFAR.mjs +186 -0
  29. package/dist/chunk-FX6URXWN.mjs +186 -0
  30. package/dist/chunk-G7SMOWOL.mjs +828 -0
  31. package/dist/chunk-GGB4G5YY.mjs +175 -0
  32. package/dist/chunk-HLRGCCIL.mjs +4839 -0
  33. package/dist/chunk-HOIUP6IF.mjs +690 -0
  34. package/dist/chunk-I3AU7GRD.mjs +120 -0
  35. package/dist/chunk-ILFGMUVD.mjs +1933 -0
  36. package/dist/chunk-IPTX5MJU.mjs +3223 -0
  37. package/dist/chunk-ITGUSH2Z.mjs +2783 -0
  38. package/dist/chunk-IXHBCAMF.mjs +3306 -0
  39. package/dist/chunk-J7TWJ3TM.mjs +2784 -0
  40. package/dist/chunk-JDPLDGVF.mjs +4810 -0
  41. package/dist/chunk-K53XP2DL.mjs +148 -0
  42. package/dist/chunk-K5HX2SVL.mjs +1902 -0
  43. package/dist/chunk-KFGYOOVS.mjs +214 -0
  44. package/dist/chunk-KFVVOS5N.mjs +925 -0
  45. package/dist/chunk-L2OZ4CDV.mjs +113 -0
  46. package/dist/chunk-MIZV3TAN.mjs +3293 -0
  47. package/dist/chunk-NKKLQE5V.mjs +148 -0
  48. package/dist/chunk-NOW23XFZ.mjs +186 -0
  49. package/dist/chunk-NRXQKQ74.mjs +148 -0
  50. package/dist/chunk-OWI6XWCD.mjs +3375 -0
  51. package/dist/chunk-PRUMNNDI.mjs +3192 -0
  52. package/dist/chunk-QTBD5B3F.mjs +148 -0
  53. package/dist/chunk-SKSDPPNT.mjs +3788 -0
  54. package/dist/chunk-SP2YUS33.mjs +186 -0
  55. package/dist/chunk-SU4E6E7B.mjs +3153 -0
  56. package/dist/chunk-SYUUKW5A.mjs +3379 -0
  57. package/dist/chunk-UL2XZEMA.mjs +3128 -0
  58. package/dist/chunk-XMWUHQVV.mjs +939 -0
  59. package/dist/chunk-XZNEDRGN.mjs +3876 -0
  60. package/dist/chunk-Y6FXYEAI.mjs +10 -0
  61. package/dist/chunk-YFS6JMYO.mjs +3342 -0
  62. package/dist/chunk-Z6AIQ4KL.mjs +113 -0
  63. package/dist/cli/index.d.mts +1 -0
  64. package/dist/cli/index.d.ts +1 -0
  65. package/dist/cli/index.js +11585 -0
  66. package/dist/cli/index.mjs +701 -0
  67. package/dist/codemod/cli.d.mts +1 -0
  68. package/dist/codemod/cli.d.ts +1 -0
  69. package/dist/codemod/cli.js +1104 -0
  70. package/dist/codemod/cli.mjs +157 -0
  71. package/dist/codemod/index.d.mts +148 -0
  72. package/dist/codemod/index.d.ts +148 -0
  73. package/dist/codemod/index.js +981 -0
  74. package/dist/codemod/index.mjs +25 -0
  75. package/dist/dev-server-Bs_sz2DG.d.mts +111 -0
  76. package/dist/dev-server-Bs_sz2DG.d.ts +111 -0
  77. package/dist/dev-server-CjoufJ-u.d.mts +109 -0
  78. package/dist/dev-server-CjoufJ-u.d.ts +109 -0
  79. package/dist/dev-server.d.mts +3 -0
  80. package/dist/dev-server.d.ts +3 -0
  81. package/dist/dev-server.js +7603 -0
  82. package/dist/dev-server.mjs +11 -0
  83. package/dist/envelope-DD7v0v6E.d.mts +265 -0
  84. package/dist/envelope-DD7v0v6E.d.ts +265 -0
  85. package/dist/envelope-vCVjrHlo.d.mts +265 -0
  86. package/dist/envelope-vCVjrHlo.d.ts +265 -0
  87. package/dist/envelope.d.mts +2 -0
  88. package/dist/envelope.d.ts +2 -0
  89. package/dist/envelope.js +5184 -0
  90. package/dist/envelope.mjs +9 -0
  91. package/dist/index-B5gSgvnd.d.mts +44 -0
  92. package/dist/index-B5gSgvnd.d.ts +44 -0
  93. package/dist/index-Bs0MnR54.d.mts +103 -0
  94. package/dist/index-Bs0MnR54.d.ts +103 -0
  95. package/dist/index-DR0nNc_f.d.mts +101 -0
  96. package/dist/index-DR0nNc_f.d.ts +101 -0
  97. package/dist/index-revho_gS.d.mts +104 -0
  98. package/dist/index-revho_gS.d.ts +104 -0
  99. package/dist/index.d.mts +1099 -0
  100. package/dist/index.d.ts +1099 -0
  101. package/dist/index.js +10162 -0
  102. package/dist/index.mjs +372 -0
  103. package/dist/init-IXEE2RCF.mjs +340 -0
  104. package/dist/project-compiler-EGJUTAJU.mjs +10 -0
  105. package/dist/project-compiler-VFR6CSDX.mjs +10 -0
  106. package/dist/project-decompiler-5GY2KSG4.mjs +7 -0
  107. package/dist/pull-A2QUHW4K.mjs +109 -0
  108. package/dist/pull-JBEQWVPE.mjs +109 -0
  109. package/dist/testing/index.d.mts +211 -0
  110. package/dist/testing/index.d.ts +211 -0
  111. package/dist/testing/index.js +5106 -0
  112. package/dist/testing/index.mjs +247 -0
  113. package/dist/vite/index.d.mts +59 -0
  114. package/dist/vite/index.d.ts +59 -0
  115. package/dist/vite/index.js +5023 -0
  116. package/dist/vite/index.mjs +8 -0
  117. package/examples/README.md +72 -0
  118. package/examples/authentication/main.workflow.tsx +139 -0
  119. package/examples/authentication/mm.config.ts +22 -0
  120. package/examples/authentication/models/auth.ts +45 -0
  121. package/examples/authentication/pages/LoginPage.tsx +79 -0
  122. package/examples/authentication/pages/SignupPage.tsx +87 -0
  123. package/examples/counter.workflow.tsx +65 -0
  124. package/examples/dashboard.workflow.tsx +419 -0
  125. package/examples/invoice-approval/actions/invoice.server.ts +72 -0
  126. package/examples/invoice-approval/main.workflow.tsx +168 -0
  127. package/examples/invoice-approval/mm.config.ts +18 -0
  128. package/examples/invoice-approval/models/invoice.ts +46 -0
  129. package/examples/invoice-approval/pages/InvoiceDetailPage.tsx +175 -0
  130. package/examples/invoice-approval/pages/InvoiceFormPage.tsx +198 -0
  131. package/examples/invoice-approval/pages/InvoiceListPage.tsx +141 -0
  132. package/examples/todo-app.workflow.tsx +131 -0
  133. package/examples/uber-app/actions/matching.server.ts +177 -0
  134. package/examples/uber-app/actions/notifications.server.ts +176 -0
  135. package/examples/uber-app/actions/payments.server.ts +184 -0
  136. package/examples/uber-app/actions/pricing.server.ts +176 -0
  137. package/examples/uber-app/app/admin/analytics.tsx +102 -0
  138. package/examples/uber-app/app/admin/fleet.tsx +102 -0
  139. package/examples/uber-app/app/admin/surge-pricing.tsx +95 -0
  140. package/examples/uber-app/app/driver/dashboard.tsx +87 -0
  141. package/examples/uber-app/app/driver/earnings.tsx +101 -0
  142. package/examples/uber-app/app/driver/navigation.tsx +94 -0
  143. package/examples/uber-app/app/driver/ride-acceptance.tsx +103 -0
  144. package/examples/uber-app/app/rider/home.tsx +109 -0
  145. package/examples/uber-app/app/rider/payment-methods.tsx +134 -0
  146. package/examples/uber-app/app/rider/ride-history.tsx +90 -0
  147. package/examples/uber-app/app/rider/ride-tracking.tsx +108 -0
  148. package/examples/uber-app/components/DriverCard.tsx +176 -0
  149. package/examples/uber-app/components/MapView.tsx +216 -0
  150. package/examples/uber-app/components/RatingStars.tsx +227 -0
  151. package/examples/uber-app/components/RideCard.tsx +167 -0
  152. package/examples/uber-app/mm.config.ts +30 -0
  153. package/examples/uber-app/models/location.model.ts +70 -0
  154. package/examples/uber-app/models/payment.model.ts +87 -0
  155. package/examples/uber-app/models/rating.model.ts +54 -0
  156. package/examples/uber-app/models/ride.model.ts +118 -0
  157. package/examples/uber-app/models/user.model.ts +66 -0
  158. package/examples/uber-app/models/vehicle.model.ts +63 -0
  159. package/examples/uber-app/tests/payment.test.tsx +129 -0
  160. package/examples/uber-app/tests/ride-flow.test.tsx +123 -0
  161. package/examples/uber-app/workflows/dispute-resolution.workflow.tsx +205 -0
  162. package/examples/uber-app/workflows/driver-onboarding.workflow.tsx +227 -0
  163. package/examples/uber-app/workflows/payment-processing.workflow.tsx +223 -0
  164. package/examples/uber-app/workflows/ride-request.workflow.tsx +194 -0
  165. package/package.json +77 -0
  166. package/package.json.backup +86 -0
  167. package/scripts/decompile.ts +226 -0
  168. package/scripts/seed-auth.ts +267 -0
  169. package/scripts/seed-uber.ts +248 -0
  170. package/scripts/validate-uber.ts +119 -0
  171. package/seed-blueprint-chat.mjs +444 -0
  172. package/seed-blueprint-glass-console.mjs +445 -0
  173. package/seed-compiled.mjs +318 -0
  174. package/src/RoundTripValidator.ts +400 -0
  175. package/src/__tests__/atom-rendering-coverage.test.ts +680 -0
  176. package/src/__tests__/auth-module-compilation.test.ts +247 -0
  177. package/src/__tests__/auth-template-compilation.test.ts +589 -0
  178. package/src/__tests__/change-extractor.test.ts +142 -0
  179. package/src/__tests__/cli-pull.test.ts +73 -0
  180. package/src/__tests__/cli-test.test.ts +72 -0
  181. package/src/__tests__/component-extractor.test.ts +331 -0
  182. package/src/__tests__/context-extractor.test.ts +145 -0
  183. package/src/__tests__/decompiler.test.ts +718 -0
  184. package/src/__tests__/define-blueprint.test.ts +133 -0
  185. package/src/__tests__/definition-validator.test.ts +519 -0
  186. package/src/__tests__/during-extractor.test.ts +152 -0
  187. package/src/__tests__/effect-extractor.test.ts +107 -0
  188. package/src/__tests__/event-emission.test.ts +127 -0
  189. package/src/__tests__/examples.test.ts +236 -0
  190. package/src/__tests__/full-blueprint-coverage.test.ts +1221 -0
  191. package/src/__tests__/golden-suite.test.ts +403 -0
  192. package/src/__tests__/grammar-island-extractor.test.ts +289 -0
  193. package/src/__tests__/instance-key.test.ts +82 -0
  194. package/src/__tests__/ir-migration.test.ts +255 -0
  195. package/src/__tests__/lock-file.test.ts +117 -0
  196. package/src/__tests__/model-extractor.test.ts +195 -0
  197. package/src/__tests__/model-field-acl.test.ts +237 -0
  198. package/src/__tests__/model-hooks.test.ts +130 -0
  199. package/src/__tests__/model-ref-resolution.test.ts +268 -0
  200. package/src/__tests__/model-roundtrip.test.ts +502 -0
  201. package/src/__tests__/model-runtime.test.ts +112 -0
  202. package/src/__tests__/model-transitions.test.ts +183 -0
  203. package/src/__tests__/nrt-action-trace.test.ts +391 -0
  204. package/src/__tests__/pipeline-hardening.test.ts +413 -0
  205. package/src/__tests__/project-compiler.test.ts +546 -0
  206. package/src/__tests__/project-decompiler.test.ts +343 -0
  207. package/src/__tests__/query-compilation.test.ts +145 -0
  208. package/src/__tests__/round-trip/PLAN.md +158 -0
  209. package/src/__tests__/round-trip/README.md +52 -0
  210. package/src/__tests__/round-trip/RESULTS.md +86 -0
  211. package/src/__tests__/round-trip/fixtures/data-heavy/main.workflow.tsx +55 -0
  212. package/src/__tests__/round-trip/fixtures/data-heavy/mm.config.ts +11 -0
  213. package/src/__tests__/round-trip/fixtures/data-heavy/models/contact.ts +54 -0
  214. package/src/__tests__/round-trip/fixtures/full-workflow/main.workflow.tsx +79 -0
  215. package/src/__tests__/round-trip/fixtures/full-workflow/mm.config.ts +12 -0
  216. package/src/__tests__/round-trip/fixtures/full-workflow/models/order.ts +50 -0
  217. package/src/__tests__/round-trip/fixtures/simple-crud/main.workflow.tsx +25 -0
  218. package/src/__tests__/round-trip/fixtures/simple-crud/mm.config.ts +11 -0
  219. package/src/__tests__/round-trip/fixtures/simple-crud/models/task.ts +32 -0
  220. package/src/__tests__/round-trip/fixtures/view-heavy/main.workflow.tsx +79 -0
  221. package/src/__tests__/round-trip/fixtures/view-heavy/mm.config.ts +10 -0
  222. package/src/__tests__/round-trip/round-trip.test.ts +2598 -0
  223. package/src/__tests__/round-trip-ir.test.ts +300 -0
  224. package/src/__tests__/round-trip.test.ts +1212 -0
  225. package/src/__tests__/route-merging.test.ts +372 -0
  226. package/src/__tests__/router-composition.test.ts +489 -0
  227. package/src/__tests__/router-extractor.test.ts +176 -0
  228. package/src/__tests__/server-action-extractor.test.ts +128 -0
  229. package/src/__tests__/smart-type-inference.test.ts +365 -0
  230. package/src/__tests__/source-envelope.test.ts +284 -0
  231. package/src/__tests__/source-fidelity.test.ts +516 -0
  232. package/src/__tests__/state-extractor.test.ts +115 -0
  233. package/src/__tests__/strict-mode.test.ts +227 -0
  234. package/src/__tests__/transition-effect-extractor.test.ts +119 -0
  235. package/src/__tests__/transition-extractor.test.ts +68 -0
  236. package/src/__tests__/ts-to-expression.test.ts +462 -0
  237. package/src/__tests__/type-generator.test.ts +201 -0
  238. package/src/__tests__/uber-validation.test.ts +502 -0
  239. package/src/action-compiler.ts +361 -0
  240. package/src/babel/emitters/experience-transform.ts +199 -0
  241. package/src/babel/emitters/ir-to-tsx-emitter.ts +110 -0
  242. package/src/babel/emitters/pure-form-emitter.ts +1023 -0
  243. package/src/babel/emitters/runtime-glue-emitter.ts +39 -0
  244. package/src/babel/extractors/change-extractor.ts +199 -0
  245. package/src/babel/extractors/component-extractor.ts +907 -0
  246. package/src/babel/extractors/computed-extractor.ts +262 -0
  247. package/src/babel/extractors/context-extractor.ts +277 -0
  248. package/src/babel/extractors/during-extractor.ts +295 -0
  249. package/src/babel/extractors/effect-extractor.ts +340 -0
  250. package/src/babel/extractors/event-extractor.ts +235 -0
  251. package/src/babel/extractors/grammar-island-extractor.ts +302 -0
  252. package/src/babel/extractors/model-extractor.ts +1018 -0
  253. package/src/babel/extractors/router-extractor.ts +303 -0
  254. package/src/babel/extractors/server-action-extractor.ts +173 -0
  255. package/src/babel/extractors/server-action-hook-extractor.ts +72 -0
  256. package/src/babel/extractors/server-state-extractor.ts +88 -0
  257. package/src/babel/extractors/state-extractor.ts +214 -0
  258. package/src/babel/extractors/transition-effect-extractor.ts +176 -0
  259. package/src/babel/extractors/transition-extractor.ts +143 -0
  260. package/src/babel/index.ts +24 -0
  261. package/src/babel/transpilers/ts-to-expression.ts +674 -0
  262. package/src/babel/visitor.ts +807 -0
  263. package/src/cli/auth.ts +255 -0
  264. package/src/cli/build.ts +288 -0
  265. package/src/cli/deploy.ts +206 -0
  266. package/src/cli/index.ts +328 -0
  267. package/src/cli/init.ts +388 -0
  268. package/src/cli/installer.ts +261 -0
  269. package/src/cli/lock-file.ts +94 -0
  270. package/src/cli/mmrc.ts +22 -0
  271. package/src/cli/pull.ts +172 -0
  272. package/src/cli/registry-client.ts +175 -0
  273. package/src/cli/test.ts +397 -0
  274. package/src/cli/type-generator.ts +243 -0
  275. package/src/codemod/__tests__/forward.test.ts +239 -0
  276. package/src/codemod/__tests__/reverse.test.ts +145 -0
  277. package/src/codemod/__tests__/round-trip.test.ts +137 -0
  278. package/src/codemod/annotation.ts +97 -0
  279. package/src/codemod/classify.ts +197 -0
  280. package/src/codemod/cli.ts +207 -0
  281. package/src/codemod/control-flow.ts +409 -0
  282. package/src/codemod/forward.ts +244 -0
  283. package/src/codemod/import-manager.ts +171 -0
  284. package/src/codemod/index.ts +120 -0
  285. package/src/codemod/reverse.ts +197 -0
  286. package/src/codemod/rules.ts +174 -0
  287. package/src/codemod/state-transform.ts +126 -0
  288. package/src/decompiler/ast-builder.ts +538 -0
  289. package/src/decompiler/config-generator.ts +151 -0
  290. package/src/decompiler/index.ts +315 -0
  291. package/src/decompiler/project-decompiler.ts +1776 -0
  292. package/src/decompiler/project.ts +862 -0
  293. package/src/decompiler/split-strategy.ts +140 -0
  294. package/src/decompiler/state-emitter.ts +1053 -0
  295. package/src/decompiler/sx-emitter.ts +318 -0
  296. package/src/decompiler/workspace-hydrator.ts +189 -0
  297. package/src/dev-server.ts +238 -0
  298. package/src/envelope/fs-tree.ts +217 -0
  299. package/src/envelope/source-envelope.ts +264 -0
  300. package/src/envelope.ts +315 -0
  301. package/src/incremental-compiler.ts +401 -0
  302. package/src/index.ts +99 -0
  303. package/src/model-compiler.ts +277 -0
  304. package/src/project-compiler.ts +1629 -0
  305. package/src/route-extractor.ts +333 -0
  306. package/src/testing/index.ts +32 -0
  307. package/src/testing/snapshot.ts +252 -0
  308. package/src/testing/test-utils.ts +226 -0
  309. package/src/types.ts +68 -0
  310. package/src/vite/index.ts +288 -0
  311. package/test-compile.mjs +142 -0
  312. package/tsconfig.json +25 -0
  313. package/tsup.config.ts +23 -0
  314. package/vitest.config.ts +9 -0
@@ -0,0 +1,171 @@
1
+ /**
2
+ * Import manager — tracks atom usage and rewrites imports.
3
+ */
4
+
5
+ import type { NodePath } from '@babel/traverse';
6
+ import * as t from '@babel/types';
7
+ import { ATOM_IMPORT_SOURCE, ALL_ATOM_NAMES } from './rules';
8
+
9
+ export interface ImportTracker {
10
+ /** Mark an atom as used (forward: add to imports) */
11
+ addAtom(name: string): void;
12
+ /** Mark an atom as removed (reverse: remove from imports) */
13
+ removeAtom(name: string): void;
14
+ /** Get the set of atoms that need importing */
15
+ getUsedAtoms(): Set<string>;
16
+ /** Get the set of atoms that were removed */
17
+ getRemovedAtoms(): Set<string>;
18
+ }
19
+
20
+ export function createImportTracker(): ImportTracker {
21
+ const used = new Set<string>();
22
+ const removed = new Set<string>();
23
+
24
+ return {
25
+ addAtom(name: string) {
26
+ used.add(name);
27
+ removed.delete(name);
28
+ },
29
+ removeAtom(name: string) {
30
+ removed.add(name);
31
+ },
32
+ getUsedAtoms() {
33
+ return used;
34
+ },
35
+ getRemovedAtoms() {
36
+ return removed;
37
+ },
38
+ };
39
+ }
40
+
41
+ /**
42
+ * Rewrite imports after forward transform.
43
+ *
44
+ * - Adds `import { Stack, Row, ... } from '@mindmatrix/react/atoms'`
45
+ * - Preserves all other imports
46
+ */
47
+ export function rewriteImportsForward(
48
+ programPath: NodePath<t.Program>,
49
+ tracker: ImportTracker,
50
+ ): void {
51
+ const usedAtoms = tracker.getUsedAtoms();
52
+ if (usedAtoms.size === 0) return;
53
+
54
+ // Check if there's already an atom import
55
+ let existingImport: NodePath<t.ImportDeclaration> | null = null;
56
+
57
+ programPath.traverse({
58
+ ImportDeclaration(path: NodePath<t.ImportDeclaration>) {
59
+ if (path.node.source.value === ATOM_IMPORT_SOURCE) {
60
+ existingImport = path;
61
+ }
62
+ },
63
+ });
64
+
65
+ if (existingImport) {
66
+ // Merge new atoms into existing import
67
+ const existing = new Set(
68
+ (existingImport as NodePath<t.ImportDeclaration>).node.specifiers
69
+ .filter((s): s is t.ImportSpecifier => s.type === 'ImportSpecifier')
70
+ .map(s => (s.imported as t.Identifier).name),
71
+ );
72
+
73
+ for (const atom of usedAtoms) {
74
+ if (!existing.has(atom)) {
75
+ (existingImport as NodePath<t.ImportDeclaration>).node.specifiers.push(
76
+ t.importSpecifier(t.identifier(atom), t.identifier(atom)),
77
+ );
78
+ }
79
+ }
80
+ } else {
81
+ // Create new import declaration
82
+ const specifiers = Array.from(usedAtoms)
83
+ .sort()
84
+ .map(name => t.importSpecifier(t.identifier(name), t.identifier(name)));
85
+
86
+ const importDecl = t.importDeclaration(
87
+ specifiers,
88
+ t.stringLiteral(ATOM_IMPORT_SOURCE),
89
+ );
90
+
91
+ // Insert after existing imports
92
+ const lastImport = programPath
93
+ .get('body')
94
+ .filter((p): p is NodePath<t.ImportDeclaration> => p.isImportDeclaration())
95
+ .pop();
96
+
97
+ if (lastImport) {
98
+ lastImport.insertAfter(importDecl);
99
+ } else {
100
+ (programPath.get('body')[0] as NodePath).insertBefore(importDecl);
101
+ }
102
+ }
103
+ }
104
+
105
+ /**
106
+ * Rewrite imports after reverse transform.
107
+ *
108
+ * - Removes unused atoms from `@mindmatrix/react/atoms` import
109
+ * - Removes the import entirely if empty
110
+ * - Ensures `import React from 'react'` exists
111
+ */
112
+ export function rewriteImportsReverse(
113
+ programPath: NodePath<t.Program>,
114
+ tracker: ImportTracker,
115
+ ): void {
116
+ const removed = tracker.getRemovedAtoms();
117
+ if (removed.size === 0) return;
118
+
119
+ programPath.traverse({
120
+ ImportDeclaration(path: NodePath<t.ImportDeclaration>) {
121
+ if (path.node.source.value !== ATOM_IMPORT_SOURCE) return;
122
+
123
+ // Remove specifiers for removed atoms
124
+ path.node.specifiers = path.node.specifiers.filter(spec => {
125
+ if (spec.type !== 'ImportSpecifier') return true;
126
+ const name = (spec.imported as t.Identifier).name;
127
+ return !removed.has(name);
128
+ });
129
+
130
+ // If no specifiers left, remove the entire import
131
+ if (path.node.specifiers.length === 0) {
132
+ path.remove();
133
+ }
134
+ },
135
+ });
136
+
137
+ // Ensure React import exists
138
+ let hasReactImport = false;
139
+ programPath.traverse({
140
+ ImportDeclaration(path: NodePath<t.ImportDeclaration>) {
141
+ if (path.node.source.value === 'react') {
142
+ hasReactImport = true;
143
+ }
144
+ },
145
+ });
146
+
147
+ if (!hasReactImport) {
148
+ const reactImport = t.importDeclaration(
149
+ [t.importDefaultSpecifier(t.identifier('React'))],
150
+ t.stringLiteral('react'),
151
+ );
152
+ (programPath.get('body')[0] as NodePath).insertBefore(reactImport);
153
+ }
154
+ }
155
+
156
+ /**
157
+ * Collect all atoms currently referenced in JSX (for dead-import cleanup).
158
+ */
159
+ export function collectUsedAtoms(programPath: NodePath<t.Program>): Set<string> {
160
+ const used = new Set<string>();
161
+
162
+ programPath.traverse({
163
+ JSXIdentifier(path: NodePath<t.JSXIdentifier>) {
164
+ if (ALL_ATOM_NAMES.has(path.node.name)) {
165
+ used.add(path.node.name);
166
+ }
167
+ },
168
+ });
169
+
170
+ return used;
171
+ }
@@ -0,0 +1,120 @@
1
+ /**
2
+ * Bidirectional React ↔ Workflow Atom Codemod
3
+ *
4
+ * Babel plugin entry point. Dispatches by `direction` option.
5
+ *
6
+ * Usage as Babel plugin:
7
+ * plugins: [['@mindmatrix/react-compiler/codemod', { direction: 'forward' }]]
8
+ *
9
+ * Usage programmatic:
10
+ * import { transform } from '@mindmatrix/react-compiler/codemod';
11
+ * const result = transform(source, { direction: 'forward' });
12
+ */
13
+
14
+ import { transformSync, type TransformOptions } from '@babel/core';
15
+ import type { PluginObj } from '@babel/core';
16
+ // @ts-ignore — no types available for helper-plugin-utils
17
+ import { declare } from '@babel/helper-plugin-utils';
18
+ import { createForwardVisitor } from './forward';
19
+ import { createReverseVisitor } from './reverse';
20
+ import { createStateVisitor, type StateMode } from './state-transform';
21
+ import {
22
+ createImportTracker,
23
+ rewriteImportsForward,
24
+ rewriteImportsReverse,
25
+ } from './import-manager';
26
+
27
+ export interface CodemodOptions {
28
+ /** Transform direction */
29
+ direction: 'forward' | 'reverse';
30
+ /** Add/strip @mm-original annotations */
31
+ annotate?: boolean;
32
+ /** State transform mode (default: 'preserve') */
33
+ stateMode?: StateMode;
34
+ }
35
+
36
+ /**
37
+ * Babel plugin — use via .babelrc or programmatic config.
38
+ */
39
+ export const codemodPlugin = declare(
40
+ (api: any, options: CodemodOptions): PluginObj => {
41
+ api.assertVersion(7);
42
+
43
+ const direction = options.direction || 'forward';
44
+ const stateMode = options.stateMode || 'preserve';
45
+
46
+ return {
47
+ name: 'mm-codemod',
48
+ visitor: {
49
+ Program: {
50
+ enter(programPath) {
51
+ const imports = createImportTracker();
52
+ const stateVisitor = createStateVisitor(stateMode);
53
+
54
+ if (direction === 'forward') {
55
+ const forwardVisitor = createForwardVisitor(imports, {
56
+ annotate: options.annotate ?? true,
57
+ stateMode,
58
+ });
59
+
60
+ // Apply state transforms first
61
+ programPath.traverse(stateVisitor);
62
+ // Then apply element + control flow transforms
63
+ programPath.traverse(forwardVisitor);
64
+ // Finally rewrite imports
65
+ rewriteImportsForward(programPath, imports);
66
+ } else {
67
+ const reverseVisitor = createReverseVisitor(imports, {
68
+ stripAnnotations: !options.annotate,
69
+ });
70
+
71
+ programPath.traverse(reverseVisitor);
72
+ rewriteImportsReverse(programPath, imports);
73
+ }
74
+ },
75
+ },
76
+ },
77
+ };
78
+ },
79
+ );
80
+
81
+ export default codemodPlugin;
82
+
83
+ /**
84
+ * Programmatic transform API.
85
+ *
86
+ * @param source - Source code string
87
+ * @param options - Codemod options
88
+ * @returns Transformed source code
89
+ */
90
+ export function transform(
91
+ source: string,
92
+ options: CodemodOptions,
93
+ ): { code: string; map: any } {
94
+ const babelOptions: TransformOptions = {
95
+ filename: 'file.tsx',
96
+ plugins: [
97
+ ['@babel/plugin-syntax-typescript', { isTSX: true }],
98
+ [codemodPlugin, options],
99
+ ],
100
+ // Preserve formatting as much as possible
101
+ retainLines: true,
102
+ compact: false,
103
+ sourceMaps: true,
104
+ };
105
+
106
+ const result = transformSync(source, babelOptions);
107
+
108
+ if (!result || !result.code) {
109
+ throw new Error('Codemod transform produced no output');
110
+ }
111
+
112
+ return { code: result.code, map: result.map };
113
+ }
114
+
115
+ // Re-export types and utilities for testing
116
+ export { createImportTracker } from './import-manager';
117
+ export { classifyElement, reverseClassifyAtom, extractStaticClasses } from './classify';
118
+ export type { ClassifyResult, ReverseClassifyResult } from './classify';
119
+ export type { MappingRule, ReverseRule } from './rules';
120
+ export { FORWARD_RULES, REVERSE_RULES, ATOM_IMPORT_SOURCE } from './rules';
@@ -0,0 +1,197 @@
1
+ /**
2
+ * Reverse transform: Workflow Atoms → Raw React
3
+ *
4
+ * Babel visitor that:
5
+ * 1. Renames atoms to HTML elements (Stack→div, Row→div, Text→span, etc.)
6
+ * 2. Transforms control flow (<Show> → &&, <Each> → .map())
7
+ * 3. Injects implicit className classes (flex, flex-col)
8
+ * 4. Reads @mm-original annotations for precise restoration
9
+ * 5. Updates imports
10
+ */
11
+
12
+ import type { NodePath, Visitor } from '@babel/traverse';
13
+ import * as t from '@babel/types';
14
+ import { reverseClassifyAtom, addClassesToString } from './classify';
15
+ import { matchShowElement, matchEachElement, showToConditional, eachToMap } from './control-flow';
16
+ import { readAnnotationComment } from './annotation';
17
+ import type { ImportTracker } from './import-manager';
18
+
19
+ export interface ReverseOptions {
20
+ /** Strip @mm-original annotations from output */
21
+ stripAnnotations?: boolean;
22
+ }
23
+
24
+ /**
25
+ * Create the reverse visitor (Atoms → Raw React).
26
+ */
27
+ export function createReverseVisitor(
28
+ imports: ImportTracker,
29
+ _options: ReverseOptions = {},
30
+ ): Visitor {
31
+ return {
32
+ JSXElement: {
33
+ // Use exit so children are processed first (bottom-up)
34
+ exit(path: NodePath<t.JSXElement>) {
35
+ const opening = path.node.openingElement;
36
+ const tag = opening.name;
37
+
38
+ if (tag.type !== 'JSXIdentifier') return;
39
+ const atomName = tag.name;
40
+
41
+ // Handle <Show> → {condition && children}
42
+ const showMatch = matchShowElement(path.node);
43
+ if (showMatch) {
44
+ const replacement = showToConditional(showMatch.condition, showMatch.children);
45
+ imports.removeAtom('Show');
46
+ path.replaceWith(replacement);
47
+ return;
48
+ }
49
+
50
+ // Handle <Each> → {items.map(callback)}
51
+ const eachMatch = matchEachElement(path.node);
52
+ if (eachMatch) {
53
+ const replacement = eachToMap(
54
+ eachMatch.items,
55
+ eachMatch.paramName,
56
+ eachMatch.keyExpression,
57
+ eachMatch.children,
58
+ );
59
+ imports.removeAtom('Each');
60
+ path.replaceWith(replacement);
61
+ return;
62
+ }
63
+
64
+ // Check annotation for precise restoration
65
+ const annotation = readAnnotationComment(path.node);
66
+
67
+ // Build props map for reverse classification
68
+ const propsMap = new Map<string, t.Node>();
69
+ for (const attr of opening.attributes) {
70
+ if (
71
+ attr.type === 'JSXAttribute' &&
72
+ attr.name.type === 'JSXIdentifier'
73
+ ) {
74
+ propsMap.set(attr.name.name, attr.value ?? t.booleanLiteral(true));
75
+ }
76
+ }
77
+
78
+ // Classify the atom back to HTML
79
+ const result = reverseClassifyAtom(atomName, propsMap);
80
+ if (!result) return;
81
+
82
+ // Use annotation tag if available, otherwise use classification
83
+ const targetTag = annotation?.tag ?? result.htmlTag;
84
+
85
+ // Rename the element
86
+ tag.name = targetTag;
87
+ if (path.node.closingElement) {
88
+ (path.node.closingElement.name as t.JSXIdentifier).name = targetTag;
89
+ }
90
+
91
+ // Track removal
92
+ imports.removeAtom(atomName);
93
+
94
+ // Remove props that are encoded in the HTML tag
95
+ if (result.removeProps.length > 0) {
96
+ opening.attributes = opening.attributes.filter(
97
+ attr =>
98
+ !(
99
+ attr.type === 'JSXAttribute' &&
100
+ attr.name.type === 'JSXIdentifier' &&
101
+ result.removeProps.includes(attr.name.name)
102
+ ),
103
+ );
104
+ }
105
+
106
+ // Inject implicit classes into className
107
+ if (result.injectClasses.length > 0) {
108
+ injectClasses(opening, result.injectClasses, annotation?.classes);
109
+ }
110
+
111
+ // Rename event handlers back
112
+ renameEventHandlers(opening);
113
+ },
114
+ },
115
+ };
116
+ }
117
+
118
+ /**
119
+ * Inject classes into a className attribute.
120
+ * If annotation provides original classes, use those instead.
121
+ */
122
+ function injectClasses(
123
+ opening: t.JSXOpeningElement,
124
+ injectClasses: string[],
125
+ _annotationClasses?: string[],
126
+ ): void {
127
+ const classAttr = opening.attributes.find(
128
+ (a): a is t.JSXAttribute =>
129
+ a.type === 'JSXAttribute' &&
130
+ a.name.type === 'JSXIdentifier' &&
131
+ a.name.name === 'className',
132
+ );
133
+
134
+ if (classAttr && classAttr.value) {
135
+ // Modify existing className
136
+ if (classAttr.value.type === 'StringLiteral') {
137
+ classAttr.value = t.stringLiteral(
138
+ addClassesToString(classAttr.value.value, injectClasses),
139
+ );
140
+ return;
141
+ }
142
+
143
+ if (classAttr.value.type === 'JSXExpressionContainer') {
144
+ const expr = classAttr.value.expression;
145
+
146
+ // Template literal
147
+ if (expr.type === 'TemplateLiteral' && expr.quasis.length > 0) {
148
+ const first = expr.quasis[0];
149
+ first.value.raw = addClassesToString(first.value.raw, injectClasses);
150
+ first.value.cooked = addClassesToString(first.value.cooked ?? first.value.raw, injectClasses);
151
+ return;
152
+ }
153
+
154
+ // cn() call
155
+ if (
156
+ expr.type === 'CallExpression' &&
157
+ expr.callee.type === 'Identifier' &&
158
+ expr.callee.name === 'cn' &&
159
+ expr.arguments.length > 0 &&
160
+ expr.arguments[0].type === 'StringLiteral'
161
+ ) {
162
+ expr.arguments[0] = t.stringLiteral(
163
+ addClassesToString((expr.arguments[0] as t.StringLiteral).value, injectClasses),
164
+ );
165
+ return;
166
+ }
167
+ }
168
+ } else {
169
+ // No className — add one with injected classes
170
+ const classValue = injectClasses.join(' ');
171
+ if (classValue) {
172
+ opening.attributes.push(
173
+ t.jsxAttribute(
174
+ t.jsxIdentifier('className'),
175
+ t.stringLiteral(classValue),
176
+ ),
177
+ );
178
+ }
179
+ }
180
+ }
181
+
182
+ /**
183
+ * Rename atom event handlers back to HTML conventions.
184
+ */
185
+ function renameEventHandlers(opening: t.JSXOpeningElement): void {
186
+ const map: Record<string, string> = { onPress: 'onClick' };
187
+
188
+ for (const attr of opening.attributes) {
189
+ if (
190
+ attr.type === 'JSXAttribute' &&
191
+ attr.name.type === 'JSXIdentifier' &&
192
+ attr.name.name in map
193
+ ) {
194
+ attr.name.name = map[attr.name.name];
195
+ }
196
+ }
197
+ }
@@ -0,0 +1,174 @@
1
+ /**
2
+ * Bidirectional element mapping rules.
3
+ *
4
+ * Each rule defines:
5
+ * - htmlTag: The raw React element tag name
6
+ * - atom: The workflow atom component name
7
+ * - classSignal: Tailwind classes that trigger this mapping (forward)
8
+ * - implicitClasses: Classes that are implicit in the atom and should be
9
+ * removed on forward / added on reverse
10
+ * - props: Additional props to set on forward / remove on reverse
11
+ */
12
+
13
+ export interface MappingRule {
14
+ htmlTag: string;
15
+ atom: string;
16
+ /** Classes in className that signal this atom (forward classification) */
17
+ classSignal?: string[];
18
+ /** Classes implicit in the atom — removed on forward, injected on reverse */
19
+ implicitClasses?: string[];
20
+ /** Props added on forward, removed on reverse */
21
+ props?: Record<string, string | boolean>;
22
+ /** Priority when multiple rules match (higher wins) */
23
+ priority?: number;
24
+ }
25
+
26
+ /**
27
+ * Forward rules: HTML element → Atom
28
+ * Ordered by specificity — more specific signals checked first.
29
+ */
30
+ export const FORWARD_RULES: MappingRule[] = [
31
+ // div with flex-col → Stack
32
+ {
33
+ htmlTag: 'div',
34
+ atom: 'Stack',
35
+ classSignal: ['flex-col'],
36
+ implicitClasses: ['flex', 'flex-col'],
37
+ priority: 20,
38
+ },
39
+ // div with grid → Grid
40
+ {
41
+ htmlTag: 'div',
42
+ atom: 'Grid',
43
+ classSignal: ['grid'],
44
+ implicitClasses: ['grid'],
45
+ priority: 20,
46
+ },
47
+ // div with flex (no flex-col) → Row
48
+ {
49
+ htmlTag: 'div',
50
+ atom: 'Row',
51
+ classSignal: ['flex'],
52
+ implicitClasses: ['flex'],
53
+ priority: 10,
54
+ },
55
+ // div with no layout signal → Stack (default container)
56
+ {
57
+ htmlTag: 'div',
58
+ atom: 'Stack',
59
+ classSignal: [],
60
+ implicitClasses: [],
61
+ priority: 0,
62
+ },
63
+
64
+ // Semantic HTML
65
+ { htmlTag: 'main', atom: 'Stack', implicitClasses: [] },
66
+ { htmlTag: 'aside', atom: 'Stack', implicitClasses: [] },
67
+ { htmlTag: 'nav', atom: 'Stack', implicitClasses: [] },
68
+ { htmlTag: 'section', atom: 'Section', implicitClasses: [] },
69
+ { htmlTag: 'header', atom: 'Row', implicitClasses: [] },
70
+ { htmlTag: 'footer', atom: 'Row', implicitClasses: [] },
71
+
72
+ // Text elements
73
+ { htmlTag: 'span', atom: 'Text', implicitClasses: [] },
74
+ { htmlTag: 'p', atom: 'Text', implicitClasses: [] },
75
+ { htmlTag: 'label', atom: 'Text', implicitClasses: [] },
76
+ { htmlTag: 'h1', atom: 'Text', implicitClasses: [], props: { variant: 'h1' } },
77
+ { htmlTag: 'h2', atom: 'Text', implicitClasses: [], props: { variant: 'h2' } },
78
+ { htmlTag: 'h3', atom: 'Text', implicitClasses: [], props: { variant: 'h3' } },
79
+ { htmlTag: 'h4', atom: 'Text', implicitClasses: [], props: { variant: 'h4' } },
80
+ { htmlTag: 'h5', atom: 'Text', implicitClasses: [], props: { variant: 'h5' } },
81
+ { htmlTag: 'h6', atom: 'Text', implicitClasses: [], props: { variant: 'h6' } },
82
+
83
+ // Interactive elements
84
+ { htmlTag: 'button', atom: 'Button', implicitClasses: [] },
85
+ { htmlTag: 'a', atom: 'Link', implicitClasses: [] },
86
+
87
+ // Form elements
88
+ { htmlTag: 'input', atom: 'TextInput', implicitClasses: [] },
89
+ {
90
+ htmlTag: 'textarea',
91
+ atom: 'TextInput',
92
+ implicitClasses: [],
93
+ props: { multiline: true },
94
+ },
95
+
96
+ // Media
97
+ { htmlTag: 'img', atom: 'Image', implicitClasses: [] },
98
+ ];
99
+
100
+ /**
101
+ * Reverse rules: Atom → HTML element
102
+ * Derived from forward rules but keyed by atom name.
103
+ */
104
+ export interface ReverseRule {
105
+ atom: string;
106
+ htmlTag: string;
107
+ /** Classes to inject into className on reverse */
108
+ injectClasses?: string[];
109
+ /** Props to remove on reverse (they're encoded in the HTML tag) */
110
+ removeProps?: string[];
111
+ }
112
+
113
+ export const REVERSE_RULES: ReverseRule[] = [
114
+ { atom: 'Stack', htmlTag: 'div', injectClasses: ['flex', 'flex-col'] },
115
+ { atom: 'Row', htmlTag: 'div', injectClasses: ['flex'] },
116
+ { atom: 'Grid', htmlTag: 'div', injectClasses: ['grid'] },
117
+ { atom: 'Text', htmlTag: 'span' },
118
+ { atom: 'Button', htmlTag: 'button' },
119
+ { atom: 'Link', htmlTag: 'a' },
120
+ { atom: 'TextInput', htmlTag: 'input' },
121
+ { atom: 'Image', htmlTag: 'img' },
122
+ { atom: 'Section', htmlTag: 'section' },
123
+ ];
124
+
125
+ /**
126
+ * Atoms that are NOT HTML elements — control flow and structural.
127
+ * These are handled separately by control-flow.ts, not element mapping.
128
+ */
129
+ export const CONTROL_FLOW_ATOMS = new Set([
130
+ 'Show',
131
+ 'Each',
132
+ 'Modal',
133
+ 'Slot',
134
+ 'Divider',
135
+ ]);
136
+
137
+ /**
138
+ * Components that should never be transformed (third-party, icons, etc.)
139
+ */
140
+ export const PASSTHROUGH_COMPONENTS = new Set([
141
+ // Framer Motion
142
+ 'AnimatePresence',
143
+ // TanStack
144
+ 'useVirtualizer',
145
+ // Fragment
146
+ 'Fragment',
147
+ ]);
148
+
149
+ /**
150
+ * Atom import source
151
+ */
152
+ export const ATOM_IMPORT_SOURCE = '@mindmatrix/react/atoms';
153
+
154
+ /**
155
+ * Atoms that may appear in import but are control-flow, not elements.
156
+ */
157
+ export const ALL_ATOM_NAMES = new Set([
158
+ 'Stack', 'Row', 'Grid', 'Text', 'Button', 'Link',
159
+ 'TextInput', 'Image', 'Section',
160
+ 'Show', 'Each', 'Modal', 'Slot', 'Divider',
161
+ 'Badge', 'Icon',
162
+ ]);
163
+
164
+ /**
165
+ * Text variant → heading tag mapping for reverse.
166
+ */
167
+ export const TEXT_VARIANT_TO_TAG: Record<string, string> = {
168
+ h1: 'h1',
169
+ h2: 'h2',
170
+ h3: 'h3',
171
+ h4: 'h4',
172
+ h5: 'h5',
173
+ h6: 'h6',
174
+ };