@alepha/ui 0.14.1 → 0.14.2

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 (173) hide show
  1. package/dist/admin/AdminAudits-B3EhKhN7.js +3 -0
  2. package/dist/admin/{AdminAudits-CwvH8e8c.js → AdminAudits-DIrCCPk3.js} +3 -2
  3. package/dist/admin/AdminAudits-DIrCCPk3.js.map +1 -0
  4. package/dist/admin/AdminFiles-C8OG4dtD.js +3 -0
  5. package/dist/admin/{AdminFiles-C_w1tb_x.js → AdminFiles-RsL178Ta.js} +2 -2
  6. package/dist/admin/{AdminFiles-C_w1tb_x.js.map → AdminFiles-RsL178Ta.js.map} +1 -1
  7. package/dist/admin/AdminNotifications-BSL4B2fQ.js +3 -0
  8. package/dist/admin/{AdminNotifications-DuYy74AN.js → AdminNotifications-cIbywWKi.js} +2 -2
  9. package/dist/admin/{AdminNotifications-DuYy74AN.js.map → AdminNotifications-cIbywWKi.js.map} +1 -1
  10. package/dist/admin/{AdminParameters-DYg48Jwe.js → AdminParameters-BKObzzpN.js} +1 -1
  11. package/dist/admin/{AdminParameters-YagqWTG3.js → AdminParameters-D-q3Qmhv.js} +2 -2
  12. package/dist/admin/{AdminParameters-YagqWTG3.js.map → AdminParameters-D-q3Qmhv.js.map} +1 -1
  13. package/dist/admin/AdminSessions-DHG9zPfr.js +3 -0
  14. package/dist/admin/{AdminSessions-BCjgJ-93.js → AdminSessions-vOgkrQ2U.js} +3 -2
  15. package/dist/admin/AdminSessions-vOgkrQ2U.js.map +1 -0
  16. package/dist/admin/{AdminUserAudits-B_PUXCKC.js → AdminUserAudits-CSsN1fIC.js} +3 -2
  17. package/dist/admin/AdminUserAudits-CSsN1fIC.js.map +1 -0
  18. package/dist/admin/{AdminUserAudits-D7cTcElL.js → AdminUserAudits-DmAnivo3.js} +1 -1
  19. package/dist/admin/{AdminUserCreate-DzfRbGZ4.js → AdminUserCreate-B72nu-3W.js} +3 -2
  20. package/dist/admin/AdminUserCreate-B72nu-3W.js.map +1 -0
  21. package/dist/admin/{AdminUserCreate-oUA1KDIl.js → AdminUserCreate-DpA13zwj.js} +1 -1
  22. package/dist/admin/{AdminUserDetails-DeTrJm-t.js → AdminUserDetails-BCt8Su-4.js} +3 -2
  23. package/dist/admin/AdminUserDetails-BCt8Su-4.js.map +1 -0
  24. package/dist/admin/{AdminUserDetails-y1H5DW8Y.js → AdminUserDetails-z1y8kJeB.js} +1 -1
  25. package/dist/admin/{AdminUserLayout-CsfrrZkD.js → AdminUserLayout-Ck0GLRE5.js} +3 -2
  26. package/dist/admin/AdminUserLayout-Ck0GLRE5.js.map +1 -0
  27. package/dist/admin/{AdminUserLayout-Dejnz13m.js → AdminUserLayout-DyQYacQQ.js} +1 -1
  28. package/dist/admin/AdminUserSessions-D9X2_HMA.js +3 -0
  29. package/dist/admin/{AdminUserSessions-DO9H85O-.js → AdminUserSessions-DEaGu6n6.js} +3 -2
  30. package/dist/admin/AdminUserSessions-DEaGu6n6.js.map +1 -0
  31. package/dist/admin/AdminUserSettings-CE66UTIP.js +3 -0
  32. package/dist/admin/{AdminUserSettings-B3jA8g3p.js → AdminUserSettings-CR7MxX_R.js} +3 -2
  33. package/dist/admin/AdminUserSettings-CR7MxX_R.js.map +1 -0
  34. package/dist/admin/{AdminUsers-ebbrJBT0.js → AdminUsers-BnGIRvmV.js} +3 -2
  35. package/dist/admin/AdminUsers-BnGIRvmV.js.map +1 -0
  36. package/dist/admin/AdminUsers-CG9-2Z8W.js +3 -0
  37. package/dist/admin/index.d.ts +16 -16
  38. package/dist/admin/index.d.ts.map +1 -1
  39. package/dist/admin/index.js +26 -25
  40. package/dist/admin/index.js.map +1 -1
  41. package/dist/auth/{AuthLayout-BAZJHzDG.js → AuthLayout-B1sUB8fB.js} +2 -2
  42. package/dist/auth/AuthLayout-B1sUB8fB.js.map +1 -0
  43. package/dist/auth/Login-BWi-pPbO.js +4 -0
  44. package/dist/auth/{Login-CeNZZjrr.js → Login-Cjxv3EDi.js} +2 -2
  45. package/dist/auth/Login-Cjxv3EDi.js.map +1 -0
  46. package/dist/auth/{Register-s4ENeyiE.js → Register-BKBIpHhW.js} +3 -2
  47. package/dist/auth/Register-BKBIpHhW.js.map +1 -0
  48. package/dist/auth/Register-CtdvihIM.js +4 -0
  49. package/dist/auth/ResetPassword-BUdM7T_R.js +3 -0
  50. package/dist/auth/{ResetPassword-GLIFkJT7.js → ResetPassword-DvqD_1SJ.js} +3 -2
  51. package/dist/auth/ResetPassword-DvqD_1SJ.js.map +1 -0
  52. package/dist/auth/VerifyEmail-BYmtnkEl.js +3 -0
  53. package/dist/auth/{VerifyEmail-R79sSej_.js → VerifyEmail-VaBruOnO.js} +3 -2
  54. package/dist/auth/VerifyEmail-VaBruOnO.js.map +1 -0
  55. package/dist/auth/index.d.ts +11 -11
  56. package/dist/auth/index.d.ts.map +1 -1
  57. package/dist/auth/index.js +10 -10
  58. package/dist/auth/index.js.map +1 -1
  59. package/dist/core/index.d.ts +36 -55
  60. package/dist/core/index.d.ts.map +1 -1
  61. package/dist/core/index.js +44 -345
  62. package/dist/core/index.js.map +1 -1
  63. package/dist/demo/DemoDataTable-2mzzf__a.js +150 -0
  64. package/dist/demo/DemoDataTable-2mzzf__a.js.map +1 -0
  65. package/dist/demo/DemoHome-CnuL5WV9.js +25 -0
  66. package/dist/demo/DemoHome-CnuL5WV9.js.map +1 -0
  67. package/dist/demo/DemoHome-D6Z7EE4V.js +3 -0
  68. package/dist/demo/DemoJsonViewer-CYUggLop.js +4 -0
  69. package/dist/demo/DemoJsonViewer-NUGst5wW.js +430 -0
  70. package/dist/demo/DemoJsonViewer-NUGst5wW.js.map +1 -0
  71. package/dist/demo/DemoLayout-ZFDzyvY3.js +3 -0
  72. package/dist/demo/DemoLayout-dvbeuBBf.js +47 -0
  73. package/dist/demo/DemoLayout-dvbeuBBf.js.map +1 -0
  74. package/dist/demo/DemoLogin--wE44i23.js +327 -0
  75. package/dist/demo/DemoLogin--wE44i23.js.map +1 -0
  76. package/dist/demo/DemoRegister-BtrMksx6.js +488 -0
  77. package/dist/demo/DemoRegister-BtrMksx6.js.map +1 -0
  78. package/dist/demo/DemoResetPassword-DVXiiiX7.js +341 -0
  79. package/dist/demo/DemoResetPassword-DVXiiiX7.js.map +1 -0
  80. package/dist/demo/DemoSidebar-DWnjYHoP.js +82 -0
  81. package/dist/demo/DemoSidebar-DWnjYHoP.js.map +1 -0
  82. package/dist/demo/DemoTypeForm-P5_VInW2.js +83 -0
  83. package/dist/demo/DemoTypeForm-P5_VInW2.js.map +1 -0
  84. package/dist/demo/DemoVerifyEmail-C_ooC5u8.js +152 -0
  85. package/dist/demo/DemoVerifyEmail-C_ooC5u8.js.map +1 -0
  86. package/dist/demo/IconGoogle-DvmFiEDB.js +58 -0
  87. package/dist/demo/IconGoogle-DvmFiEDB.js.map +1 -0
  88. package/dist/demo/Showcase-vemLuO2t.js +187 -0
  89. package/dist/demo/Showcase-vemLuO2t.js.map +1 -0
  90. package/dist/demo/index.d.ts +97 -0
  91. package/dist/demo/index.d.ts.map +1 -0
  92. package/dist/demo/index.js +121 -0
  93. package/dist/demo/index.js.map +1 -0
  94. package/dist/json/index.d.ts +58 -0
  95. package/dist/json/index.d.ts.map +1 -0
  96. package/dist/json/index.js +325 -0
  97. package/dist/json/index.js.map +1 -0
  98. package/package.json +17 -6
  99. package/src/admin/AdminRouter.ts +1 -1
  100. package/src/admin/MainRouter.ts +1 -1
  101. package/src/admin/components/audits/AdminAudits.tsx +2 -1
  102. package/src/admin/components/sessions/AdminSessions.tsx +2 -1
  103. package/src/admin/components/users/AdminUserAudits.tsx +2 -1
  104. package/src/admin/components/users/AdminUserCreate.tsx +2 -1
  105. package/src/admin/components/users/AdminUserDetails.tsx +2 -1
  106. package/src/admin/components/users/AdminUserLayout.tsx +2 -6
  107. package/src/admin/components/users/AdminUserSessions.tsx +2 -1
  108. package/src/admin/components/users/AdminUserSettings.tsx +2 -1
  109. package/src/admin/components/users/AdminUsers.tsx +2 -1
  110. package/src/auth/AuthRouter.ts +1 -1
  111. package/src/auth/components/AuthLayout.tsx +1 -1
  112. package/src/auth/components/Login.tsx +1 -1
  113. package/src/auth/components/Register.tsx +2 -1
  114. package/src/auth/components/ResetPassword.tsx +2 -1
  115. package/src/auth/components/VerifyEmail.tsx +2 -1
  116. package/src/auth/components/buttons/UserButton.tsx +1 -1
  117. package/src/core/RootRouter.ts +1 -1
  118. package/src/core/components/buttons/ActionButton.tsx +3 -4
  119. package/src/core/components/form/Control.tsx +12 -1
  120. package/src/core/components/form/ControlNumber.tsx +5 -0
  121. package/src/core/components/layout/AdminShell.tsx +2 -1
  122. package/src/core/components/layout/AlephaMantineProvider.tsx +2 -1
  123. package/src/core/components/layout/Omnibar.tsx +2 -1
  124. package/src/core/components/layout/Sidebar.tsx +10 -13
  125. package/src/core/index.ts +1 -2
  126. package/src/core/services/DialogService.tsx +0 -17
  127. package/{styles.css → src/core/styles.css} +1 -5
  128. package/src/demo/DemoRouter.ts +123 -0
  129. package/src/demo/components/DemoHome.tsx +29 -0
  130. package/src/demo/components/DemoLayout.tsx +52 -0
  131. package/src/demo/components/auth/DemoLogin.tsx +130 -0
  132. package/src/demo/components/auth/DemoRegister.tsx +144 -0
  133. package/src/demo/components/auth/DemoResetPassword.tsx +69 -0
  134. package/src/demo/components/auth/DemoVerifyEmail.tsx +28 -0
  135. package/src/demo/components/core/DemoDataTable.tsx +174 -0
  136. package/src/demo/components/core/DemoSidebar.tsx +85 -0
  137. package/src/demo/components/core/DemoTypeForm.tsx +69 -0
  138. package/src/demo/components/json/DemoJsonViewer.tsx +128 -0
  139. package/src/demo/components/shared/MacWindow.tsx +105 -0
  140. package/src/demo/components/shared/Showcase.tsx +112 -0
  141. package/src/demo/index.ts +30 -0
  142. package/src/demo/styles.css +0 -0
  143. package/src/json/components/JsonViewer.css +25 -0
  144. package/src/json/components/JsonViewer.tsx +526 -0
  145. package/src/json/extensions/DialogService.tsx +31 -0
  146. package/src/json/index.ts +5 -0
  147. package/src/json/styles.css +1 -0
  148. package/dist/admin/AdminAudits-CwvH8e8c.js.map +0 -1
  149. package/dist/admin/AdminAudits-Dv8Vk_6r.js +0 -3
  150. package/dist/admin/AdminFiles-5CPA3lQk.js +0 -3
  151. package/dist/admin/AdminNotifications-DLjmZWtf.js +0 -3
  152. package/dist/admin/AdminSessions-BCjgJ-93.js.map +0 -1
  153. package/dist/admin/AdminSessions-DEh2uN-4.js +0 -3
  154. package/dist/admin/AdminUserAudits-B_PUXCKC.js.map +0 -1
  155. package/dist/admin/AdminUserCreate-DzfRbGZ4.js.map +0 -1
  156. package/dist/admin/AdminUserDetails-DeTrJm-t.js.map +0 -1
  157. package/dist/admin/AdminUserLayout-CsfrrZkD.js.map +0 -1
  158. package/dist/admin/AdminUserSessions-Bbhcpz4k.js +0 -3
  159. package/dist/admin/AdminUserSessions-DO9H85O-.js.map +0 -1
  160. package/dist/admin/AdminUserSettings-B3jA8g3p.js.map +0 -1
  161. package/dist/admin/AdminUserSettings-CE0xpbQc.js +0 -3
  162. package/dist/admin/AdminUsers-CegGZDhW.js +0 -3
  163. package/dist/admin/AdminUsers-ebbrJBT0.js.map +0 -1
  164. package/dist/auth/AuthLayout-BAZJHzDG.js.map +0 -1
  165. package/dist/auth/Login-CeNZZjrr.js.map +0 -1
  166. package/dist/auth/Login-hQcu1nlu.js +0 -4
  167. package/dist/auth/Register-B6HBNVHS.js +0 -4
  168. package/dist/auth/Register-s4ENeyiE.js.map +0 -1
  169. package/dist/auth/ResetPassword-Cjd-W-Nu.js +0 -3
  170. package/dist/auth/ResetPassword-GLIFkJT7.js.map +0 -1
  171. package/dist/auth/VerifyEmail-Dc9ABKUw.js +0 -3
  172. package/dist/auth/VerifyEmail-R79sSej_.js.map +0 -1
  173. package/src/core/components/data/JsonViewer.tsx +0 -361
@@ -0,0 +1,526 @@
1
+ import {
2
+ ActionIcon,
3
+ Group,
4
+ getTreeExpandedState,
5
+ type MantineSize,
6
+ Text,
7
+ Tree,
8
+ useTree,
9
+ } from "@mantine/core";
10
+ import {
11
+ IconCheck,
12
+ IconChevronDown,
13
+ IconChevronRight,
14
+ IconCopy,
15
+ } from "@tabler/icons-react";
16
+ import {
17
+ type CSSProperties,
18
+ type ReactNode,
19
+ useCallback,
20
+ useMemo,
21
+ useState,
22
+ } from "react";
23
+
24
+ // =============================================================================
25
+ // TYPES
26
+ // =============================================================================
27
+
28
+ interface JsonViewerProps {
29
+ data: any;
30
+ /**
31
+ * Depth level to expand by default (0 = collapsed, Infinity = all expanded)
32
+ */
33
+ defaultExpandedDepth?: number;
34
+ /**
35
+ * Maximum nesting depth to render
36
+ */
37
+ maxDepth?: number;
38
+ /**
39
+ * Size variant
40
+ */
41
+ size?: MantineSize;
42
+ /**
43
+ * Whether to show quotes around keys and strings
44
+ */
45
+ showQuotes?: boolean;
46
+ /**
47
+ * Show copy button on row hover
48
+ */
49
+ showCopyButton?: boolean;
50
+ /**
51
+ * Custom value formatter. Return formatted string or undefined to use default.
52
+ */
53
+ formatValue?: (
54
+ key: string | undefined,
55
+ value: any,
56
+ path: string[],
57
+ ) => string | number | undefined;
58
+ }
59
+
60
+ interface JsonTreeNode {
61
+ value: string;
62
+ label: string;
63
+ children?: JsonTreeNode[];
64
+ // Custom properties
65
+ nodeValue: any;
66
+ nodeKey: string | undefined;
67
+ path: string[];
68
+ isArrayItem: boolean;
69
+ isRoot?: boolean;
70
+ }
71
+
72
+ // =============================================================================
73
+ // CONSTANTS
74
+ // =============================================================================
75
+
76
+ const SIZE_CONFIG: Record<MantineSize, { icon: number; levelOffset: number }> =
77
+ {
78
+ xs: { icon: 14, levelOffset: 16 },
79
+ sm: { icon: 16, levelOffset: 20 },
80
+ md: { icon: 18, levelOffset: 24 },
81
+ lg: { icon: 20, levelOffset: 28 },
82
+ xl: { icon: 22, levelOffset: 32 },
83
+ };
84
+
85
+ const STYLES = {
86
+ root: {
87
+ fontFamily: "var(--mantine-font-family-monospace)",
88
+ } satisfies CSSProperties,
89
+ chevron: {
90
+ flexShrink: 0,
91
+ color: "var(--mantine-color-dimmed)",
92
+ } satisfies CSSProperties,
93
+ key: {
94
+ color: "var(--mantine-color-cyan-text)",
95
+ fontWeight: 500,
96
+ } satisfies CSSProperties,
97
+ colon: {
98
+ color: "var(--mantine-color-dimmed)",
99
+ } satisfies CSSProperties,
100
+ string: {
101
+ color: "var(--mantine-color-teal-text)",
102
+ } satisfies CSSProperties,
103
+ number: {
104
+ color: "var(--mantine-color-blue-text)",
105
+ } satisfies CSSProperties,
106
+ boolean: {
107
+ color: "var(--mantine-color-violet-text)",
108
+ } satisfies CSSProperties,
109
+ null: {
110
+ color: "var(--mantine-color-dimmed)",
111
+ fontStyle: "italic",
112
+ } satisfies CSSProperties,
113
+ preview: {
114
+ color: "var(--mantine-color-dimmed)",
115
+ } satisfies CSSProperties,
116
+ };
117
+
118
+ // =============================================================================
119
+ // HELPERS
120
+ // =============================================================================
121
+
122
+ const getValueType = (val: any): string => {
123
+ if (val === null) return "null";
124
+ if (val === undefined) return "undefined";
125
+ if (Array.isArray(val)) return "array";
126
+ return typeof val;
127
+ };
128
+
129
+ // Convert JSON to tree data structure
130
+ function buildTreeNodes(
131
+ data: any,
132
+ path: string[] = [],
133
+ key?: string,
134
+ isArrayItem = false,
135
+ maxDepth = 10,
136
+ ): JsonTreeNode | null {
137
+ const currentPath = key !== undefined ? [...path, key] : path;
138
+ const nodeId = currentPath.length > 0 ? currentPath.join(".") : "root";
139
+
140
+ if (currentPath.length > maxDepth) {
141
+ return {
142
+ value: nodeId,
143
+ label: key ?? "",
144
+ nodeValue: data,
145
+ nodeKey: key,
146
+ path: currentPath,
147
+ isArrayItem,
148
+ };
149
+ }
150
+
151
+ const type = getValueType(data);
152
+
153
+ if (type === "object" || type === "array") {
154
+ const entries =
155
+ type === "array"
156
+ ? (data as any[]).map((v, i) => [String(i), v] as const)
157
+ : Object.entries(data);
158
+
159
+ const children = entries
160
+ .map(([k, v]) =>
161
+ buildTreeNodes(v, currentPath, k, type === "array", maxDepth),
162
+ )
163
+ .filter((n): n is JsonTreeNode => n !== null);
164
+
165
+ return {
166
+ value: nodeId,
167
+ label: key ?? "",
168
+ nodeValue: data,
169
+ nodeKey: key,
170
+ path: currentPath,
171
+ isArrayItem,
172
+ children: children.length > 0 ? children : undefined,
173
+ };
174
+ }
175
+
176
+ return {
177
+ value: nodeId,
178
+ label: key ?? "",
179
+ nodeValue: data,
180
+ nodeKey: key,
181
+ path: currentPath,
182
+ isArrayItem,
183
+ };
184
+ }
185
+
186
+ // Get all expandable node IDs up to a certain depth
187
+ function getExpandedIds(
188
+ nodes: JsonTreeNode[],
189
+ targetDepth: number,
190
+ currentDepth = 0,
191
+ ): string[] {
192
+ if (currentDepth >= targetDepth) return [];
193
+ const ids: string[] = [];
194
+ for (const node of nodes) {
195
+ if (node.children) {
196
+ ids.push(node.value);
197
+ ids.push(...getExpandedIds(node.children, targetDepth, currentDepth + 1));
198
+ }
199
+ }
200
+ return ids;
201
+ }
202
+
203
+ // =============================================================================
204
+ // COPY BUTTON COMPONENT
205
+ // =============================================================================
206
+
207
+ const CopyButton = ({
208
+ value,
209
+ iconSize,
210
+ }: {
211
+ value: string;
212
+ iconSize: number;
213
+ }) => {
214
+ const [copied, setCopied] = useState(false);
215
+
216
+ const handleCopy = useCallback(
217
+ (e: React.MouseEvent) => {
218
+ e.stopPropagation();
219
+ navigator.clipboard.writeText(value);
220
+ setCopied(true);
221
+ setTimeout(() => setCopied(false), 1500);
222
+ },
223
+ [value],
224
+ );
225
+
226
+ return (
227
+ <ActionIcon
228
+ size={iconSize + 4}
229
+ variant="transparent"
230
+ c={copied ? "green" : "dimmed"}
231
+ onClick={handleCopy}
232
+ className="alepha-json-viewer-copy"
233
+ >
234
+ {copied ? <IconCheck size={iconSize} /> : <IconCopy size={iconSize} />}
235
+ </ActionIcon>
236
+ );
237
+ };
238
+
239
+ // =============================================================================
240
+ // ROW NODE COMPONENT
241
+ // =============================================================================
242
+
243
+ interface RowNodeProps {
244
+ node: JsonTreeNode;
245
+ expanded: boolean;
246
+ hasChildren: boolean;
247
+ elementProps: any;
248
+ size: MantineSize;
249
+ config: { icon: number; levelOffset: number };
250
+ showQuotes: boolean;
251
+ showCopyButton: boolean;
252
+ renderValue: (val: any, key: string | undefined, path: string[]) => ReactNode;
253
+ }
254
+
255
+ const RowNode = ({
256
+ node,
257
+ expanded,
258
+ hasChildren,
259
+ elementProps,
260
+ size,
261
+ config,
262
+ showQuotes,
263
+ showCopyButton,
264
+ renderValue,
265
+ }: RowNodeProps) => {
266
+ const { nodeValue, nodeKey, path, isArrayItem, isRoot } = node;
267
+ const type = getValueType(nodeValue);
268
+ const isExpandable = type === "object" || type === "array";
269
+
270
+ const getPreview = () => {
271
+ if (!isExpandable) return null;
272
+ const entries = type === "array" ? nodeValue : Object.keys(nodeValue);
273
+ const count = entries.length;
274
+ const label = type === "array" ? "item" : "key";
275
+
276
+ // For root node or collapsed nodes, show the count
277
+ if (!expanded) {
278
+ return (
279
+ <Text fs={"italic"} component="span" size={size} style={STYLES.preview}>
280
+ {count === 0
281
+ ? type === "array"
282
+ ? "[]"
283
+ : "{}"
284
+ : type === "array"
285
+ ? `[ ${count} ${count === 1 ? label : `${label}s`} ]`
286
+ : `{ ${count} ${count === 1 ? label : `${label}s`} }`}
287
+ </Text>
288
+ );
289
+ }
290
+
291
+ return null;
292
+ };
293
+
294
+ const getCopyValue = () =>
295
+ isExpandable ? JSON.stringify(nodeValue, null, 2) : String(nodeValue ?? "");
296
+
297
+ return (
298
+ <Group
299
+ gap={6}
300
+ wrap="nowrap"
301
+ {...elementProps}
302
+ className={`alepha-json-viewer-row ${elementProps.className || ""}`}
303
+ >
304
+ {hasChildren ? (
305
+ expanded ? (
306
+ <IconChevronDown size={config.icon} style={STYLES.chevron} />
307
+ ) : (
308
+ <IconChevronRight size={config.icon} style={STYLES.chevron} />
309
+ )
310
+ ) : (
311
+ <span style={{ width: config.icon, flexShrink: 0 }} />
312
+ )}
313
+
314
+ {nodeKey !== undefined && !isArrayItem && (
315
+ <Text component="span" size={size}>
316
+ <span style={STYLES.key}>
317
+ {showQuotes ? `"${nodeKey}"` : nodeKey}
318
+ </span>
319
+ <span style={STYLES.colon}>:</span>
320
+ </Text>
321
+ )}
322
+
323
+ {nodeKey !== undefined && isArrayItem && (
324
+ <Text component="span" size={size}>
325
+ <span style={STYLES.key}>{nodeKey}</span>
326
+ <span style={STYLES.colon}>:</span>
327
+ </Text>
328
+ )}
329
+
330
+ {hasChildren ? (
331
+ getPreview()
332
+ ) : isExpandable ? (
333
+ type === "array" ? (
334
+ <Text component="span" size={size} style={STYLES.preview}>
335
+ []
336
+ </Text>
337
+ ) : (
338
+ <Text component="span" size={size} style={STYLES.preview}>
339
+ {"{}"}
340
+ </Text>
341
+ )
342
+ ) : (
343
+ renderValue(nodeValue, nodeKey, path)
344
+ )}
345
+
346
+ {showCopyButton && (
347
+ <CopyButton value={getCopyValue()} iconSize={config.icon} />
348
+ )}
349
+ </Group>
350
+ );
351
+ };
352
+
353
+ // =============================================================================
354
+ // MAIN COMPONENT
355
+ // =============================================================================
356
+
357
+ export const JsonViewer = ({
358
+ data,
359
+ defaultExpandedDepth = 2,
360
+ maxDepth = 10,
361
+ size = "sm",
362
+ showQuotes = false,
363
+ showCopyButton = true,
364
+ formatValue,
365
+ }: JsonViewerProps) => {
366
+ const config = SIZE_CONFIG[size] || SIZE_CONFIG.sm;
367
+
368
+ // Build tree data from JSON with root wrapper
369
+ const treeData = useMemo(() => {
370
+ const type = getValueType(data);
371
+
372
+ // For objects and arrays, create a root node wrapper
373
+ if (type === "object" || type === "array") {
374
+ const entries =
375
+ type === "array"
376
+ ? (data as any[]).map((v, i) => [String(i), v] as const)
377
+ : Object.entries(data);
378
+ const children = entries
379
+ .map(([k, v]) => buildTreeNodes(v, [], k, type === "array", maxDepth))
380
+ .filter((n): n is JsonTreeNode => n !== null);
381
+
382
+ const rootNode: JsonTreeNode = {
383
+ value: "root",
384
+ label: "",
385
+ nodeValue: data,
386
+ nodeKey: undefined,
387
+ path: [],
388
+ isArrayItem: false,
389
+ isRoot: true,
390
+ children: children.length > 0 ? children : undefined,
391
+ };
392
+ return [rootNode];
393
+ }
394
+
395
+ // For primitives, just show the value directly
396
+ const node = buildTreeNodes(data, [], undefined, false, maxDepth);
397
+ return node ? [node] : [];
398
+ }, [data, maxDepth]);
399
+
400
+ // Compute initial expanded state (root is always expanded by default unless depth is 0)
401
+ const initialExpandedState = useMemo(() => {
402
+ if (defaultExpandedDepth === 0) return {};
403
+ if (defaultExpandedDepth === Infinity) {
404
+ return getTreeExpandedState(treeData, "*");
405
+ }
406
+ // Add 1 to depth to account for root node
407
+ const ids = getExpandedIds(treeData, defaultExpandedDepth + 1);
408
+ return getTreeExpandedState(treeData, ids);
409
+ }, [treeData, defaultExpandedDepth]);
410
+
411
+ const tree = useTree({ initialExpandedState });
412
+
413
+ // Render value based on type
414
+ const renderValue = useCallback(
415
+ (val: any, key: string | undefined, path: string[]): ReactNode => {
416
+ const custom = formatValue?.(key, val, path);
417
+ if (custom !== undefined) {
418
+ return (
419
+ <Text
420
+ component="span"
421
+ size={size}
422
+ style={STYLES.string}
423
+ className="alepha-json-viewer-value"
424
+ title={String(val)}
425
+ >
426
+ {custom}
427
+ </Text>
428
+ );
429
+ }
430
+
431
+ const type = getValueType(val);
432
+ switch (type) {
433
+ case "string": {
434
+ return (
435
+ <Text
436
+ style={STYLES.string}
437
+ component="span"
438
+ size={size}
439
+ className="alepha-json-viewer-value"
440
+ title={val}
441
+ >
442
+ "{val}"
443
+ </Text>
444
+ );
445
+ }
446
+ case "number":
447
+ return (
448
+ <Text component="span" size={size} style={STYLES.number}>
449
+ {val}
450
+ </Text>
451
+ );
452
+ case "boolean":
453
+ return (
454
+ <Text component="span" size={size} style={STYLES.boolean}>
455
+ {String(val)}
456
+ </Text>
457
+ );
458
+ case "null":
459
+ case "undefined":
460
+ return (
461
+ <Text component="span" size={size} style={STYLES.null}>
462
+ {type}
463
+ </Text>
464
+ );
465
+ default:
466
+ return (
467
+ <Text component="span" size={size}>
468
+ {String(val)}
469
+ </Text>
470
+ );
471
+ }
472
+ },
473
+ [formatValue, showQuotes, size],
474
+ );
475
+
476
+ // Render tree node
477
+ const renderNode = useCallback(
478
+ ({
479
+ node,
480
+ expanded,
481
+ hasChildren,
482
+ elementProps,
483
+ }: {
484
+ node: JsonTreeNode;
485
+ expanded: boolean;
486
+ hasChildren: boolean;
487
+ elementProps: any;
488
+ }): ReactNode => {
489
+ return (
490
+ <RowNode
491
+ node={node}
492
+ expanded={expanded}
493
+ hasChildren={hasChildren}
494
+ elementProps={elementProps}
495
+ size={size}
496
+ config={config}
497
+ showQuotes={showQuotes}
498
+ showCopyButton={showCopyButton}
499
+ renderValue={renderValue}
500
+ />
501
+ );
502
+ },
503
+ [config, renderValue, showCopyButton, showQuotes, size],
504
+ );
505
+
506
+ if (treeData.length === 0) {
507
+ return (
508
+ <Text size={size} style={STYLES.null}>
509
+ {data === null ? "null" : data === undefined ? "undefined" : "{}"}
510
+ </Text>
511
+ );
512
+ }
513
+
514
+ return (
515
+ <Tree
516
+ data={treeData}
517
+ tree={tree}
518
+ levelOffset={config.levelOffset}
519
+ expandOnClick
520
+ renderNode={renderNode as any}
521
+ styles={{ root: STYLES.root }}
522
+ />
523
+ );
524
+ };
525
+
526
+ export default JsonViewer;
@@ -0,0 +1,31 @@
1
+ import { type BaseDialogOptions, DialogService, ui } from "@alepha/ui";
2
+ import { Flex } from "@mantine/core";
3
+ import { JsonViewer } from "../components/JsonViewer.tsx";
4
+
5
+ declare module "@alepha/ui" {
6
+ interface DialogService {
7
+ /**
8
+ * Opens a JSON viewer dialog.
9
+ *
10
+ * @param data - The JSON data to display.
11
+ * @param options - Additional dialog options.
12
+ */
13
+ json(data?: any, options?: BaseDialogOptions): void;
14
+ }
15
+ }
16
+
17
+ DialogService.prototype.json = function (
18
+ data?: any,
19
+ options?: BaseDialogOptions,
20
+ ) {
21
+ this.open({
22
+ size: "lg",
23
+ title: options?.title || "Json Viewer",
24
+ ...options,
25
+ content: (
26
+ <Flex bdrs={"md"} w={"100%"} flex={1} p={"sm"} bg={ui.colors.surface}>
27
+ <JsonViewer size={"xs"} data={data} />
28
+ </Flex>
29
+ ),
30
+ });
31
+ };
@@ -0,0 +1,5 @@
1
+ import "./extensions/DialogService.tsx";
2
+
3
+ // ---------------------------------------------------------------------------------------------------------------------
4
+
5
+ export { JsonViewer } from "./components/JsonViewer.tsx";
@@ -0,0 +1 @@
1
+ @import "./components/JsonViewer.css";
@@ -1 +0,0 @@
1
- {"version":3,"file":"AdminAudits-CwvH8e8c.js","names":[],"sources":["../../src/admin/components/audits/AdminAudits.tsx"],"sourcesContent":["import { useClient, useRouter } from \"@alepha/react\";\nimport { useI18n } from \"@alepha/react/i18n\";\nimport { DataTable, Flex, Text } from \"@alepha/ui\";\nimport { Badge, Group, Stack, Tooltip } from \"@mantine/core\";\nimport {\n IconAlertTriangle,\n IconCheck,\n IconInfoCircle,\n IconUser,\n IconX,\n} from \"@tabler/icons-react\";\nimport { type Page, t } from \"alepha\";\nimport type { AuditController, AuditEntity } from \"alepha/api/audits\";\nimport type { AdminRouter } from \"../../AdminRouter.ts\";\n\nexport interface AdminAuditsProps {\n userRealmName?: string;\n}\n\nconst getSeverityColor = (severity: string) => {\n switch (severity) {\n case \"critical\":\n return \"red\";\n case \"warning\":\n return \"yellow\";\n default:\n return \"blue\";\n }\n};\n\nconst getSeverityIcon = (severity: string) => {\n switch (severity) {\n case \"critical\":\n return <IconAlertTriangle size={12} />;\n case \"warning\":\n return <IconAlertTriangle size={12} />;\n default:\n return <IconInfoCircle size={12} />;\n }\n};\n\nconst getTypeColor = (type: string) => {\n switch (type) {\n case \"auth\":\n return \"blue\";\n case \"user\":\n return \"grape\";\n case \"security\":\n return \"red\";\n case \"system\":\n return \"orange\";\n case \"access\":\n return \"cyan\";\n case \"payment\":\n return \"green\";\n case \"order\":\n return \"teal\";\n default:\n return \"gray\";\n }\n};\n\nconst AdminAudits = (props: AdminAuditsProps) => {\n const client = useClient<AuditController>();\n const router = useRouter<AdminRouter>();\n const { l } = useI18n();\n\n const filters = t.object({\n type: t.optional(t.text()),\n action: t.optional(t.text()),\n severity: t.optional(t.enum([\"info\", \"warning\", \"critical\"])),\n success: t.optional(t.boolean()),\n resourceType: t.optional(t.text()),\n search: t.optional(t.text()),\n from: t.optional(t.datetime()),\n to: t.optional(t.datetime()),\n });\n\n return (\n <Flex flex={1} direction=\"column\">\n <DataTable<AuditEntity, typeof filters>\n submitOnInit\n defaultSize={20}\n typeFormProps={{\n skipSubmitButton: true,\n columns: 4,\n }}\n tableProps={{\n horizontalSpacing: \"xs\",\n verticalSpacing: \"xs\",\n striped: false,\n highlightOnHover: true,\n }}\n filters={filters}\n tableTrProps={(item) => ({\n style: {\n cursor: item.userId ? \"pointer\" : \"default\",\n opacity: item.success ? 1 : 0.85,\n },\n onClick: () => {\n if (item.userId) {\n router.go(\"adminUserDetails\", {\n params: { userId: item.userId },\n });\n }\n },\n })}\n items={async (query) => {\n const response = await client.findAudits({\n query: {\n ...query,\n userRealm: props.userRealmName,\n },\n });\n return response as Page<AuditEntity>;\n }}\n columns={{\n type: {\n label: \"Type\",\n fit: true,\n value: (item) => (\n <Badge size=\"sm\" variant=\"light\" color={getTypeColor(item.type)}>\n {item.type}\n </Badge>\n ),\n },\n action: {\n label: \"Action\",\n fit: true,\n value: (item) => (\n <Badge size=\"sm\" variant=\"outline\">\n {item.action}\n </Badge>\n ),\n },\n severity: {\n label: \"Severity\",\n fit: true,\n value: (item) => (\n <Badge\n size=\"xs\"\n variant=\"light\"\n color={getSeverityColor(item.severity)}\n leftSection={getSeverityIcon(item.severity)}\n >\n {item.severity}\n </Badge>\n ),\n },\n user: {\n label: \"User\",\n fit: true,\n value: (item) =>\n item.userId ? (\n <Tooltip\n label={\n <Stack gap={2}>\n <Text size=\"xs\">{item.userEmail || \"No email\"}</Text>\n <Text size=\"xs\" c=\"dimmed\">\n {item.userRealm || \"default\"}\n </Text>\n </Stack>\n }\n >\n <Group gap={4}>\n <IconUser size={12} />\n <Text size=\"xs\" lineClamp={1} maw={100}>\n {item.userEmail?.split(\"@\")[0] || item.userId.slice(0, 8)}\n </Text>\n </Group>\n </Tooltip>\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n System\n </Text>\n ),\n },\n description: {\n label: \"Description\",\n value: (item) => (\n <Text size=\"sm\" lineClamp={1}>\n {item.description || \"-\"}\n </Text>\n ),\n },\n resource: {\n label: \"Resource\",\n fit: true,\n value: (item) =>\n item.resourceType ? (\n <Tooltip label={`${item.resourceType}: ${item.resourceId}`}>\n <Badge size=\"xs\" variant=\"dot\" color=\"gray\">\n {item.resourceType}\n </Badge>\n </Tooltip>\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n -\n </Text>\n ),\n },\n success: {\n label: \"Status\",\n fit: true,\n value: (item) =>\n item.success ? (\n <IconCheck size={14} color=\"var(--mantine-color-green-6)\" />\n ) : (\n <Tooltip label={item.errorMessage || \"Failed\"}>\n <IconX size={14} color=\"var(--mantine-color-red-6)\" />\n </Tooltip>\n ),\n },\n ipAddress: {\n label: \"IP\",\n fit: true,\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\" ff=\"monospace\">\n {item.ipAddress || \"-\"}\n </Text>\n ),\n },\n createdAt: {\n label: \"Time\",\n fit: true,\n value: (item) => (\n <Tooltip label={l(item.createdAt, { date: \"medium\" })}>\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.createdAt, { date: \"fromNow\" })}\n </Text>\n </Tooltip>\n ),\n },\n }}\n />\n </Flex>\n );\n};\n\nexport default AdminAudits;\n"],"mappings":"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"}
@@ -1,3 +0,0 @@
1
- import { t as AdminAudits_default } from "./AdminAudits-CwvH8e8c.js";
2
-
3
- export { AdminAudits_default as default };
@@ -1,3 +0,0 @@
1
- import { t as AdminFiles_default } from "./AdminFiles-C_w1tb_x.js";
2
-
3
- export { AdminFiles_default as default };
@@ -1,3 +0,0 @@
1
- import { t as AdminNotifications_default } from "./AdminNotifications-DuYy74AN.js";
2
-
3
- export { AdminNotifications_default as default };
@@ -1 +0,0 @@
1
- {"version":3,"file":"AdminSessions-BCjgJ-93.js","names":["filters"],"sources":["../../src/admin/components/sessions/AdminSessions.tsx"],"sourcesContent":["import { useClient, useRouter } from \"@alepha/react\";\nimport { useI18n } from \"@alepha/react/i18n\";\nimport { ActionButton, DataTable, Flex, Text } from \"@alepha/ui\";\nimport { Badge, Group } from \"@mantine/core\";\nimport {\n IconDeviceDesktop,\n IconDeviceMobile,\n IconDeviceTablet,\n IconTrash,\n} from \"@tabler/icons-react\";\nimport { type Page, t } from \"alepha\";\nimport {\n type SessionController,\n type SessionEntity,\n sessions,\n} from \"alepha/api/users\";\nimport { useState } from \"react\";\nimport type { AdminRouter } from \"../../AdminRouter.ts\";\n\nexport interface AdminSessionsProps {\n userRealmName?: string;\n}\n\nconst AdminSessions = (props: AdminSessionsProps) => {\n const client = useClient<SessionController>();\n const router = useRouter<AdminRouter>();\n const { l } = useI18n();\n const [refreshKey, setRefreshKey] = useState(0);\n\n const filters = t.object({\n userId: t.optional(\n t.uuid({\n $control: {\n query: t.pick(sessions.schema, [\"userId\"]),\n },\n }),\n ),\n });\n\n const getDeviceIcon = (device?: string) => {\n switch (device) {\n case \"MOBILE\":\n return <IconDeviceMobile size={14} />;\n case \"TABLET\":\n return <IconDeviceTablet size={14} />;\n default:\n return <IconDeviceDesktop size={14} />;\n }\n };\n\n const isExpired = (expiresAt: Date | string) => {\n return new Date(expiresAt) < new Date();\n };\n\n const handleDelete = async (sessionId: string) => {\n await client.deleteSession({\n params: { id: sessionId },\n query: { userRealmName: props.userRealmName },\n });\n setRefreshKey((k) => k + 1);\n };\n\n return (\n <Flex flex={1} direction=\"column\">\n <DataTable<SessionEntity, typeof filters>\n key={refreshKey}\n submitOnInit\n defaultSize={10}\n typeFormProps={{\n skipSubmitButton: true,\n columns: 3,\n }}\n tableProps={{\n horizontalSpacing: \"xs\",\n verticalSpacing: \"xs\",\n }}\n onFilterChange={(key, _value, form) => {\n if (key === \"userId\") {\n return form.submit();\n }\n }}\n filters={filters}\n tableTrProps={(item) => {\n if (isExpired(item.expiresAt)) {\n return {\n opacity: 0.5,\n };\n }\n return {};\n }}\n items={async (filters) => {\n const response = await client.findSessions({\n query: {\n ...filters,\n userRealmName: props.userRealmName,\n },\n });\n\n return response as Page<SessionEntity>;\n }}\n columns={{\n userId: {\n label: \"User\",\n value: (item) => (\n <ActionButton\n variant=\"subtle\"\n size=\"xs\"\n href={router.path(\"adminUserDetails\", {\n params: { userId: item.userId },\n })}\n >\n <Text size=\"xs\" ff=\"monospace\">\n {item.userId.slice(0, 8)}...\n </Text>\n </ActionButton>\n ),\n },\n userAgent: {\n label: \"Device\",\n fit: true,\n value: (item) => (\n <Group gap={4}>\n {item.userAgent ? (\n <>\n <Badge\n size=\"xs\"\n variant=\"light\"\n leftSection={getDeviceIcon(item.userAgent.device)}\n >\n {item.userAgent.device}\n </Badge>\n <Text size=\"xs\" c=\"dimmed\">\n {item.userAgent.browser} / {item.userAgent.os}\n </Text>\n </>\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n -\n </Text>\n )}\n </Group>\n ),\n },\n ip: {\n label: \"IP\",\n fit: true,\n value: (item) => (\n <Text size=\"xs\" ff=\"monospace\" c=\"dimmed\">\n {item.ip || \"-\"}\n </Text>\n ),\n },\n expiresAt: {\n label: \"Status\",\n fit: true,\n value: (item) => (\n <Badge\n size=\"sm\"\n variant=\"light\"\n color={isExpired(item.expiresAt) ? \"red\" : \"green\"}\n >\n {isExpired(item.expiresAt) ? \"Expired\" : \"Active\"}\n </Badge>\n ),\n },\n createdAt: {\n label: \"Created\",\n fit: true,\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.createdAt, { date: \"fromNow\" })}\n </Text>\n ),\n },\n actions: {\n label: \"\",\n fit: true,\n value: (item) => (\n <ActionButton\n size=\"xs\"\n variant=\"subtle\"\n color=\"red\"\n onClick={() => handleDelete(item.id)}\n >\n <IconTrash size={14} />\n </ActionButton>\n ),\n },\n }}\n />\n </Flex>\n );\n};\n\nexport default AdminSessions;\n"],"mappings":"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"}
@@ -1,3 +0,0 @@
1
- import { t as AdminSessions_default } from "./AdminSessions-BCjgJ-93.js";
2
-
3
- export { AdminSessions_default as default };
@@ -1 +0,0 @@
1
- {"version":3,"file":"AdminUserAudits-B_PUXCKC.js","names":[],"sources":["../../src/admin/components/users/AdminUserAudits.tsx"],"sourcesContent":["import { useClient, useRouterState } from \"@alepha/react\";\nimport { useI18n } from \"@alepha/react/i18n\";\nimport { DataTable, Flex, Text } from \"@alepha/ui\";\nimport { Badge, Group, Tooltip } from \"@mantine/core\";\nimport {\n IconAlertTriangle,\n IconCheck,\n IconInfoCircle,\n IconX,\n} from \"@tabler/icons-react\";\nimport { type Page, t } from \"alepha\";\nimport type { AuditController, AuditEntity } from \"alepha/api/audits\";\n\nexport interface AdminUserAuditsProps {\n userRealmName?: string;\n}\n\nconst getSeverityColor = (severity: string) => {\n switch (severity) {\n case \"critical\":\n return \"red\";\n case \"warning\":\n return \"yellow\";\n default:\n return \"blue\";\n }\n};\n\nconst getSeverityIcon = (severity: string) => {\n switch (severity) {\n case \"critical\":\n return <IconAlertTriangle size={12} />;\n case \"warning\":\n return <IconAlertTriangle size={12} />;\n default:\n return <IconInfoCircle size={12} />;\n }\n};\n\nconst AdminUserAudits = (_props: AdminUserAuditsProps) => {\n const state = useRouterState();\n const client = useClient<AuditController>();\n const { l } = useI18n();\n const userId = state.params.userId as string;\n\n const filters = t.object({\n type: t.optional(t.text()),\n action: t.optional(t.text()),\n severity: t.optional(t.enum([\"info\", \"warning\", \"critical\"])),\n success: t.optional(t.boolean()),\n from: t.optional(t.datetime()),\n to: t.optional(t.datetime()),\n });\n\n return (\n <Flex flex={1} direction=\"column\">\n <DataTable<AuditEntity, typeof filters>\n submitOnInit\n defaultSize={15}\n typeFormProps={{\n skipSubmitButton: true,\n columns: 4,\n }}\n tableProps={{\n horizontalSpacing: \"xs\",\n verticalSpacing: \"xs\",\n striped: false,\n highlightOnHover: true,\n }}\n filters={filters}\n items={async (query) => {\n const response = await client.findByUser({\n params: { userId },\n query,\n });\n return response as Page<AuditEntity>;\n }}\n columns={{\n type: {\n label: \"Type\",\n fit: true,\n value: (item) => (\n <Badge size=\"sm\" variant=\"light\" color=\"grape\">\n {item.type}\n </Badge>\n ),\n },\n action: {\n label: \"Action\",\n fit: true,\n value: (item) => (\n <Badge size=\"sm\" variant=\"outline\">\n {item.action}\n </Badge>\n ),\n },\n severity: {\n label: \"Severity\",\n fit: true,\n value: (item) => (\n <Badge\n size=\"sm\"\n variant=\"light\"\n color={getSeverityColor(item.severity)}\n leftSection={getSeverityIcon(item.severity)}\n >\n {item.severity}\n </Badge>\n ),\n },\n description: {\n label: \"Description\",\n value: (item) => (\n <Text size=\"sm\" lineClamp={1}>\n {item.description || \"-\"}\n </Text>\n ),\n },\n resource: {\n label: \"Resource\",\n fit: true,\n value: (item) =>\n item.resourceType ? (\n <Tooltip label={item.resourceId || \"N/A\"}>\n <Badge size=\"xs\" variant=\"dot\" color=\"gray\">\n {item.resourceType}\n </Badge>\n </Tooltip>\n ) : (\n <Text size=\"xs\" c=\"dimmed\">\n -\n </Text>\n ),\n },\n success: {\n label: \"Status\",\n fit: true,\n value: (item) =>\n item.success ? (\n <Group gap={4}>\n <IconCheck size={14} color=\"var(--mantine-color-green-6)\" />\n <Text size=\"xs\" c=\"green\">\n Success\n </Text>\n </Group>\n ) : (\n <Tooltip label={item.errorMessage || \"Failed\"}>\n <Group gap={4}>\n <IconX size={14} color=\"var(--mantine-color-red-6)\" />\n <Text size=\"xs\" c=\"red\">\n Failed\n </Text>\n </Group>\n </Tooltip>\n ),\n },\n ipAddress: {\n label: \"IP\",\n fit: true,\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\" ff=\"monospace\">\n {item.ipAddress || \"-\"}\n </Text>\n ),\n },\n createdAt: {\n label: \"Time\",\n fit: true,\n value: (item) => (\n <Tooltip label={l(item.createdAt, { date: \"medium\" })}>\n <Text size=\"xs\" c=\"dimmed\">\n {l(item.createdAt, { date: \"fromNow\" })}\n </Text>\n </Tooltip>\n ),\n },\n }}\n />\n </Flex>\n );\n};\n\nexport default AdminUserAudits;\n"],"mappings":"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"}
@@ -1 +0,0 @@
1
- {"version":3,"file":"AdminUserCreate-DzfRbGZ4.js","names":["Text"],"sources":["../../src/admin/components/users/AdminUserCreate.tsx"],"sourcesContent":["import { useClient, useRouter } from \"@alepha/react\";\nimport { useForm } from \"@alepha/react/form\";\nimport { ActionButton, Control, Flex } from \"@alepha/ui\";\nimport { Card, Stack, Text } from \"@mantine/core\";\nimport { t } from \"alepha\";\nimport type { UserController } from \"alepha/api/users\";\nimport type { AdminRouter } from \"../../AdminRouter.ts\";\n\nexport interface AdminUserCreateProps {\n userRealmName?: string;\n}\n\nconst AdminUserCreate = (props: AdminUserCreateProps) => {\n const client = useClient<UserController>();\n const router = useRouter<AdminRouter>();\n\n const form = useForm({\n schema: t.object({\n username: t.optional(\n t.shortText({\n minLength: 3,\n maxLength: 50,\n pattern: \"^[a-zA-Z0-9._-]+$\",\n }),\n ),\n email: t.optional(t.email()),\n phoneNumber: t.optional(t.e164()),\n firstName: t.optional(t.string()),\n lastName: t.optional(t.string()),\n roles: t.optional(t.array(t.string())),\n enabled: t.optional(t.boolean()),\n password: t.optional(t.string({ minLength: 8 })),\n }),\n handler: async (data) => {\n const user = await client.createUser({\n query: {\n userRealmName: props.userRealmName,\n },\n body: {\n ...data,\n enabled: data.enabled ?? true,\n },\n });\n\n await router.go(\"adminUserDetails\", {\n params: { userId: user.id },\n });\n },\n });\n\n return (\n <Flex flex={1} p=\"md\">\n <Card withBorder p=\"lg\" maw={600} w=\"100%\">\n <form {...form.props}>\n <Stack gap=\"md\">\n <Text size=\"lg\" fw={500}>\n Create New User\n </Text>\n\n <Control title=\"Username\" input={form.input.username} />\n\n <Control title=\"Email\" input={form.input.email} />\n\n <Control title=\"Phone Number\" input={form.input.phoneNumber} />\n\n <Control title=\"First Name\" input={form.input.firstName} />\n\n <Control title=\"Last Name\" input={form.input.lastName} />\n\n <Control title=\"Password\" input={form.input.password} password />\n\n <Control title=\"Roles\" input={form.input.roles} />\n\n <Control title=\"Enabled\" input={form.input.enabled} />\n\n <ActionButton form={form}>Create User</ActionButton>\n </Stack>\n </form>\n </Card>\n </Flex>\n );\n};\n\nexport default AdminUserCreate;\n"],"mappings":";;;;;;;;AAYA,MAAM,mBAAmB,UAAgC;CACvD,MAAM,SAAS,WAA2B;CAC1C,MAAM,SAAS,WAAwB;CAEvC,MAAM,OAAO,QAAQ;EACnB,QAAQ,EAAE,OAAO;GACf,UAAU,EAAE,SACV,EAAE,UAAU;IACV,WAAW;IACX,WAAW;IACX,SAAS;IACV,CAAC,CACH;GACD,OAAO,EAAE,SAAS,EAAE,OAAO,CAAC;GAC5B,aAAa,EAAE,SAAS,EAAE,MAAM,CAAC;GACjC,WAAW,EAAE,SAAS,EAAE,QAAQ,CAAC;GACjC,UAAU,EAAE,SAAS,EAAE,QAAQ,CAAC;GAChC,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,QAAQ,CAAC,CAAC;GACtC,SAAS,EAAE,SAAS,EAAE,SAAS,CAAC;GAChC,UAAU,EAAE,SAAS,EAAE,OAAO,EAAE,WAAW,GAAG,CAAC,CAAC;GACjD,CAAC;EACF,SAAS,OAAO,SAAS;GACvB,MAAM,OAAO,MAAM,OAAO,WAAW;IACnC,OAAO,EACL,eAAe,MAAM,eACtB;IACD,MAAM;KACJ,GAAG;KACH,SAAS,KAAK,WAAW;KAC1B;IACF,CAAC;AAEF,SAAM,OAAO,GAAG,oBAAoB,EAClC,QAAQ,EAAE,QAAQ,KAAK,IAAI,EAC5B,CAAC;;EAEL,CAAC;AAEF,QACE,oBAAC;EAAK,MAAM;EAAG,GAAE;YACf,oBAAC;GAAK;GAAW,GAAE;GAAK,KAAK;GAAK,GAAE;aAClC,oBAAC;IAAK,GAAI,KAAK;cACb,qBAAC;KAAM,KAAI;;MACT,oBAACA;OAAK,MAAK;OAAK,IAAI;iBAAK;QAElB;MAEP,oBAAC;OAAQ,OAAM;OAAW,OAAO,KAAK,MAAM;QAAY;MAExD,oBAAC;OAAQ,OAAM;OAAQ,OAAO,KAAK,MAAM;QAAS;MAElD,oBAAC;OAAQ,OAAM;OAAe,OAAO,KAAK,MAAM;QAAe;MAE/D,oBAAC;OAAQ,OAAM;OAAa,OAAO,KAAK,MAAM;QAAa;MAE3D,oBAAC;OAAQ,OAAM;OAAY,OAAO,KAAK,MAAM;QAAY;MAEzD,oBAAC;OAAQ,OAAM;OAAW,OAAO,KAAK,MAAM;OAAU;QAAW;MAEjE,oBAAC;OAAQ,OAAM;OAAQ,OAAO,KAAK,MAAM;QAAS;MAElD,oBAAC;OAAQ,OAAM;OAAU,OAAO,KAAK,MAAM;QAAW;MAEtD,oBAAC;OAAmB;iBAAM;QAA0B;;MAC9C;KACH;IACF;GACF;;AAIX,8BAAe"}