@alepha/devtools 0.16.1 → 0.19.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (236) hide show
  1. package/README.md +1 -5
  2. package/dist/index.browser.js +224 -0
  3. package/dist/index.browser.js.map +1 -0
  4. package/dist/index.d.ts +349 -321
  5. package/dist/index.js +293 -186
  6. package/dist/index.js.map +1 -1
  7. package/package.json +30 -23
  8. package/src/assets.ts +6 -0
  9. package/src/{api/entities → entities}/logs.ts +2 -2
  10. package/src/index.browser.ts +11 -0
  11. package/src/index.shared.ts +15 -0
  12. package/src/index.ts +11 -37
  13. package/src/{api/providers → providers}/DevToolsMetadataProvider.ts +84 -47
  14. package/src/providers/DevToolsProvider.ts +280 -0
  15. package/src/{api/schemas → schemas}/DevActionMetadata.ts +8 -0
  16. package/src/{api/schemas → schemas}/DevEntityMetadata.ts +3 -0
  17. package/src/{api/schemas → schemas}/DevMetadata.ts +13 -2
  18. package/src/{api/schemas → schemas}/DevPageMetadata.ts +3 -0
  19. package/src/{api/schemas → schemas}/DevTopicMetadata.ts +1 -0
  20. package/src/ui/AppRouter.tsx +55 -59
  21. package/src/ui/components/DevLayout.tsx +104 -84
  22. package/src/ui/components/configuration/ConfigAtoms.page.tsx +5 -0
  23. package/src/ui/components/configuration/ConfigAtoms.tsx +511 -0
  24. package/src/ui/components/configuration/ConfigEnv.page.tsx +5 -0
  25. package/src/ui/components/configuration/ConfigEnv.tsx +230 -0
  26. package/src/ui/components/configuration/DevConfiguration.tsx +36 -0
  27. package/src/ui/components/configuration/index.ts +3 -0
  28. package/src/ui/components/dashboard/DevDashboard.tsx +482 -0
  29. package/src/ui/components/database/DatabaseEditor.page.tsx +23 -0
  30. package/src/ui/components/database/DatabaseEditor.tsx +399 -0
  31. package/src/ui/components/database/DatabaseErd.page.tsx +28 -0
  32. package/src/ui/components/database/DatabaseErd.tsx +107 -0
  33. package/src/ui/components/database/DevDatabase.tsx +36 -0
  34. package/src/ui/components/database/EntityNode.tsx +83 -0
  35. package/src/ui/components/explorer/DevExplorer.tsx +351 -0
  36. package/src/ui/components/explorer/ExplorerTree.tsx +178 -0
  37. package/src/ui/components/explorer/panels/DevPanelAction.tsx +499 -0
  38. package/src/ui/components/explorer/panels/DevPanelCache.tsx +73 -0
  39. package/src/ui/components/explorer/panels/DevPanelPage.tsx +96 -0
  40. package/src/ui/components/explorer/panels/DevPanelQueue.tsx +51 -0
  41. package/src/ui/components/explorer/panels/DevPanelTopic.tsx +56 -0
  42. package/src/ui/components/explorer/panels/index.ts +5 -0
  43. package/src/ui/components/graph/DevDependencyGraph.tsx +35 -60
  44. package/src/ui/components/graph/GraphControls.tsx +10 -11
  45. package/src/ui/components/graph/NodeDetails.tsx +22 -29
  46. package/src/ui/components/graph/ProviderNode.tsx +4 -4
  47. package/src/ui/components/graph/helpers.ts +1 -1
  48. package/src/ui/components/logs/DevLogs.tsx +661 -0
  49. package/src/ui/components/logs/index.ts +1 -0
  50. package/src/ui/components/shared/TreeView.tsx +189 -0
  51. package/src/ui/main.css +17 -0
  52. package/src/ui/main.ts +2 -6
  53. package/LICENSE +0 -21
  54. package/assets/devtools/actions.html +0 -21
  55. package/assets/devtools/actions.html.br +0 -0
  56. package/assets/devtools/actions.html.gz +0 -0
  57. package/assets/devtools/asset.BZV40eAE.css +0 -1
  58. package/assets/devtools/asset.BZV40eAE.css.br +0 -0
  59. package/assets/devtools/asset.BZV40eAE.css.gz +0 -0
  60. package/assets/devtools/asset.CBnMq2vO.css +0 -1
  61. package/assets/devtools/asset.CBnMq2vO.css.br +0 -0
  62. package/assets/devtools/asset.CBnMq2vO.css.gz +0 -0
  63. package/assets/devtools/atoms.html +0 -21
  64. package/assets/devtools/atoms.html.br +0 -0
  65. package/assets/devtools/atoms.html.gz +0 -0
  66. package/assets/devtools/caches.html +0 -21
  67. package/assets/devtools/caches.html.br +0 -0
  68. package/assets/devtools/caches.html.gz +0 -0
  69. package/assets/devtools/chunk.6INqNjF0.js +0 -1
  70. package/assets/devtools/chunk.6INqNjF0.js.br +0 -0
  71. package/assets/devtools/chunk.6INqNjF0.js.gz +0 -0
  72. package/assets/devtools/chunk.9vpWpXSF.js +0 -1
  73. package/assets/devtools/chunk.9vpWpXSF.js.br +0 -0
  74. package/assets/devtools/chunk.9vpWpXSF.js.gz +0 -0
  75. package/assets/devtools/chunk.B4peH6PS.js +0 -1
  76. package/assets/devtools/chunk.B4peH6PS.js.br +0 -0
  77. package/assets/devtools/chunk.B4peH6PS.js.gz +0 -0
  78. package/assets/devtools/chunk.B8CNjZzU.js +0 -1
  79. package/assets/devtools/chunk.B8CNjZzU.js.br +0 -0
  80. package/assets/devtools/chunk.B8CNjZzU.js.gz +0 -0
  81. package/assets/devtools/chunk.Bgd10SVI.js +0 -1
  82. package/assets/devtools/chunk.Bgd10SVI.js.br +0 -0
  83. package/assets/devtools/chunk.Bgd10SVI.js.gz +0 -0
  84. package/assets/devtools/chunk.BjFrJKj1.js +0 -1
  85. package/assets/devtools/chunk.BjFrJKj1.js.br +0 -2
  86. package/assets/devtools/chunk.BjFrJKj1.js.gz +0 -0
  87. package/assets/devtools/chunk.BlqFPyLh.js +0 -1
  88. package/assets/devtools/chunk.BlqFPyLh.js.br +0 -0
  89. package/assets/devtools/chunk.BlqFPyLh.js.gz +0 -0
  90. package/assets/devtools/chunk.BqBNmfN9.js +0 -1
  91. package/assets/devtools/chunk.BqBNmfN9.js.br +0 -0
  92. package/assets/devtools/chunk.BqBNmfN9.js.gz +0 -0
  93. package/assets/devtools/chunk.Bt0_vkJm.js +0 -2
  94. package/assets/devtools/chunk.Bt0_vkJm.js.br +0 -0
  95. package/assets/devtools/chunk.Bt0_vkJm.js.gz +0 -0
  96. package/assets/devtools/chunk.C3GuU4pz.js +0 -2
  97. package/assets/devtools/chunk.C3GuU4pz.js.br +0 -0
  98. package/assets/devtools/chunk.C3GuU4pz.js.gz +0 -0
  99. package/assets/devtools/chunk.CGwoN_Mo.js +0 -1
  100. package/assets/devtools/chunk.CGwoN_Mo.js.br +0 -0
  101. package/assets/devtools/chunk.CGwoN_Mo.js.gz +0 -0
  102. package/assets/devtools/chunk.CJCvhHA7.js +0 -1
  103. package/assets/devtools/chunk.CJCvhHA7.js.br +0 -2
  104. package/assets/devtools/chunk.CJCvhHA7.js.gz +0 -0
  105. package/assets/devtools/chunk.CKr2VE6v.js +0 -1
  106. package/assets/devtools/chunk.CKr2VE6v.js.br +0 -0
  107. package/assets/devtools/chunk.CKr2VE6v.js.gz +0 -0
  108. package/assets/devtools/chunk.CLvTwbkw.js +0 -1
  109. package/assets/devtools/chunk.CLvTwbkw.js.br +0 -0
  110. package/assets/devtools/chunk.CLvTwbkw.js.gz +0 -0
  111. package/assets/devtools/chunk.CR13dZhE.js +0 -7
  112. package/assets/devtools/chunk.CR13dZhE.js.br +0 -0
  113. package/assets/devtools/chunk.CR13dZhE.js.gz +0 -0
  114. package/assets/devtools/chunk.C_C-cVqs.js +0 -1
  115. package/assets/devtools/chunk.C_C-cVqs.js.br +0 -1
  116. package/assets/devtools/chunk.C_C-cVqs.js.gz +0 -0
  117. package/assets/devtools/chunk.CjevPbPy.js +0 -1
  118. package/assets/devtools/chunk.CjevPbPy.js.br +0 -0
  119. package/assets/devtools/chunk.CjevPbPy.js.gz +0 -0
  120. package/assets/devtools/chunk.CkNMZqAe.js +0 -1
  121. package/assets/devtools/chunk.CkNMZqAe.js.br +0 -0
  122. package/assets/devtools/chunk.CkNMZqAe.js.gz +0 -0
  123. package/assets/devtools/chunk.Cl1Mlnqx.js +0 -1
  124. package/assets/devtools/chunk.Cl1Mlnqx.js.br +0 -0
  125. package/assets/devtools/chunk.Cl1Mlnqx.js.gz +0 -0
  126. package/assets/devtools/chunk.CyY8OGdZ.js +0 -1
  127. package/assets/devtools/chunk.CyY8OGdZ.js.br +0 -0
  128. package/assets/devtools/chunk.CyY8OGdZ.js.gz +0 -0
  129. package/assets/devtools/chunk.Cyx9kLqD.js +0 -1
  130. package/assets/devtools/chunk.Cyx9kLqD.js.br +0 -0
  131. package/assets/devtools/chunk.Cyx9kLqD.js.gz +0 -0
  132. package/assets/devtools/chunk.D1MGgxUI.js +0 -1
  133. package/assets/devtools/chunk.D1MGgxUI.js.br +0 -0
  134. package/assets/devtools/chunk.D1MGgxUI.js.gz +0 -0
  135. package/assets/devtools/chunk.D5Ci-dwk.js +0 -1
  136. package/assets/devtools/chunk.D5Ci-dwk.js.br +0 -0
  137. package/assets/devtools/chunk.D5Ci-dwk.js.gz +0 -0
  138. package/assets/devtools/chunk.DFrWQW5x.js +0 -9
  139. package/assets/devtools/chunk.DFrWQW5x.js.br +0 -0
  140. package/assets/devtools/chunk.DFrWQW5x.js.gz +0 -0
  141. package/assets/devtools/chunk.DaVlli3f.js +0 -1
  142. package/assets/devtools/chunk.DaVlli3f.js.br +0 -0
  143. package/assets/devtools/chunk.DaVlli3f.js.gz +0 -0
  144. package/assets/devtools/chunk.DdyBCs50.js +0 -1
  145. package/assets/devtools/chunk.DdyBCs50.js.br +0 -0
  146. package/assets/devtools/chunk.DdyBCs50.js.gz +0 -0
  147. package/assets/devtools/chunk.Dl0THvrP.js +0 -1
  148. package/assets/devtools/chunk.Dl0THvrP.js.br +0 -0
  149. package/assets/devtools/chunk.Dl0THvrP.js.gz +0 -0
  150. package/assets/devtools/chunk.DwUNDm68.js +0 -1
  151. package/assets/devtools/chunk.DwUNDm68.js.br +0 -0
  152. package/assets/devtools/chunk.DwUNDm68.js.gz +0 -0
  153. package/assets/devtools/chunk.DzDkh4C6.js +0 -1
  154. package/assets/devtools/chunk.DzDkh4C6.js.br +0 -0
  155. package/assets/devtools/chunk.DzDkh4C6.js.gz +0 -0
  156. package/assets/devtools/chunk.QTExp4CY.js +0 -1
  157. package/assets/devtools/chunk.QTExp4CY.js.br +0 -0
  158. package/assets/devtools/chunk.QTExp4CY.js.gz +0 -0
  159. package/assets/devtools/chunk.ReCPcJln.js +0 -1
  160. package/assets/devtools/chunk.ReCPcJln.js.br +0 -0
  161. package/assets/devtools/chunk.ReCPcJln.js.gz +0 -0
  162. package/assets/devtools/chunk.UEhIKOMY.js +0 -1
  163. package/assets/devtools/chunk.UEhIKOMY.js.br +0 -0
  164. package/assets/devtools/chunk.UEhIKOMY.js.gz +0 -0
  165. package/assets/devtools/chunk.mWQqK3dU.js +0 -1
  166. package/assets/devtools/chunk.mWQqK3dU.js.br +0 -0
  167. package/assets/devtools/chunk.mWQqK3dU.js.gz +0 -0
  168. package/assets/devtools/chunk.uyVen0u2.js +0 -1
  169. package/assets/devtools/chunk.uyVen0u2.js.br +0 -0
  170. package/assets/devtools/chunk.uyVen0u2.js.gz +0 -0
  171. package/assets/devtools/chunk.yLRX_cUF.js +0 -1
  172. package/assets/devtools/chunk.yLRX_cUF.js.br +0 -0
  173. package/assets/devtools/chunk.yLRX_cUF.js.gz +0 -0
  174. package/assets/devtools/chunk.zuZxBYZg.js +0 -1
  175. package/assets/devtools/chunk.zuZxBYZg.js.br +0 -0
  176. package/assets/devtools/chunk.zuZxBYZg.js.gz +0 -0
  177. package/assets/devtools/db.html +0 -21
  178. package/assets/devtools/db.html.br +0 -0
  179. package/assets/devtools/db.html.gz +0 -0
  180. package/assets/devtools/entry.Cry3rxEI.js +0 -79
  181. package/assets/devtools/entry.Cry3rxEI.js.br +0 -0
  182. package/assets/devtools/entry.Cry3rxEI.js.gz +0 -0
  183. package/assets/devtools/env.html +0 -21
  184. package/assets/devtools/env.html.br +0 -0
  185. package/assets/devtools/env.html.gz +0 -0
  186. package/assets/devtools/graph.html +0 -22
  187. package/assets/devtools/graph.html.br +0 -0
  188. package/assets/devtools/graph.html.gz +0 -0
  189. package/assets/devtools/index.html +0 -21
  190. package/assets/devtools/index.html.br +0 -0
  191. package/assets/devtools/index.html.gz +0 -0
  192. package/assets/devtools/logs.html +0 -21
  193. package/assets/devtools/logs.html.br +0 -0
  194. package/assets/devtools/logs.html.gz +0 -0
  195. package/assets/devtools/queues.html +0 -21
  196. package/assets/devtools/queues.html.br +0 -0
  197. package/assets/devtools/queues.html.gz +0 -0
  198. package/assets/devtools/topics.html +0 -21
  199. package/assets/devtools/topics.html.br +0 -0
  200. package/assets/devtools/topics.html.gz +0 -0
  201. package/src/api/DevToolsProvider.ts +0 -157
  202. package/src/api/providers/DevToolsDatabaseProvider.ts +0 -27
  203. package/src/api/repositories/LogRepository.ts +0 -8
  204. package/src/api/schemas/DevCommandMetadata.ts +0 -9
  205. package/src/ui/components/DevAtomsViewer.tsx +0 -637
  206. package/src/ui/components/DevCacheInspector.tsx +0 -423
  207. package/src/ui/components/DevDashboard.tsx +0 -38
  208. package/src/ui/components/DevEnvExplorer.tsx +0 -462
  209. package/src/ui/components/DevLogViewer.tsx +0 -252
  210. package/src/ui/components/DevQueueMonitor.tsx +0 -51
  211. package/src/ui/components/DevTopicsViewer.tsx +0 -686
  212. package/src/ui/components/actions/ActionGroup.tsx +0 -37
  213. package/src/ui/components/actions/ActionItem.tsx +0 -138
  214. package/src/ui/components/actions/DevActionsExplorer.tsx +0 -132
  215. package/src/ui/components/actions/MethodBadge.tsx +0 -18
  216. package/src/ui/components/actions/SchemaViewer.tsx +0 -21
  217. package/src/ui/components/actions/TryItPanel.tsx +0 -140
  218. package/src/ui/components/actions/constants.ts +0 -7
  219. package/src/ui/components/actions/helpers.ts +0 -18
  220. package/src/ui/components/actions/index.ts +0 -8
  221. package/src/ui/components/db/ColumnBadge.tsx +0 -55
  222. package/src/ui/components/db/DevDbStudio.tsx +0 -485
  223. package/src/ui/components/db/constants.ts +0 -11
  224. package/src/ui/components/db/index.ts +0 -4
  225. package/src/ui/components/db/types.ts +0 -7
  226. package/src/ui/styles.css +0 -1
  227. /package/src/{api/schemas → schemas}/DevAtomMetadata.ts +0 -0
  228. /package/src/{api/schemas → schemas}/DevBucketMetadata.ts +0 -0
  229. /package/src/{api/schemas → schemas}/DevCacheMetadata.ts +0 -0
  230. /package/src/{api/schemas → schemas}/DevEnvMetadata.ts +0 -0
  231. /package/src/{api/schemas → schemas}/DevModuleMetadata.ts +0 -0
  232. /package/src/{api/schemas → schemas}/DevProviderMetadata.ts +0 -0
  233. /package/src/{api/schemas → schemas}/DevQueueMetadata.ts +0 -0
  234. /package/src/{api/schemas → schemas}/DevRealmMetadata.ts +0 -0
  235. /package/src/{api/schemas → schemas}/DevRouteMetadata.ts +0 -0
  236. /package/src/{api/schemas → schemas}/DevSchedulerMetadata.ts +0 -0
@@ -0,0 +1,511 @@
1
+ import type { DevAtomMetadata } from "@alepha/devtools";
2
+ import { devMetadataSchema } from "@alepha/devtools";
3
+ import { ActionButton, Flex, JsonViewer, ui } from "@alepha/ui";
4
+ import {
5
+ Badge,
6
+ NumberInput,
7
+ ScrollArea,
8
+ Switch,
9
+ Text,
10
+ Textarea,
11
+ TextInput,
12
+ } from "@mantine/core";
13
+ import {
14
+ IconAtom,
15
+ IconDeviceFloppy,
16
+ IconRefresh,
17
+ IconSearch,
18
+ } from "@tabler/icons-react";
19
+ import { useInject } from "alepha/react";
20
+ import { HttpClient } from "alepha/server";
21
+ import { useCallback, useEffect, useMemo, useState } from "react";
22
+ import { TreeView, type TreeViewNode } from "../shared/TreeView.tsx";
23
+
24
+ interface AtomTreeNode {
25
+ key: string;
26
+ label: string;
27
+ atom?: DevAtomMetadata;
28
+ children: AtomTreeNode[];
29
+ }
30
+
31
+ const buildAtomTree = (atoms: DevAtomMetadata[]): AtomTreeNode[] => {
32
+ const root: AtomTreeNode = { key: "", label: "", children: [] };
33
+
34
+ for (const atom of atoms) {
35
+ const parts = atom.name.split(".");
36
+ let current = root;
37
+
38
+ for (let i = 0; i < parts.length; i++) {
39
+ const key = parts.slice(0, i + 1).join(".");
40
+ let child = current.children.find((c) => c.key === key);
41
+ if (!child) {
42
+ child = { key, label: parts[i], children: [] };
43
+ current.children.push(child);
44
+ }
45
+ if (i === parts.length - 1) {
46
+ child.atom = atom;
47
+ }
48
+ current = child;
49
+ }
50
+ }
51
+
52
+ return root.children.map(collapseChain);
53
+ };
54
+
55
+ const collapseChain = (node: AtomTreeNode): AtomTreeNode => {
56
+ while (!node.atom && node.children.length === 1) {
57
+ const child = node.children[0];
58
+ node = { ...child, label: `${node.label}.${child.label}` };
59
+ }
60
+ return { ...node, children: node.children.map(collapseChain) };
61
+ };
62
+
63
+ const toTreeViewNode = (node: AtomTreeNode): TreeViewNode => {
64
+ const isLeaf = !!node.atom && node.children.length === 0;
65
+
66
+ return {
67
+ id: node.atom ? `atom:${node.atom.name}` : `folder:${node.key}`,
68
+ label: node.label,
69
+ icon: isLeaf ? (
70
+ <IconAtom
71
+ size={13}
72
+ color="var(--mantine-color-violet-text)"
73
+ style={{ flexShrink: 0 }}
74
+ />
75
+ ) : undefined,
76
+ badge:
77
+ isLeaf && node.atom?.currentValue !== undefined ? (
78
+ <Badge size="xs" variant="light" color="green">
79
+ set
80
+ </Badge>
81
+ ) : undefined,
82
+ children:
83
+ node.children.length > 0 ? node.children.map(toTreeViewNode) : undefined,
84
+ };
85
+ };
86
+
87
+ const SchemaEditor = ({
88
+ schema,
89
+ value,
90
+ onChange,
91
+ }: {
92
+ schema: any;
93
+ value: any;
94
+ onChange: (value: any) => void;
95
+ }) => {
96
+ if (!schema) return null;
97
+
98
+ let actualSchema = schema;
99
+ if (schema.anyOf) {
100
+ const nonNull = schema.anyOf.find((t: any) => t.type !== "null");
101
+ if (nonNull) actualSchema = nonNull;
102
+ }
103
+
104
+ const type = actualSchema.type;
105
+
106
+ if (type === "object" && actualSchema.properties) {
107
+ return (
108
+ <Flex direction="column" gap="sm">
109
+ {Object.entries(actualSchema.properties).map(
110
+ ([key, propSchema]: [string, any]) => (
111
+ <Flex key={key}>
112
+ <Text fz="xs" fw={500} mb={4}>
113
+ {propSchema.title || key}
114
+ {propSchema.description && (
115
+ <Text span fz={10} c="dimmed" ml="xs">
116
+ {propSchema.description}
117
+ </Text>
118
+ )}
119
+ </Text>
120
+ <SchemaEditor
121
+ schema={propSchema}
122
+ value={value?.[key]}
123
+ onChange={(v) => onChange({ ...value, [key]: v })}
124
+ />
125
+ </Flex>
126
+ ),
127
+ )}
128
+ </Flex>
129
+ );
130
+ }
131
+
132
+ if (type === "boolean") {
133
+ return (
134
+ <Switch
135
+ checked={value ?? false}
136
+ onChange={(e) => onChange(e.currentTarget.checked)}
137
+ size="sm"
138
+ />
139
+ );
140
+ }
141
+
142
+ if (type === "number" || type === "integer") {
143
+ return (
144
+ <NumberInput
145
+ size="xs"
146
+ value={value ?? ""}
147
+ onChange={(v) => onChange(v === "" ? undefined : v)}
148
+ allowDecimal={type === "number"}
149
+ />
150
+ );
151
+ }
152
+
153
+ if (actualSchema.enum) {
154
+ return (
155
+ <select
156
+ value={value ?? ""}
157
+ onChange={(e) => onChange(e.target.value || undefined)}
158
+ style={{
159
+ padding: "4px 8px",
160
+ borderRadius: 4,
161
+ border: `1px solid ${ui.colors.border}`,
162
+ backgroundColor: ui.colors.surface,
163
+ color: "inherit",
164
+ fontSize: 12,
165
+ width: "100%",
166
+ }}
167
+ >
168
+ <option value="">Select...</option>
169
+ {actualSchema.enum.map((opt: string) => (
170
+ <option key={opt} value={opt}>
171
+ {opt}
172
+ </option>
173
+ ))}
174
+ </select>
175
+ );
176
+ }
177
+
178
+ return (
179
+ <TextInput
180
+ size="xs"
181
+ value={value ?? ""}
182
+ onChange={(e) => onChange(e.target.value || undefined)}
183
+ />
184
+ );
185
+ };
186
+
187
+ const AtomDetailPanel = ({
188
+ atom,
189
+ onSave,
190
+ }: {
191
+ atom: DevAtomMetadata;
192
+ onSave: (value: any) => void;
193
+ }) => {
194
+ const [editValue, setEditValue] = useState<any>(
195
+ atom.currentValue ?? atom.defaultValue,
196
+ );
197
+ const [jsonMode, setJsonMode] = useState(false);
198
+ const [jsonText, setJsonText] = useState("");
199
+ const [jsonError, setJsonError] = useState<string | null>(null);
200
+
201
+ useEffect(() => {
202
+ const val = atom.currentValue ?? atom.defaultValue;
203
+ setEditValue(val);
204
+ setJsonText(JSON.stringify(val, null, 2));
205
+ setJsonError(null);
206
+ }, [atom]);
207
+
208
+ const handleJsonChange = (text: string) => {
209
+ setJsonText(text);
210
+ try {
211
+ const parsed = JSON.parse(text);
212
+ setEditValue(parsed);
213
+ setJsonError(null);
214
+ } catch {
215
+ setJsonError("Invalid JSON");
216
+ }
217
+ };
218
+
219
+ const hasValue = atom.currentValue !== undefined;
220
+ const schemaType = atom.schema?.type;
221
+ const useJsonByDefault = !schemaType || schemaType === "array";
222
+
223
+ return (
224
+ <Flex direction="column" h="100%">
225
+ {/* Header */}
226
+ <Flex
227
+ px="md"
228
+ py="sm"
229
+ style={{
230
+ borderBottom: `1px solid ${ui.colors.border}`,
231
+ flexShrink: 0,
232
+ }}
233
+ >
234
+ <Flex align="center" gap="sm">
235
+ <IconAtom size={16} opacity={0.5} />
236
+ <Text fz="sm" fw={600} ff="monospace">
237
+ {atom.name}
238
+ </Text>
239
+ <Badge size="xs" variant="light" color={hasValue ? "green" : "gray"}>
240
+ {hasValue ? "has value" : "default"}
241
+ </Badge>
242
+ </Flex>
243
+ {atom.description && (
244
+ <Text fz="xs" c="dimmed" mt={4}>
245
+ {atom.description}
246
+ </Text>
247
+ )}
248
+ </Flex>
249
+
250
+ <ScrollArea style={{ flex: 1 }} p="md">
251
+ <Flex direction="column" gap="lg">
252
+ {/* Current / Default values */}
253
+ <Flex gap="lg" wrap="wrap">
254
+ <Flex style={{ flex: 1, minWidth: 200 }}>
255
+ <Text
256
+ fz={10}
257
+ c="dimmed"
258
+ tt="uppercase"
259
+ fw={600}
260
+ lts={0.5}
261
+ mb="xs"
262
+ >
263
+ Current Value
264
+ </Text>
265
+ {atom.currentValue !== undefined ? (
266
+ <JsonViewer data={atom.currentValue} maxDepth={3} />
267
+ ) : (
268
+ <Text fz="xs" c="dimmed" fs="italic">
269
+ (using default)
270
+ </Text>
271
+ )}
272
+ </Flex>
273
+ <Flex style={{ flex: 1, minWidth: 200 }}>
274
+ <Text
275
+ fz={10}
276
+ c="dimmed"
277
+ tt="uppercase"
278
+ fw={600}
279
+ lts={0.5}
280
+ mb="xs"
281
+ >
282
+ Default Value
283
+ </Text>
284
+ {atom.defaultValue !== undefined ? (
285
+ <JsonViewer data={atom.defaultValue} maxDepth={3} />
286
+ ) : (
287
+ <Text fz="xs" c="dimmed" fs="italic">
288
+ (no default)
289
+ </Text>
290
+ )}
291
+ </Flex>
292
+ </Flex>
293
+
294
+ {/* Editor */}
295
+ <Flex>
296
+ <Flex align="center" justify="space-between" mb="sm">
297
+ <Text fz={10} c="dimmed" tt="uppercase" fw={600} lts={0.5}>
298
+ Edit Value
299
+ </Text>
300
+ <Flex gap="xs" align="center">
301
+ <Switch
302
+ size="xs"
303
+ label="JSON"
304
+ checked={jsonMode || useJsonByDefault}
305
+ onChange={(e) => setJsonMode(e.currentTarget.checked)}
306
+ disabled={useJsonByDefault}
307
+ />
308
+ <ActionButton
309
+ size="sm"
310
+ variant="light"
311
+ tooltip="Reset to default"
312
+ onClick={() => {
313
+ const val = atom.defaultValue;
314
+ setEditValue(val);
315
+ setJsonText(JSON.stringify(val, null, 2));
316
+ setJsonError(null);
317
+ }}
318
+ icon={<IconRefresh size={14} />}
319
+ />
320
+ <ActionButton
321
+ size="xs"
322
+ icon={<IconDeviceFloppy size={14} />}
323
+ onClick={() => onSave(editValue)}
324
+ disabled={jsonMode && !!jsonError}
325
+ >
326
+ Save
327
+ </ActionButton>
328
+ </Flex>
329
+ </Flex>
330
+
331
+ {jsonMode || useJsonByDefault ? (
332
+ <Textarea
333
+ value={jsonText}
334
+ onChange={(e) => handleJsonChange(e.target.value)}
335
+ autosize
336
+ minRows={6}
337
+ maxRows={20}
338
+ error={jsonError ?? undefined}
339
+ styles={{
340
+ input: { fontFamily: "monospace", fontSize: 12 },
341
+ }}
342
+ />
343
+ ) : (
344
+ <Flex
345
+ p="sm"
346
+ style={{
347
+ border: `1px solid ${ui.colors.border}`,
348
+ borderRadius: 8,
349
+ background: ui.colors.surface,
350
+ }}
351
+ >
352
+ <SchemaEditor
353
+ schema={atom.schema}
354
+ value={editValue}
355
+ onChange={setEditValue}
356
+ />
357
+ </Flex>
358
+ )}
359
+ </Flex>
360
+
361
+ {/* Schema info */}
362
+ <Flex>
363
+ <Text fz={10} c="dimmed" tt="uppercase" fw={600} lts={0.5} mb="xs">
364
+ Schema
365
+ </Text>
366
+ <JsonViewer data={atom.schema} maxDepth={2} />
367
+ </Flex>
368
+ </Flex>
369
+ </ScrollArea>
370
+ </Flex>
371
+ );
372
+ };
373
+
374
+ export const ConfigAtoms = () => {
375
+ const http = useInject(HttpClient);
376
+ const [atoms, setAtoms] = useState<DevAtomMetadata[]>([]);
377
+ const [search, setSearch] = useState("");
378
+ const [selectedName, setSelectedName] = useState("");
379
+ const [openNodes, setOpenNodes] = useState<Set<string>>(new Set());
380
+
381
+ const fetchAtoms = useCallback(() => {
382
+ http
383
+ .fetch("/__devtools/api/metadata", {
384
+ schema: { response: devMetadataSchema },
385
+ })
386
+ .then((res) => {
387
+ setAtoms(res.data.atoms);
388
+ // Auto-expand top-level tree nodes
389
+ const topLevelKeys = new Set<string>();
390
+ for (const atom of res.data.atoms) {
391
+ const first = atom.name.split(".")[0];
392
+ topLevelKeys.add(`folder:${first}`);
393
+ }
394
+ setOpenNodes((prev) => (prev.size > 0 ? prev : topLevelKeys));
395
+ })
396
+ .catch(() => {});
397
+ }, [http]);
398
+
399
+ useEffect(() => {
400
+ fetchAtoms();
401
+ }, [fetchAtoms]);
402
+
403
+ const filteredAtoms = useMemo(() => {
404
+ if (!search) return atoms;
405
+ const s = search.toLowerCase();
406
+ return atoms.filter((a) => a.name.toLowerCase().includes(s));
407
+ }, [atoms, search]);
408
+
409
+ const tree = useMemo(() => buildAtomTree(filteredAtoms), [filteredAtoms]);
410
+
411
+ const treeViewNodes = useMemo(() => tree.map(toTreeViewNode), [tree]);
412
+
413
+ const selectedAtom = useMemo(
414
+ () => atoms.find((a) => a.name === selectedName),
415
+ [atoms, selectedName],
416
+ );
417
+
418
+ const handleSave = useCallback(
419
+ async (value: any) => {
420
+ if (!selectedAtom) return;
421
+ try {
422
+ await http.fetch("/__devtools/api/atoms", {
423
+ method: "POST",
424
+ body: JSON.stringify({ name: selectedAtom.name, value }),
425
+ });
426
+ fetchAtoms();
427
+ } catch (error) {
428
+ console.error("Failed to save atom:", error);
429
+ }
430
+ },
431
+ [http, selectedAtom, fetchAtoms],
432
+ );
433
+
434
+ const handleToggle = useCallback((id: string) => {
435
+ setOpenNodes((prev) => {
436
+ const next = new Set(prev);
437
+ if (next.has(id)) next.delete(id);
438
+ else next.add(id);
439
+ return next;
440
+ });
441
+ }, []);
442
+
443
+ const handleSelect = useCallback((id: string) => {
444
+ // id is "atom:<name>"
445
+ const name = id.replace(/^atom:/, "");
446
+ setSelectedName(name);
447
+ }, []);
448
+
449
+ if (atoms.length === 0) {
450
+ return (
451
+ <Flex align="center" justify="center" style={{ flex: 1 }} c="dimmed">
452
+ <Flex direction="column" align="center" gap="xs">
453
+ <IconAtom size={40} opacity={0.3} />
454
+ <Text fz="sm">No atoms found</Text>
455
+ <Text fz="xs" c="dimmed">
456
+ Use $atom primitive to define state atoms
457
+ </Text>
458
+ </Flex>
459
+ </Flex>
460
+ );
461
+ }
462
+
463
+ const selectedId = selectedName ? `atom:${selectedName}` : "";
464
+
465
+ return (
466
+ <Flex style={{ flex: 1, overflow: "hidden" }}>
467
+ {/* Tree sidebar */}
468
+ <Flex
469
+ w={260}
470
+ style={{
471
+ borderRight: `1px solid ${ui.colors.border}`,
472
+ flexShrink: 0,
473
+ display: "flex",
474
+ flexDirection: "column",
475
+ }}
476
+ >
477
+ <Flex p="xs" style={{ flexShrink: 0 }}>
478
+ <TextInput
479
+ placeholder="Search atoms..."
480
+ leftSection={<IconSearch size={14} />}
481
+ size="xs"
482
+ value={search}
483
+ onChange={(e) => setSearch(e.currentTarget.value)}
484
+ />
485
+ </Flex>
486
+ <ScrollArea style={{ flex: 1 }} px="xs" pb="xs">
487
+ <TreeView
488
+ nodes={treeViewNodes}
489
+ selectedId={selectedId}
490
+ openNodes={openNodes}
491
+ onSelect={handleSelect}
492
+ onToggle={handleToggle}
493
+ />
494
+ </ScrollArea>
495
+ </Flex>
496
+
497
+ {/* Detail panel */}
498
+ <Flex style={{ flex: 1, overflow: "hidden" }}>
499
+ {selectedAtom ? (
500
+ <AtomDetailPanel atom={selectedAtom} onSave={handleSave} />
501
+ ) : (
502
+ <Flex align="center" justify="center" h="100%">
503
+ <Text c="dimmed" fz="sm">
504
+ Select an atom to view its details
505
+ </Text>
506
+ </Flex>
507
+ )}
508
+ </Flex>
509
+ </Flex>
510
+ );
511
+ };
@@ -0,0 +1,5 @@
1
+ import { ConfigEnv } from "./ConfigEnv.tsx";
2
+
3
+ const ConfigEnvPage = () => <ConfigEnv />;
4
+
5
+ export default ConfigEnvPage;