@mmapp/react-compiler 0.1.0-alpha.1 → 0.1.0-alpha.5

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 (265) hide show
  1. package/ATOM-PIPELINE.md +144 -0
  2. package/README.md +88 -40
  3. package/dist/babel/index.js +2814 -277
  4. package/dist/babel/index.mjs +2 -2
  5. package/dist/chunk-3USIFFE4.mjs +2190 -0
  6. package/dist/chunk-45YMGEVT.mjs +186 -0
  7. package/dist/chunk-4FN2AISW.mjs +148 -0
  8. package/dist/chunk-4OPI5L7G.mjs +2593 -0
  9. package/dist/chunk-4RYTKOOJ.mjs +186 -0
  10. package/dist/chunk-52XHYD2V.mjs +214 -0
  11. package/dist/chunk-5GUFFFGL.mjs +148 -0
  12. package/dist/chunk-5RKTOVR5.mjs +244 -0
  13. package/dist/chunk-5YDMOO4X.mjs +214 -0
  14. package/dist/chunk-64ZWEMLJ.mjs +148 -0
  15. package/dist/chunk-6XP4KSWQ.mjs +2190 -0
  16. package/dist/chunk-72QWL54I.mjs +175 -0
  17. package/dist/chunk-7B4TRI7C.mjs +4835 -0
  18. package/dist/chunk-7ZKGHTNB.mjs +4952 -0
  19. package/dist/chunk-CIESM3BP.mjs +33 -0
  20. package/dist/chunk-DE3ZGQAC.mjs +148 -0
  21. package/dist/chunk-DMCY3BBG.mjs +1933 -0
  22. package/dist/chunk-DPIK3PJS.mjs +244 -0
  23. package/dist/chunk-E5IVH4RE.mjs +186 -0
  24. package/dist/chunk-E6FZNUR5.mjs +4953 -0
  25. package/dist/chunk-EJRBDQDP.mjs +2607 -0
  26. package/dist/chunk-ELO4TXJL.mjs +186 -0
  27. package/dist/chunk-EO6SYNCG.mjs +175 -0
  28. package/dist/chunk-FKRO52XH.mjs +3446 -0
  29. package/dist/chunk-FL4YAKU6.mjs +4941 -0
  30. package/dist/chunk-FYT47UBU.mjs +5076 -0
  31. package/dist/chunk-GCLGPOJZ.mjs +148 -0
  32. package/dist/chunk-GXB4JOP7.mjs +5072 -0
  33. package/dist/chunk-HFXOUMTD.mjs +175 -0
  34. package/dist/chunk-HWIZ47US.mjs +214 -0
  35. package/dist/chunk-IB7MNPQL.mjs +4953 -0
  36. package/dist/chunk-ICSIHQCG.mjs +148 -0
  37. package/dist/chunk-J7JUAHS4.mjs +186 -0
  38. package/dist/chunk-JLA5VNQ3.mjs +186 -0
  39. package/dist/chunk-JQLWFCTM.mjs +214 -0
  40. package/dist/chunk-KFJJCQAL.mjs +148 -0
  41. package/dist/chunk-KJUIIEQE.mjs +186 -0
  42. package/dist/chunk-KNWTHRVQ.mjs +175 -0
  43. package/dist/chunk-KSG4XSZF.mjs +175 -0
  44. package/dist/chunk-LF5N6DOU.mjs +175 -0
  45. package/dist/chunk-LJQCM2IM.mjs +214 -0
  46. package/dist/chunk-NTB7OEX2.mjs +2918 -0
  47. package/dist/chunk-NW6555WJ.mjs +186 -0
  48. package/dist/chunk-OMZE6VLQ.mjs +214 -0
  49. package/dist/chunk-OPJKP747.mjs +7506 -0
  50. package/dist/chunk-P4BR7WVO.mjs +2190 -0
  51. package/dist/chunk-QQHVYH2X.mjs +244 -0
  52. package/dist/chunk-S5QLWLLT.mjs +186 -0
  53. package/dist/chunk-SCWGT2FY.mjs +2190 -0
  54. package/dist/chunk-SMKJUSB3.mjs +2190 -0
  55. package/dist/chunk-THFYE5ZX.mjs +244 -0
  56. package/dist/chunk-VCAY2KGM.mjs +175 -0
  57. package/dist/chunk-WBYMW4NQ.mjs +3450 -0
  58. package/dist/chunk-WECAV6QB.mjs +148 -0
  59. package/dist/chunk-WMKBXUCE.mjs +3228 -0
  60. package/dist/chunk-XAJ5BKKL.mjs +4947 -0
  61. package/dist/chunk-XG2X7AEA.mjs +175 -0
  62. package/dist/chunk-XG7Z23NQ.mjs +148 -0
  63. package/dist/chunk-XWZAOCQ7.mjs +2607 -0
  64. package/dist/chunk-Y6MA7ULW.mjs +148 -0
  65. package/dist/chunk-YMS7Q7LG.mjs +214 -0
  66. package/dist/chunk-ZA37XTGA.mjs +175 -0
  67. package/dist/cli/index.js +13189 -6838
  68. package/dist/cli/index.mjs +140 -22
  69. package/dist/codemod/cli.mjs +1 -1
  70. package/dist/codemod/index.mjs +1 -1
  71. package/dist/config-PL24KEWL.mjs +219 -0
  72. package/dist/dev-server-RmGHIntF.d.mts +113 -0
  73. package/dist/dev-server-RmGHIntF.d.ts +113 -0
  74. package/dist/dev-server.d.mts +1 -1
  75. package/dist/dev-server.d.ts +1 -1
  76. package/dist/dev-server.js +4135 -440
  77. package/dist/dev-server.mjs +5 -5
  78. package/dist/envelope.js +2812 -275
  79. package/dist/envelope.mjs +3 -3
  80. package/dist/index.d.mts +161 -2
  81. package/dist/index.d.ts +161 -2
  82. package/dist/index.js +4429 -428
  83. package/dist/index.mjs +217 -9
  84. package/{src/cli/init.ts → dist/init-7JQMAAXS.mjs} +70 -95
  85. package/dist/init-DQDX3QK6.mjs +369 -0
  86. package/dist/init-EHO4VQ22.mjs +369 -0
  87. package/dist/init-UC3FWPIW.mjs +367 -0
  88. package/dist/init-UNSMVKIK.mjs +366 -0
  89. package/dist/init-UNV5XIDE.mjs +367 -0
  90. package/dist/project-compiler-2P4N4DR7.mjs +10 -0
  91. package/dist/project-compiler-D2LCC27O.mjs +10 -0
  92. package/dist/project-compiler-EJ3GANJE.mjs +10 -0
  93. package/dist/project-compiler-LOQKVRZJ.mjs +10 -0
  94. package/dist/project-compiler-OP2VVGJQ.mjs +10 -0
  95. package/dist/project-compiler-RQ6OQKRM.mjs +10 -0
  96. package/dist/project-compiler-VWNNCHGO.mjs +10 -0
  97. package/dist/project-compiler-XVAAU4C5.mjs +10 -0
  98. package/dist/project-compiler-YES5FGMD.mjs +10 -0
  99. package/dist/project-compiler-ZKMQDLGU.mjs +10 -0
  100. package/dist/project-decompiler-FLXCEJHS.mjs +7 -0
  101. package/dist/project-decompiler-US7GAVIC.mjs +7 -0
  102. package/dist/project-decompiler-VLPR22QF.mjs +7 -0
  103. package/dist/pull-FUS5QYZS.mjs +109 -0
  104. package/dist/pull-LD5ENLGY.mjs +109 -0
  105. package/dist/pull-P44LDRWB.mjs +109 -0
  106. package/dist/testing/index.js +2822 -285
  107. package/dist/testing/index.mjs +2 -2
  108. package/dist/verify-SEIXUGN4.mjs +1833 -0
  109. package/dist/vite/index.js +2815 -278
  110. package/dist/vite/index.mjs +3 -3
  111. package/examples/uber-app/app/admin/fleet.tsx +19 -19
  112. package/package.json +16 -6
  113. package/compile-blueprint-chat.mjs +0 -99
  114. package/compile-blueprint-glass-console.mjs +0 -98
  115. package/compile-chat-defs.mjs +0 -92
  116. package/examples/uber-app/tests/payment.test.tsx +0 -129
  117. package/examples/uber-app/tests/ride-flow.test.tsx +0 -123
  118. package/package.json.backup +0 -86
  119. package/scripts/decompile.ts +0 -226
  120. package/scripts/seed-auth.ts +0 -267
  121. package/scripts/seed-uber.ts +0 -248
  122. package/scripts/validate-uber.ts +0 -119
  123. package/seed-blueprint-chat.mjs +0 -444
  124. package/seed-blueprint-glass-console.mjs +0 -445
  125. package/seed-compiled.mjs +0 -318
  126. package/src/RoundTripValidator.ts +0 -400
  127. package/src/__tests__/atom-rendering-coverage.test.ts +0 -680
  128. package/src/__tests__/auth-module-compilation.test.ts +0 -247
  129. package/src/__tests__/auth-template-compilation.test.ts +0 -589
  130. package/src/__tests__/change-extractor.test.ts +0 -142
  131. package/src/__tests__/cli-pull.test.ts +0 -73
  132. package/src/__tests__/cli-test.test.ts +0 -72
  133. package/src/__tests__/component-extractor.test.ts +0 -331
  134. package/src/__tests__/context-extractor.test.ts +0 -145
  135. package/src/__tests__/decompiler.test.ts +0 -718
  136. package/src/__tests__/define-blueprint.test.ts +0 -133
  137. package/src/__tests__/definition-validator.test.ts +0 -519
  138. package/src/__tests__/during-extractor.test.ts +0 -152
  139. package/src/__tests__/effect-extractor.test.ts +0 -107
  140. package/src/__tests__/event-emission.test.ts +0 -127
  141. package/src/__tests__/examples.test.ts +0 -236
  142. package/src/__tests__/full-blueprint-coverage.test.ts +0 -1221
  143. package/src/__tests__/golden-suite.test.ts +0 -403
  144. package/src/__tests__/grammar-island-extractor.test.ts +0 -289
  145. package/src/__tests__/instance-key.test.ts +0 -82
  146. package/src/__tests__/ir-migration.test.ts +0 -255
  147. package/src/__tests__/lock-file.test.ts +0 -117
  148. package/src/__tests__/model-extractor.test.ts +0 -195
  149. package/src/__tests__/model-field-acl.test.ts +0 -237
  150. package/src/__tests__/model-hooks.test.ts +0 -130
  151. package/src/__tests__/model-ref-resolution.test.ts +0 -268
  152. package/src/__tests__/model-roundtrip.test.ts +0 -502
  153. package/src/__tests__/model-runtime.test.ts +0 -112
  154. package/src/__tests__/model-transitions.test.ts +0 -183
  155. package/src/__tests__/nrt-action-trace.test.ts +0 -391
  156. package/src/__tests__/pipeline-hardening.test.ts +0 -413
  157. package/src/__tests__/project-compiler.test.ts +0 -546
  158. package/src/__tests__/project-decompiler.test.ts +0 -343
  159. package/src/__tests__/query-compilation.test.ts +0 -145
  160. package/src/__tests__/round-trip/PLAN.md +0 -158
  161. package/src/__tests__/round-trip/README.md +0 -52
  162. package/src/__tests__/round-trip/RESULTS.md +0 -86
  163. package/src/__tests__/round-trip/fixtures/data-heavy/main.workflow.tsx +0 -55
  164. package/src/__tests__/round-trip/fixtures/data-heavy/mm.config.ts +0 -11
  165. package/src/__tests__/round-trip/fixtures/data-heavy/models/contact.ts +0 -54
  166. package/src/__tests__/round-trip/fixtures/full-workflow/main.workflow.tsx +0 -79
  167. package/src/__tests__/round-trip/fixtures/full-workflow/mm.config.ts +0 -12
  168. package/src/__tests__/round-trip/fixtures/full-workflow/models/order.ts +0 -50
  169. package/src/__tests__/round-trip/fixtures/simple-crud/main.workflow.tsx +0 -25
  170. package/src/__tests__/round-trip/fixtures/simple-crud/mm.config.ts +0 -11
  171. package/src/__tests__/round-trip/fixtures/simple-crud/models/task.ts +0 -32
  172. package/src/__tests__/round-trip/fixtures/view-heavy/main.workflow.tsx +0 -79
  173. package/src/__tests__/round-trip/fixtures/view-heavy/mm.config.ts +0 -10
  174. package/src/__tests__/round-trip/round-trip.test.ts +0 -2598
  175. package/src/__tests__/round-trip-ir.test.ts +0 -300
  176. package/src/__tests__/round-trip.test.ts +0 -1212
  177. package/src/__tests__/route-merging.test.ts +0 -372
  178. package/src/__tests__/router-composition.test.ts +0 -489
  179. package/src/__tests__/router-extractor.test.ts +0 -176
  180. package/src/__tests__/server-action-extractor.test.ts +0 -128
  181. package/src/__tests__/smart-type-inference.test.ts +0 -365
  182. package/src/__tests__/source-envelope.test.ts +0 -284
  183. package/src/__tests__/source-fidelity.test.ts +0 -516
  184. package/src/__tests__/state-extractor.test.ts +0 -115
  185. package/src/__tests__/strict-mode.test.ts +0 -227
  186. package/src/__tests__/transition-effect-extractor.test.ts +0 -119
  187. package/src/__tests__/transition-extractor.test.ts +0 -68
  188. package/src/__tests__/ts-to-expression.test.ts +0 -462
  189. package/src/__tests__/type-generator.test.ts +0 -201
  190. package/src/__tests__/uber-validation.test.ts +0 -502
  191. package/src/action-compiler.ts +0 -361
  192. package/src/babel/emitters/experience-transform.ts +0 -199
  193. package/src/babel/emitters/ir-to-tsx-emitter.ts +0 -110
  194. package/src/babel/emitters/pure-form-emitter.ts +0 -1023
  195. package/src/babel/emitters/runtime-glue-emitter.ts +0 -39
  196. package/src/babel/extractors/change-extractor.ts +0 -199
  197. package/src/babel/extractors/component-extractor.ts +0 -907
  198. package/src/babel/extractors/computed-extractor.ts +0 -262
  199. package/src/babel/extractors/context-extractor.ts +0 -277
  200. package/src/babel/extractors/during-extractor.ts +0 -295
  201. package/src/babel/extractors/effect-extractor.ts +0 -340
  202. package/src/babel/extractors/event-extractor.ts +0 -235
  203. package/src/babel/extractors/grammar-island-extractor.ts +0 -302
  204. package/src/babel/extractors/model-extractor.ts +0 -1018
  205. package/src/babel/extractors/router-extractor.ts +0 -303
  206. package/src/babel/extractors/server-action-extractor.ts +0 -173
  207. package/src/babel/extractors/server-action-hook-extractor.ts +0 -72
  208. package/src/babel/extractors/server-state-extractor.ts +0 -88
  209. package/src/babel/extractors/state-extractor.ts +0 -214
  210. package/src/babel/extractors/transition-effect-extractor.ts +0 -176
  211. package/src/babel/extractors/transition-extractor.ts +0 -143
  212. package/src/babel/index.ts +0 -24
  213. package/src/babel/transpilers/ts-to-expression.ts +0 -674
  214. package/src/babel/visitor.ts +0 -807
  215. package/src/cli/auth.ts +0 -255
  216. package/src/cli/build.ts +0 -288
  217. package/src/cli/deploy.ts +0 -206
  218. package/src/cli/index.ts +0 -328
  219. package/src/cli/installer.ts +0 -261
  220. package/src/cli/lock-file.ts +0 -94
  221. package/src/cli/mmrc.ts +0 -22
  222. package/src/cli/pull.ts +0 -172
  223. package/src/cli/registry-client.ts +0 -175
  224. package/src/cli/test.ts +0 -397
  225. package/src/cli/type-generator.ts +0 -243
  226. package/src/codemod/__tests__/forward.test.ts +0 -239
  227. package/src/codemod/__tests__/reverse.test.ts +0 -145
  228. package/src/codemod/__tests__/round-trip.test.ts +0 -137
  229. package/src/codemod/annotation.ts +0 -97
  230. package/src/codemod/classify.ts +0 -197
  231. package/src/codemod/cli.ts +0 -207
  232. package/src/codemod/control-flow.ts +0 -409
  233. package/src/codemod/forward.ts +0 -244
  234. package/src/codemod/import-manager.ts +0 -171
  235. package/src/codemod/index.ts +0 -120
  236. package/src/codemod/reverse.ts +0 -197
  237. package/src/codemod/rules.ts +0 -174
  238. package/src/codemod/state-transform.ts +0 -126
  239. package/src/decompiler/ast-builder.ts +0 -538
  240. package/src/decompiler/config-generator.ts +0 -151
  241. package/src/decompiler/index.ts +0 -315
  242. package/src/decompiler/project-decompiler.ts +0 -1776
  243. package/src/decompiler/project.ts +0 -862
  244. package/src/decompiler/split-strategy.ts +0 -140
  245. package/src/decompiler/state-emitter.ts +0 -1053
  246. package/src/decompiler/sx-emitter.ts +0 -318
  247. package/src/decompiler/workspace-hydrator.ts +0 -189
  248. package/src/dev-server.ts +0 -238
  249. package/src/envelope/fs-tree.ts +0 -217
  250. package/src/envelope/source-envelope.ts +0 -264
  251. package/src/envelope.ts +0 -315
  252. package/src/incremental-compiler.ts +0 -401
  253. package/src/index.ts +0 -99
  254. package/src/model-compiler.ts +0 -277
  255. package/src/project-compiler.ts +0 -1629
  256. package/src/route-extractor.ts +0 -333
  257. package/src/testing/index.ts +0 -32
  258. package/src/testing/snapshot.ts +0 -252
  259. package/src/testing/test-utils.ts +0 -226
  260. package/src/types.ts +0 -68
  261. package/src/vite/index.ts +0 -288
  262. package/test-compile.mjs +0 -142
  263. package/tsconfig.json +0 -25
  264. package/tsup.config.ts +0 -23
  265. package/vitest.config.ts +0 -9
@@ -1,171 +0,0 @@
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
- }
@@ -1,120 +0,0 @@
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';
@@ -1,197 +0,0 @@
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
- }
@@ -1,174 +0,0 @@
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
- };