@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.
- package/ATOM-PIPELINE.md +144 -0
- package/README.md +88 -40
- package/dist/babel/index.js +113 -6
- package/dist/babel/index.mjs +2 -2
- package/dist/chunk-3USIFFE4.mjs +2190 -0
- package/dist/chunk-45YMGEVT.mjs +186 -0
- package/dist/chunk-4FN2AISW.mjs +148 -0
- package/dist/chunk-4OPI5L7G.mjs +2593 -0
- package/dist/chunk-4RYTKOOJ.mjs +186 -0
- package/dist/chunk-5RKTOVR5.mjs +244 -0
- package/dist/chunk-5YDMOO4X.mjs +214 -0
- package/dist/chunk-64ZWEMLJ.mjs +148 -0
- package/dist/chunk-6XP4KSWQ.mjs +2190 -0
- package/dist/chunk-72QWL54I.mjs +175 -0
- package/dist/chunk-7B4TRI7C.mjs +4835 -0
- package/dist/chunk-7ZKGHTNB.mjs +4952 -0
- package/dist/chunk-CIESM3BP.mjs +33 -0
- package/dist/chunk-DE3ZGQAC.mjs +148 -0
- package/dist/chunk-DMCY3BBG.mjs +1933 -0
- package/dist/chunk-DPIK3PJS.mjs +244 -0
- package/dist/chunk-E5IVH4RE.mjs +186 -0
- package/dist/chunk-E6FZNUR5.mjs +4953 -0
- package/dist/chunk-EJRBDQDP.mjs +2607 -0
- package/dist/chunk-ELO4TXJL.mjs +186 -0
- package/dist/chunk-FKRO52XH.mjs +3446 -0
- package/dist/chunk-FL4YAKU6.mjs +4941 -0
- package/dist/chunk-FYT47UBU.mjs +5076 -0
- package/dist/chunk-GCLGPOJZ.mjs +148 -0
- package/dist/chunk-GXB4JOP7.mjs +5072 -0
- package/dist/chunk-HFXOUMTD.mjs +175 -0
- package/dist/chunk-HWIZ47US.mjs +214 -0
- package/dist/chunk-IB7MNPQL.mjs +4953 -0
- package/dist/chunk-ICSIHQCG.mjs +148 -0
- package/dist/chunk-JLA5VNQ3.mjs +186 -0
- package/dist/chunk-JQLWFCTM.mjs +214 -0
- package/dist/chunk-KFJJCQAL.mjs +148 -0
- package/dist/chunk-KJUIIEQE.mjs +186 -0
- package/dist/chunk-KNWTHRVQ.mjs +175 -0
- package/dist/chunk-KSG4XSZF.mjs +175 -0
- package/dist/chunk-LF5N6DOU.mjs +175 -0
- package/dist/chunk-LJQCM2IM.mjs +214 -0
- package/dist/chunk-NW6555WJ.mjs +186 -0
- package/dist/chunk-OMZE6VLQ.mjs +214 -0
- package/dist/chunk-P4BR7WVO.mjs +2190 -0
- package/dist/chunk-QQHVYH2X.mjs +244 -0
- package/dist/chunk-S5QLWLLT.mjs +186 -0
- package/dist/chunk-SCWGT2FY.mjs +2190 -0
- package/dist/chunk-SMKJUSB3.mjs +2190 -0
- package/dist/chunk-VCAY2KGM.mjs +175 -0
- package/dist/chunk-WECAV6QB.mjs +148 -0
- package/dist/chunk-WMKBXUCE.mjs +3228 -0
- package/dist/chunk-XAJ5BKKL.mjs +4947 -0
- package/dist/chunk-XG2X7AEA.mjs +175 -0
- package/dist/chunk-XG7Z23NQ.mjs +148 -0
- package/dist/chunk-XWZAOCQ7.mjs +2607 -0
- package/dist/chunk-Y6MA7ULW.mjs +148 -0
- package/dist/chunk-YMS7Q7LG.mjs +214 -0
- package/dist/chunk-ZA37XTGA.mjs +175 -0
- package/dist/cli/index.js +1616 -366
- package/dist/cli/index.mjs +8 -8
- package/dist/codemod/cli.mjs +1 -1
- package/dist/codemod/index.mjs +1 -1
- package/dist/dev-server-RmGHIntF.d.mts +113 -0
- package/dist/dev-server-RmGHIntF.d.ts +113 -0
- package/dist/dev-server.d.mts +1 -1
- package/dist/dev-server.d.ts +1 -1
- package/dist/dev-server.js +982 -53
- package/dist/dev-server.mjs +5 -5
- package/dist/envelope.js +113 -6
- package/dist/envelope.mjs +3 -3
- package/dist/index.d.mts +5 -1
- package/dist/index.d.ts +5 -1
- package/dist/index.js +992 -63
- package/dist/index.mjs +8 -8
- package/{src/cli/init.ts → dist/init-7JQMAAXS.mjs} +70 -95
- package/dist/init-EHO4VQ22.mjs +369 -0
- package/dist/init-UC3FWPIW.mjs +367 -0
- package/dist/init-UNSMVKIK.mjs +366 -0
- package/dist/init-UNV5XIDE.mjs +367 -0
- package/dist/project-compiler-2P4N4DR7.mjs +10 -0
- package/dist/project-compiler-D2LCC27O.mjs +10 -0
- package/dist/project-compiler-EJ3GANJE.mjs +10 -0
- package/dist/project-compiler-LOQKVRZJ.mjs +10 -0
- package/dist/project-compiler-RQ6OQKRM.mjs +10 -0
- package/dist/project-compiler-VWNNCHGO.mjs +10 -0
- package/dist/project-compiler-XVAAU4C5.mjs +10 -0
- package/dist/project-compiler-YES5FGMD.mjs +10 -0
- package/dist/project-compiler-ZKMQDLGU.mjs +10 -0
- package/dist/project-decompiler-FLXCEJHS.mjs +7 -0
- package/dist/project-decompiler-VLPR22QF.mjs +7 -0
- package/dist/pull-FUS5QYZS.mjs +109 -0
- package/dist/pull-LD5ENLGY.mjs +109 -0
- package/dist/testing/index.js +113 -6
- package/dist/testing/index.mjs +2 -2
- package/dist/vite/index.js +113 -6
- package/dist/vite/index.mjs +3 -3
- package/examples/uber-app/app/admin/fleet.tsx +19 -19
- package/package.json +4 -3
- package/compile-blueprint-chat.mjs +0 -99
- package/compile-blueprint-glass-console.mjs +0 -98
- package/compile-chat-defs.mjs +0 -92
- package/examples/uber-app/tests/payment.test.tsx +0 -129
- package/examples/uber-app/tests/ride-flow.test.tsx +0 -123
- package/package.json.backup +0 -86
- package/scripts/decompile.ts +0 -226
- package/scripts/seed-auth.ts +0 -267
- package/scripts/seed-uber.ts +0 -248
- package/scripts/validate-uber.ts +0 -119
- package/seed-blueprint-chat.mjs +0 -444
- package/seed-blueprint-glass-console.mjs +0 -445
- package/seed-compiled.mjs +0 -318
- package/src/RoundTripValidator.ts +0 -400
- package/src/__tests__/atom-rendering-coverage.test.ts +0 -680
- package/src/__tests__/auth-module-compilation.test.ts +0 -247
- package/src/__tests__/auth-template-compilation.test.ts +0 -589
- package/src/__tests__/change-extractor.test.ts +0 -142
- package/src/__tests__/cli-pull.test.ts +0 -73
- package/src/__tests__/cli-test.test.ts +0 -72
- package/src/__tests__/component-extractor.test.ts +0 -331
- package/src/__tests__/context-extractor.test.ts +0 -145
- package/src/__tests__/decompiler.test.ts +0 -718
- package/src/__tests__/define-blueprint.test.ts +0 -133
- package/src/__tests__/definition-validator.test.ts +0 -519
- package/src/__tests__/during-extractor.test.ts +0 -152
- package/src/__tests__/effect-extractor.test.ts +0 -107
- package/src/__tests__/event-emission.test.ts +0 -127
- package/src/__tests__/examples.test.ts +0 -236
- package/src/__tests__/full-blueprint-coverage.test.ts +0 -1221
- package/src/__tests__/golden-suite.test.ts +0 -403
- package/src/__tests__/grammar-island-extractor.test.ts +0 -289
- package/src/__tests__/instance-key.test.ts +0 -82
- package/src/__tests__/ir-migration.test.ts +0 -255
- package/src/__tests__/lock-file.test.ts +0 -117
- package/src/__tests__/model-extractor.test.ts +0 -195
- package/src/__tests__/model-field-acl.test.ts +0 -237
- package/src/__tests__/model-hooks.test.ts +0 -130
- package/src/__tests__/model-ref-resolution.test.ts +0 -268
- package/src/__tests__/model-roundtrip.test.ts +0 -502
- package/src/__tests__/model-runtime.test.ts +0 -112
- package/src/__tests__/model-transitions.test.ts +0 -183
- package/src/__tests__/nrt-action-trace.test.ts +0 -391
- package/src/__tests__/pipeline-hardening.test.ts +0 -413
- package/src/__tests__/project-compiler.test.ts +0 -546
- package/src/__tests__/project-decompiler.test.ts +0 -343
- package/src/__tests__/query-compilation.test.ts +0 -145
- package/src/__tests__/round-trip/PLAN.md +0 -158
- package/src/__tests__/round-trip/README.md +0 -52
- package/src/__tests__/round-trip/RESULTS.md +0 -86
- package/src/__tests__/round-trip/fixtures/data-heavy/main.workflow.tsx +0 -55
- package/src/__tests__/round-trip/fixtures/data-heavy/mm.config.ts +0 -11
- package/src/__tests__/round-trip/fixtures/data-heavy/models/contact.ts +0 -54
- package/src/__tests__/round-trip/fixtures/full-workflow/main.workflow.tsx +0 -79
- package/src/__tests__/round-trip/fixtures/full-workflow/mm.config.ts +0 -12
- package/src/__tests__/round-trip/fixtures/full-workflow/models/order.ts +0 -50
- package/src/__tests__/round-trip/fixtures/simple-crud/main.workflow.tsx +0 -25
- package/src/__tests__/round-trip/fixtures/simple-crud/mm.config.ts +0 -11
- package/src/__tests__/round-trip/fixtures/simple-crud/models/task.ts +0 -32
- package/src/__tests__/round-trip/fixtures/view-heavy/main.workflow.tsx +0 -79
- package/src/__tests__/round-trip/fixtures/view-heavy/mm.config.ts +0 -10
- package/src/__tests__/round-trip/round-trip.test.ts +0 -2598
- package/src/__tests__/round-trip-ir.test.ts +0 -300
- package/src/__tests__/round-trip.test.ts +0 -1212
- package/src/__tests__/route-merging.test.ts +0 -372
- package/src/__tests__/router-composition.test.ts +0 -489
- package/src/__tests__/router-extractor.test.ts +0 -176
- package/src/__tests__/server-action-extractor.test.ts +0 -128
- package/src/__tests__/smart-type-inference.test.ts +0 -365
- package/src/__tests__/source-envelope.test.ts +0 -284
- package/src/__tests__/source-fidelity.test.ts +0 -516
- package/src/__tests__/state-extractor.test.ts +0 -115
- package/src/__tests__/strict-mode.test.ts +0 -227
- package/src/__tests__/transition-effect-extractor.test.ts +0 -119
- package/src/__tests__/transition-extractor.test.ts +0 -68
- package/src/__tests__/ts-to-expression.test.ts +0 -462
- package/src/__tests__/type-generator.test.ts +0 -201
- package/src/__tests__/uber-validation.test.ts +0 -502
- package/src/action-compiler.ts +0 -361
- package/src/babel/emitters/experience-transform.ts +0 -199
- package/src/babel/emitters/ir-to-tsx-emitter.ts +0 -110
- package/src/babel/emitters/pure-form-emitter.ts +0 -1023
- package/src/babel/emitters/runtime-glue-emitter.ts +0 -39
- package/src/babel/extractors/change-extractor.ts +0 -199
- package/src/babel/extractors/component-extractor.ts +0 -907
- package/src/babel/extractors/computed-extractor.ts +0 -262
- package/src/babel/extractors/context-extractor.ts +0 -277
- package/src/babel/extractors/during-extractor.ts +0 -295
- package/src/babel/extractors/effect-extractor.ts +0 -340
- package/src/babel/extractors/event-extractor.ts +0 -235
- package/src/babel/extractors/grammar-island-extractor.ts +0 -302
- package/src/babel/extractors/model-extractor.ts +0 -1018
- package/src/babel/extractors/router-extractor.ts +0 -303
- package/src/babel/extractors/server-action-extractor.ts +0 -173
- package/src/babel/extractors/server-action-hook-extractor.ts +0 -72
- package/src/babel/extractors/server-state-extractor.ts +0 -88
- package/src/babel/extractors/state-extractor.ts +0 -214
- package/src/babel/extractors/transition-effect-extractor.ts +0 -176
- package/src/babel/extractors/transition-extractor.ts +0 -143
- package/src/babel/index.ts +0 -24
- package/src/babel/transpilers/ts-to-expression.ts +0 -674
- package/src/babel/visitor.ts +0 -807
- package/src/cli/auth.ts +0 -255
- package/src/cli/build.ts +0 -288
- package/src/cli/deploy.ts +0 -206
- package/src/cli/index.ts +0 -328
- package/src/cli/installer.ts +0 -261
- package/src/cli/lock-file.ts +0 -94
- package/src/cli/mmrc.ts +0 -22
- package/src/cli/pull.ts +0 -172
- package/src/cli/registry-client.ts +0 -175
- package/src/cli/test.ts +0 -397
- package/src/cli/type-generator.ts +0 -243
- package/src/codemod/__tests__/forward.test.ts +0 -239
- package/src/codemod/__tests__/reverse.test.ts +0 -145
- package/src/codemod/__tests__/round-trip.test.ts +0 -137
- package/src/codemod/annotation.ts +0 -97
- package/src/codemod/classify.ts +0 -197
- package/src/codemod/cli.ts +0 -207
- package/src/codemod/control-flow.ts +0 -409
- package/src/codemod/forward.ts +0 -244
- package/src/codemod/import-manager.ts +0 -171
- package/src/codemod/index.ts +0 -120
- package/src/codemod/reverse.ts +0 -197
- package/src/codemod/rules.ts +0 -174
- package/src/codemod/state-transform.ts +0 -126
- package/src/decompiler/ast-builder.ts +0 -538
- package/src/decompiler/config-generator.ts +0 -151
- package/src/decompiler/index.ts +0 -315
- package/src/decompiler/project-decompiler.ts +0 -1776
- package/src/decompiler/project.ts +0 -862
- package/src/decompiler/split-strategy.ts +0 -140
- package/src/decompiler/state-emitter.ts +0 -1053
- package/src/decompiler/sx-emitter.ts +0 -318
- package/src/decompiler/workspace-hydrator.ts +0 -189
- package/src/dev-server.ts +0 -238
- package/src/envelope/fs-tree.ts +0 -217
- package/src/envelope/source-envelope.ts +0 -264
- package/src/envelope.ts +0 -315
- package/src/incremental-compiler.ts +0 -401
- package/src/index.ts +0 -99
- package/src/model-compiler.ts +0 -277
- package/src/project-compiler.ts +0 -1629
- package/src/route-extractor.ts +0 -333
- package/src/testing/index.ts +0 -32
- package/src/testing/snapshot.ts +0 -252
- package/src/testing/test-utils.ts +0 -226
- package/src/types.ts +0 -68
- package/src/vite/index.ts +0 -288
- package/test-compile.mjs +0 -142
- package/tsconfig.json +0 -25
- package/tsup.config.ts +0 -23
- 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
|
-
}
|
package/src/codemod/index.ts
DELETED
|
@@ -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';
|
package/src/codemod/reverse.ts
DELETED
|
@@ -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
|
-
}
|
package/src/codemod/rules.ts
DELETED
|
@@ -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
|
-
};
|