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

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