@alepha/ui 0.13.1 → 0.13.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (118) hide show
  1. package/dist/admin/AdminLayout-JakF7ESb.js +388 -0
  2. package/dist/admin/AdminLayout-JakF7ESb.js.map +1 -0
  3. package/dist/admin/AdminLayout-qNsIyl30.js +3 -0
  4. package/dist/admin/AdminNotifications-BPrxALdS.js +154 -0
  5. package/dist/admin/AdminNotifications-BPrxALdS.js.map +1 -0
  6. package/dist/admin/AdminNotifications-DV-35Fi3.js +3 -0
  7. package/dist/admin/{AdminSessions-CmDVneE2.js → AdminSessions-CMmBtbSw.js} +36 -9
  8. package/dist/admin/AdminSessions-CMmBtbSw.js.map +1 -0
  9. package/dist/admin/AdminSessions-Df2VYzlE.js +3 -0
  10. package/dist/admin/AdminUserCreate-Coa_yi6m.js +103 -0
  11. package/dist/admin/AdminUserCreate-Coa_yi6m.js.map +1 -0
  12. package/dist/admin/AdminUserCreate-DjiCcAk0.js +3 -0
  13. package/dist/admin/AdminUserDetails-BCFwOm9w.js +221 -0
  14. package/dist/admin/AdminUserDetails-BCFwOm9w.js.map +1 -0
  15. package/dist/admin/AdminUserDetails-C5yeJNa3.js +3 -0
  16. package/dist/admin/AdminUserLayout-B8ga5QvP.js +3 -0
  17. package/dist/admin/AdminUserLayout-CR2OqV9Z.js +153 -0
  18. package/dist/admin/AdminUserLayout-CR2OqV9Z.js.map +1 -0
  19. package/dist/admin/AdminUserSessions-A_5KkqTY.js +3 -0
  20. package/dist/admin/AdminUserSessions-Bcf6-rjG.js +129 -0
  21. package/dist/admin/AdminUserSessions-Bcf6-rjG.js.map +1 -0
  22. package/dist/admin/AdminUserSettings-DAsAhFjX.js +3 -0
  23. package/dist/admin/AdminUserSettings-DRYVdW6S.js +164 -0
  24. package/dist/admin/AdminUserSettings-DRYVdW6S.js.map +1 -0
  25. package/dist/admin/AdminUsers-Dd9a5UqO.js +3 -0
  26. package/dist/admin/{AdminUsers-88De5pev.js → AdminUsers-IN_2yHKt.js} +32 -14
  27. package/dist/admin/AdminUsers-IN_2yHKt.js.map +1 -0
  28. package/dist/admin/index.d.ts +6052 -908
  29. package/dist/admin/index.js +299 -41
  30. package/dist/admin/index.js.map +1 -1
  31. package/dist/auth/AuthLayout-BSL8ZHgr.js +19 -0
  32. package/dist/auth/AuthLayout-BSL8ZHgr.js.map +1 -0
  33. package/dist/auth/Login-DDsyCNAA.js +4 -0
  34. package/dist/auth/{Login-OCrvjs9U.js → Login-kBfaRgKG.js} +5 -4
  35. package/dist/auth/Login-kBfaRgKG.js.map +1 -0
  36. package/dist/auth/{Register-Ei34GSba.js → Register-BxJmOqpF.js} +9 -6
  37. package/dist/auth/Register-BxJmOqpF.js.map +1 -0
  38. package/dist/auth/Register-D10MnlQc.js +4 -0
  39. package/dist/auth/{ResetPassword-tO0oMzfo.js → ResetPassword-BhyZ9ek4.js} +3 -3
  40. package/dist/auth/ResetPassword-BhyZ9ek4.js.map +1 -0
  41. package/dist/auth/ResetPassword-llBG-STp.js +3 -0
  42. package/dist/auth/VerifyEmail-BvOG-IUC.js +3 -0
  43. package/dist/auth/VerifyEmail-DeLct3oQ.js +131 -0
  44. package/dist/auth/VerifyEmail-DeLct3oQ.js.map +1 -0
  45. package/dist/auth/index.d.ts +3773 -3568
  46. package/dist/auth/index.js +96 -20
  47. package/dist/auth/index.js.map +1 -1
  48. package/dist/core/index.d.ts +340 -155
  49. package/dist/core/index.js +1391 -395
  50. package/dist/core/index.js.map +1 -1
  51. package/package.json +11 -8
  52. package/src/admin/AdminRouter.ts +116 -29
  53. package/src/admin/MainRouter.ts +23 -0
  54. package/src/admin/components/AdminLayout.tsx +86 -103
  55. package/src/admin/components/AdminNotifications.tsx +196 -12
  56. package/src/admin/components/AdminParameters.tsx +1 -1
  57. package/src/admin/components/AdminSessions.tsx +43 -7
  58. package/src/admin/components/AdminUserCreate.tsx +84 -0
  59. package/src/admin/components/AdminUserDetails.tsx +180 -0
  60. package/src/admin/components/AdminUserLayout.tsx +172 -0
  61. package/src/admin/components/AdminUserSessions.tsx +158 -0
  62. package/src/admin/components/AdminUserSettings.tsx +165 -0
  63. package/src/admin/components/AdminUsers.tsx +29 -9
  64. package/src/admin/index.ts +12 -3
  65. package/src/auth/AuthI18n.ts +22 -0
  66. package/src/auth/AuthRouter.ts +82 -8
  67. package/src/auth/components/AuthLayout.tsx +12 -0
  68. package/src/auth/components/Login.tsx +13 -11
  69. package/src/auth/components/Register.tsx +6 -5
  70. package/src/auth/components/ResetPassword.tsx +1 -1
  71. package/src/auth/components/VerifyEmail.tsx +102 -0
  72. package/src/auth/components/buttons/UserButton.tsx +6 -2
  73. package/src/auth/index.ts +1 -0
  74. package/src/core/components/buttons/ActionButton.tsx +11 -4
  75. package/src/core/components/buttons/DarkModeButton.tsx +1 -1
  76. package/src/core/components/buttons/OmnibarButton.tsx +10 -5
  77. package/src/core/components/buttons/ThemeButton.tsx +31 -0
  78. package/src/core/components/layout/AdminShell.tsx +4 -2
  79. package/src/core/components/layout/AlephaMantineProvider.tsx +10 -4
  80. package/src/core/components/layout/Omnibar.tsx +27 -10
  81. package/src/core/components/layout/Sidebar.tsx +33 -15
  82. package/src/core/components/table/DataTable.tsx +9 -5
  83. package/src/core/hooks/useTheme.ts +25 -0
  84. package/src/core/index.ts +9 -4
  85. package/src/core/providers/ThemeProvider.ts +87 -0
  86. package/src/core/themes/aurora.ts +107 -0
  87. package/src/core/themes/crystal.ts +107 -0
  88. package/src/core/themes/default.ts +7 -0
  89. package/src/core/themes/ember.ts +107 -0
  90. package/src/core/themes/index.ts +7 -0
  91. package/src/core/themes/midnight.ts +104 -0
  92. package/src/core/themes/remoraid.ts +278 -0
  93. package/src/core/themes/slate.ts +81 -0
  94. package/dist/admin/AdminJobs-BOq6AZOW.js +0 -3
  95. package/dist/admin/AdminJobs-CDnVxEv6.js +0 -125
  96. package/dist/admin/AdminJobs-CDnVxEv6.js.map +0 -1
  97. package/dist/admin/AdminLayout-Bgx25J8m.js +0 -3
  98. package/dist/admin/AdminLayout-CervL8LV.js +0 -88
  99. package/dist/admin/AdminLayout-CervL8LV.js.map +0 -1
  100. package/dist/admin/AdminNotifications-BDQXt3-e.js +0 -3
  101. package/dist/admin/AdminNotifications-DvI2989x.js +0 -40
  102. package/dist/admin/AdminNotifications-DvI2989x.js.map +0 -1
  103. package/dist/admin/AdminParameters-CWi7crdn.js +0 -40
  104. package/dist/admin/AdminParameters-CWi7crdn.js.map +0 -1
  105. package/dist/admin/AdminParameters-DKRAVen3.js +0 -3
  106. package/dist/admin/AdminSessions-CmDVneE2.js.map +0 -1
  107. package/dist/admin/AdminSessions-Dkk_fzWK.js +0 -3
  108. package/dist/admin/AdminUsers-88De5pev.js.map +0 -1
  109. package/dist/admin/AdminUsers-oyAXqZ5l.js +0 -3
  110. package/dist/admin/AdminVerifications-D93TKymL.js +0 -3
  111. package/dist/admin/AdminVerifications-DBVEoqJe.js +0 -40
  112. package/dist/admin/AdminVerifications-DBVEoqJe.js.map +0 -1
  113. package/dist/auth/Login-BC2jTczq.js +0 -4
  114. package/dist/auth/Login-OCrvjs9U.js.map +0 -1
  115. package/dist/auth/Register-Dh0lsQmI.js +0 -4
  116. package/dist/auth/Register-Ei34GSba.js.map +0 -1
  117. package/dist/auth/ResetPassword-BnlAQAOE.js +0 -3
  118. package/dist/auth/ResetPassword-tO0oMzfo.js.map +0 -1
@@ -1,16 +1,19 @@
1
- import { DataTable, Flex, Text } from "@alepha/ui";
1
+ import { ActionButton, DataTable, Flex, Text } from "@alepha/ui";
2
2
  import { t } from "alepha";
3
- import { useClient } from "@alepha/react";
3
+ import { useClient, useRouter } from "@alepha/react";
4
+ import { IconDeviceDesktop, IconDeviceMobile, IconDeviceTablet, IconTrash } from "@tabler/icons-react";
4
5
  import { useI18n } from "@alepha/react/i18n";
5
6
  import { Badge, Group } from "@mantine/core";
6
7
  import { Fragment, jsx, jsxs } from "react/jsx-runtime";
7
- import { IconDeviceDesktop, IconDeviceMobile, IconDeviceTablet } from "@tabler/icons-react";
8
+ import { useState } from "react";
8
9
  import { sessions } from "alepha/api/users";
9
10
 
10
11
  //#region src/admin/components/AdminSessions.tsx
11
12
  const AdminSessions = (props) => {
12
13
  const client = useClient();
14
+ const router = useRouter();
13
15
  const { l } = useI18n();
16
+ const [refreshKey, setRefreshKey] = useState(0);
14
17
  const filters = t.object({ userId: t.optional(t.uuid({ $control: { query: t.pick(sessions.schema, ["userId"]) } })) });
15
18
  const getDeviceIcon = (device) => {
16
19
  switch (device) {
@@ -22,8 +25,16 @@ const AdminSessions = (props) => {
22
25
  const isExpired = (expiresAt) => {
23
26
  return new Date(expiresAt) < /* @__PURE__ */ new Date();
24
27
  };
28
+ const handleDelete = async (sessionId) => {
29
+ await client.deleteSession({
30
+ params: { id: sessionId },
31
+ query: { userRealmName: props.userRealmName }
32
+ });
33
+ setRefreshKey((k) => k + 1);
34
+ };
25
35
  return /* @__PURE__ */ jsx(Flex, {
26
36
  flex: 1,
37
+ direction: "column",
27
38
  children: /* @__PURE__ */ jsx(DataTable, {
28
39
  submitOnInit: true,
29
40
  defaultSize: 10,
@@ -51,11 +62,16 @@ const AdminSessions = (props) => {
51
62
  },
52
63
  columns: {
53
64
  userId: {
54
- label: "User ID",
55
- value: (item) => /* @__PURE__ */ jsxs(Text, {
65
+ label: "User",
66
+ value: (item) => /* @__PURE__ */ jsx(ActionButton, {
67
+ variant: "subtle",
56
68
  size: "xs",
57
- ff: "monospace",
58
- children: [item.userId.slice(0, 8), "..."]
69
+ href: router.path("adminUserDetails", { params: { userId: item.userId } }),
70
+ children: /* @__PURE__ */ jsxs(Text, {
71
+ size: "xs",
72
+ ff: "monospace",
73
+ children: [item.userId.slice(0, 8), "..."]
74
+ })
59
75
  })
60
76
  },
61
77
  userAgent: {
@@ -111,13 +127,24 @@ const AdminSessions = (props) => {
111
127
  c: "dimmed",
112
128
  children: l(item.createdAt, { date: "fromNow" })
113
129
  })
130
+ },
131
+ actions: {
132
+ label: "",
133
+ fit: true,
134
+ value: (item) => /* @__PURE__ */ jsx(ActionButton, {
135
+ size: "xs",
136
+ variant: "subtle",
137
+ color: "red",
138
+ onClick: () => handleDelete(item.id),
139
+ children: /* @__PURE__ */ jsx(IconTrash, { size: 14 })
140
+ })
114
141
  }
115
142
  }
116
- })
143
+ }, refreshKey)
117
144
  });
118
145
  };
119
146
  var AdminSessions_default = AdminSessions;
120
147
 
121
148
  //#endregion
122
149
  export { AdminSessions_default as t };
123
- //# sourceMappingURL=AdminSessions-CmDVneE2.js.map
150
+ //# sourceMappingURL=AdminSessions-CMmBtbSw.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AdminSessions-CMmBtbSw.js","names":["filters"],"sources":["../../src/admin/components/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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"}
@@ -0,0 +1,3 @@
1
+ import { t as AdminSessions_default } from "./AdminSessions-CMmBtbSw.js";
2
+
3
+ export { AdminSessions_default as default };
@@ -0,0 +1,103 @@
1
+ import { ActionButton, Control, Flex } from "@alepha/ui";
2
+ import { t } from "alepha";
3
+ import { useClient, useRouter } from "@alepha/react";
4
+ import { Card, Stack, Text as Text$1 } from "@mantine/core";
5
+ import { jsx, jsxs } from "react/jsx-runtime";
6
+ import { useForm } from "@alepha/react/form";
7
+
8
+ //#region src/admin/components/AdminUserCreate.tsx
9
+ const AdminUserCreate = (props) => {
10
+ const client = useClient();
11
+ const router = useRouter();
12
+ const form = useForm({
13
+ schema: t.object({
14
+ username: t.optional(t.shortText({
15
+ minLength: 3,
16
+ maxLength: 50,
17
+ pattern: "^[a-zA-Z0-9._-]+$"
18
+ })),
19
+ email: t.optional(t.email()),
20
+ phoneNumber: t.optional(t.e164()),
21
+ firstName: t.optional(t.string()),
22
+ lastName: t.optional(t.string()),
23
+ roles: t.optional(t.array(t.string())),
24
+ enabled: t.optional(t.boolean()),
25
+ password: t.optional(t.string({ minLength: 8 }))
26
+ }),
27
+ handler: async (data) => {
28
+ const user = await client.createUser({
29
+ query: { userRealmName: props.userRealmName },
30
+ body: {
31
+ ...data,
32
+ enabled: data.enabled ?? true
33
+ }
34
+ });
35
+ await router.go("adminUserDetails", { params: { userId: user.id } });
36
+ }
37
+ });
38
+ return /* @__PURE__ */ jsx(Flex, {
39
+ flex: 1,
40
+ p: "md",
41
+ children: /* @__PURE__ */ jsx(Card, {
42
+ withBorder: true,
43
+ p: "lg",
44
+ maw: 600,
45
+ w: "100%",
46
+ children: /* @__PURE__ */ jsx("form", {
47
+ ...form.props,
48
+ children: /* @__PURE__ */ jsxs(Stack, {
49
+ gap: "md",
50
+ children: [
51
+ /* @__PURE__ */ jsx(Text$1, {
52
+ size: "lg",
53
+ fw: 500,
54
+ children: "Create New User"
55
+ }),
56
+ /* @__PURE__ */ jsx(Control, {
57
+ title: "Username",
58
+ input: form.input.username
59
+ }),
60
+ /* @__PURE__ */ jsx(Control, {
61
+ title: "Email",
62
+ input: form.input.email
63
+ }),
64
+ /* @__PURE__ */ jsx(Control, {
65
+ title: "Phone Number",
66
+ input: form.input.phoneNumber
67
+ }),
68
+ /* @__PURE__ */ jsx(Control, {
69
+ title: "First Name",
70
+ input: form.input.firstName
71
+ }),
72
+ /* @__PURE__ */ jsx(Control, {
73
+ title: "Last Name",
74
+ input: form.input.lastName
75
+ }),
76
+ /* @__PURE__ */ jsx(Control, {
77
+ title: "Password",
78
+ input: form.input.password,
79
+ password: true
80
+ }),
81
+ /* @__PURE__ */ jsx(Control, {
82
+ title: "Roles",
83
+ input: form.input.roles
84
+ }),
85
+ /* @__PURE__ */ jsx(Control, {
86
+ title: "Enabled",
87
+ input: form.input.enabled
88
+ }),
89
+ /* @__PURE__ */ jsx(ActionButton, {
90
+ form,
91
+ children: "Create User"
92
+ })
93
+ ]
94
+ })
95
+ })
96
+ })
97
+ });
98
+ };
99
+ var AdminUserCreate_default = AdminUserCreate;
100
+
101
+ //#endregion
102
+ export { AdminUserCreate_default as t };
103
+ //# sourceMappingURL=AdminUserCreate-Coa_yi6m.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AdminUserCreate-Coa_yi6m.js","names":["Text"],"sources":["../../src/admin/components/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"}
@@ -0,0 +1,3 @@
1
+ import { t as AdminUserCreate_default } from "./AdminUserCreate-Coa_yi6m.js";
2
+
3
+ export { AdminUserCreate_default as default };
@@ -0,0 +1,221 @@
1
+ import { ActionButton, Control, Flex, Text } from "@alepha/ui";
2
+ import { t } from "alepha";
3
+ import { useClient, useRouterState } from "@alepha/react";
4
+ import { IconCheck, IconX } from "@tabler/icons-react";
5
+ import { useI18n } from "@alepha/react/i18n";
6
+ import { Card, Group, Loader, Stack } from "@mantine/core";
7
+ import { jsx, jsxs } from "react/jsx-runtime";
8
+ import { useForm } from "@alepha/react/form";
9
+ import { useEffect, useState } from "react";
10
+
11
+ //#region src/admin/components/AdminUserDetails.tsx
12
+ const AdminUserDetails = (props) => {
13
+ const state = useRouterState();
14
+ const client = useClient();
15
+ const { l } = useI18n();
16
+ const userId = state.params.userId;
17
+ const [user, setUser] = useState(null);
18
+ const [loading, setLoading] = useState(true);
19
+ useEffect(() => {
20
+ const loadUser = async () => {
21
+ try {
22
+ setUser(await client.getUser({
23
+ params: { id: userId },
24
+ query: { userRealmName: props.userRealmName }
25
+ }));
26
+ } finally {
27
+ setLoading(false);
28
+ }
29
+ };
30
+ loadUser();
31
+ }, [userId]);
32
+ const form = useForm({
33
+ schema: t.object({
34
+ email: t.optional(t.email()),
35
+ phoneNumber: t.optional(t.e164()),
36
+ firstName: t.optional(t.string()),
37
+ lastName: t.optional(t.string()),
38
+ roles: t.optional(t.array(t.string())),
39
+ enabled: t.optional(t.boolean())
40
+ }),
41
+ handler: async (data) => {
42
+ setUser(await client.updateUser({
43
+ params: { id: userId },
44
+ query: { userRealmName: props.userRealmName },
45
+ body: data
46
+ }));
47
+ }
48
+ });
49
+ useEffect(() => {
50
+ if (user) {
51
+ form.input.email?.set(user.email ?? "");
52
+ form.input.phoneNumber?.set(user.phoneNumber ?? "");
53
+ form.input.firstName?.set(user.firstName ?? "");
54
+ form.input.lastName?.set(user.lastName ?? "");
55
+ form.input.roles?.set(user.roles ?? []);
56
+ form.input.enabled?.set(user.enabled);
57
+ }
58
+ }, [user]);
59
+ if (loading) return /* @__PURE__ */ jsx(Flex, {
60
+ flex: 1,
61
+ justify: "center",
62
+ align: "center",
63
+ children: /* @__PURE__ */ jsx(Loader, {})
64
+ });
65
+ if (!user) return /* @__PURE__ */ jsx(Flex, {
66
+ flex: 1,
67
+ justify: "center",
68
+ align: "center",
69
+ children: /* @__PURE__ */ jsx(Text, {
70
+ c: "dimmed",
71
+ children: "User not found"
72
+ })
73
+ });
74
+ return /* @__PURE__ */ jsxs(Flex, {
75
+ flex: 1,
76
+ direction: "column",
77
+ gap: "md",
78
+ children: [/* @__PURE__ */ jsx(Card, {
79
+ withBorder: true,
80
+ p: "lg",
81
+ children: /* @__PURE__ */ jsxs(Stack, {
82
+ gap: "md",
83
+ children: [/* @__PURE__ */ jsx(Text, {
84
+ size: "lg",
85
+ fw: 500,
86
+ children: "User Details"
87
+ }), /* @__PURE__ */ jsxs(Group, {
88
+ gap: "xl",
89
+ children: [
90
+ /* @__PURE__ */ jsxs(Stack, {
91
+ gap: 4,
92
+ children: [/* @__PURE__ */ jsx(Text, {
93
+ size: "xs",
94
+ c: "dimmed",
95
+ children: "User ID"
96
+ }), /* @__PURE__ */ jsx(Text, {
97
+ size: "sm",
98
+ ff: "monospace",
99
+ children: user.id
100
+ })]
101
+ }),
102
+ /* @__PURE__ */ jsxs(Stack, {
103
+ gap: 4,
104
+ children: [/* @__PURE__ */ jsx(Text, {
105
+ size: "xs",
106
+ c: "dimmed",
107
+ children: "Username"
108
+ }), /* @__PURE__ */ jsx(Text, {
109
+ size: "sm",
110
+ children: user.username || "-"
111
+ })]
112
+ }),
113
+ /* @__PURE__ */ jsxs(Stack, {
114
+ gap: 4,
115
+ children: [/* @__PURE__ */ jsx(Text, {
116
+ size: "xs",
117
+ c: "dimmed",
118
+ children: "Email Verified"
119
+ }), user.emailVerified ? /* @__PURE__ */ jsxs(Group, {
120
+ gap: 4,
121
+ children: [/* @__PURE__ */ jsx(IconCheck, {
122
+ size: 14,
123
+ color: "var(--mantine-color-green-6)"
124
+ }), /* @__PURE__ */ jsx(Text, {
125
+ size: "sm",
126
+ c: "green",
127
+ children: "Verified"
128
+ })]
129
+ }) : /* @__PURE__ */ jsxs(Group, {
130
+ gap: 4,
131
+ children: [/* @__PURE__ */ jsx(IconX, {
132
+ size: 14,
133
+ color: "var(--mantine-color-red-6)"
134
+ }), /* @__PURE__ */ jsx(Text, {
135
+ size: "sm",
136
+ c: "red",
137
+ children: "Not Verified"
138
+ })]
139
+ })]
140
+ }),
141
+ /* @__PURE__ */ jsxs(Stack, {
142
+ gap: 4,
143
+ children: [/* @__PURE__ */ jsx(Text, {
144
+ size: "xs",
145
+ c: "dimmed",
146
+ children: "Created"
147
+ }), /* @__PURE__ */ jsx(Text, {
148
+ size: "sm",
149
+ children: l(user.createdAt, { date: "medium" })
150
+ })]
151
+ }),
152
+ /* @__PURE__ */ jsxs(Stack, {
153
+ gap: 4,
154
+ children: [/* @__PURE__ */ jsx(Text, {
155
+ size: "xs",
156
+ c: "dimmed",
157
+ children: "Updated"
158
+ }), /* @__PURE__ */ jsx(Text, {
159
+ size: "sm",
160
+ children: l(user.updatedAt, { date: "medium" })
161
+ })]
162
+ })
163
+ ]
164
+ })]
165
+ })
166
+ }), /* @__PURE__ */ jsx(Card, {
167
+ withBorder: true,
168
+ p: "lg",
169
+ children: /* @__PURE__ */ jsx("form", {
170
+ ...form.props,
171
+ children: /* @__PURE__ */ jsxs(Stack, {
172
+ gap: "md",
173
+ children: [
174
+ /* @__PURE__ */ jsx(Text, {
175
+ size: "lg",
176
+ fw: 500,
177
+ children: "Edit User"
178
+ }),
179
+ /* @__PURE__ */ jsxs(Group, {
180
+ grow: true,
181
+ children: [/* @__PURE__ */ jsx(Control, {
182
+ title: "Email",
183
+ input: form.input.email
184
+ }), /* @__PURE__ */ jsx(Control, {
185
+ title: "Phone Number",
186
+ input: form.input.phoneNumber
187
+ })]
188
+ }),
189
+ /* @__PURE__ */ jsxs(Group, {
190
+ grow: true,
191
+ children: [/* @__PURE__ */ jsx(Control, {
192
+ title: "First Name",
193
+ input: form.input.firstName
194
+ }), /* @__PURE__ */ jsx(Control, {
195
+ title: "Last Name",
196
+ input: form.input.lastName
197
+ })]
198
+ }),
199
+ /* @__PURE__ */ jsx(Control, {
200
+ title: "Roles",
201
+ input: form.input.roles
202
+ }),
203
+ /* @__PURE__ */ jsx(Control, {
204
+ title: "Enabled",
205
+ input: form.input.enabled
206
+ }),
207
+ /* @__PURE__ */ jsx(Group, { children: /* @__PURE__ */ jsx(ActionButton, {
208
+ form,
209
+ children: "Save Changes"
210
+ }) })
211
+ ]
212
+ })
213
+ })
214
+ })]
215
+ });
216
+ };
217
+ var AdminUserDetails_default = AdminUserDetails;
218
+
219
+ //#endregion
220
+ export { AdminUserDetails_default as t };
221
+ //# sourceMappingURL=AdminUserDetails-BCFwOm9w.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AdminUserDetails-BCFwOm9w.js","names":[],"sources":["../../src/admin/components/AdminUserDetails.tsx"],"sourcesContent":["import { useClient, useRouterState } from \"@alepha/react\";\nimport { useForm } from \"@alepha/react/form\";\nimport { useI18n } from \"@alepha/react/i18n\";\nimport { ActionButton, Control, Flex, Text } from \"@alepha/ui\";\nimport { Card, Group, Loader, Stack } from \"@mantine/core\";\nimport { IconCheck, IconX } from \"@tabler/icons-react\";\nimport { t } from \"alepha\";\nimport type { UserController, UserEntity } from \"alepha/api/users\";\nimport { useEffect, useState } from \"react\";\n\nexport interface AdminUserDetailsProps {\n userRealmName?: string;\n}\n\nconst AdminUserDetails = (props: AdminUserDetailsProps) => {\n const state = useRouterState();\n const client = useClient<UserController>();\n const { l } = useI18n();\n const userId = state.params.userId as string;\n\n const [user, setUser] = useState<UserEntity | null>(null);\n const [loading, setLoading] = useState(true);\n\n useEffect(() => {\n const loadUser = async () => {\n try {\n const data = await client.getUser({\n params: { id: userId },\n query: { userRealmName: props.userRealmName },\n });\n setUser(data);\n } finally {\n setLoading(false);\n }\n };\n\n loadUser();\n }, [userId]);\n\n const form = useForm({\n schema: t.object({\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 }),\n handler: async (data) => {\n const updated = await client.updateUser({\n params: { id: userId },\n query: { userRealmName: props.userRealmName },\n body: data,\n });\n setUser(updated);\n },\n });\n\n useEffect(() => {\n if (user) {\n form.input.email?.set(user.email ?? \"\");\n form.input.phoneNumber?.set(user.phoneNumber ?? \"\");\n form.input.firstName?.set(user.firstName ?? \"\");\n form.input.lastName?.set(user.lastName ?? \"\");\n form.input.roles?.set(user.roles ?? []);\n form.input.enabled?.set(user.enabled);\n }\n }, [user]);\n\n if (loading) {\n return (\n <Flex flex={1} justify=\"center\" align=\"center\">\n <Loader />\n </Flex>\n );\n }\n\n if (!user) {\n return (\n <Flex flex={1} justify=\"center\" align=\"center\">\n <Text c=\"dimmed\">User not found</Text>\n </Flex>\n );\n }\n\n return (\n <Flex flex={1} direction=\"column\" gap=\"md\">\n <Card withBorder p=\"lg\">\n <Stack gap=\"md\">\n <Text size=\"lg\" fw={500}>\n User Details\n </Text>\n\n <Group gap=\"xl\">\n <Stack gap={4}>\n <Text size=\"xs\" c=\"dimmed\">\n User ID\n </Text>\n <Text size=\"sm\" ff=\"monospace\">\n {user.id}\n </Text>\n </Stack>\n\n <Stack gap={4}>\n <Text size=\"xs\" c=\"dimmed\">\n Username\n </Text>\n <Text size=\"sm\">{user.username || \"-\"}</Text>\n </Stack>\n\n <Stack gap={4}>\n <Text size=\"xs\" c=\"dimmed\">\n Email Verified\n </Text>\n {user.emailVerified ? (\n <Group gap={4}>\n <IconCheck size={14} color=\"var(--mantine-color-green-6)\" />\n <Text size=\"sm\" c=\"green\">\n Verified\n </Text>\n </Group>\n ) : (\n <Group gap={4}>\n <IconX size={14} color=\"var(--mantine-color-red-6)\" />\n <Text size=\"sm\" c=\"red\">\n Not Verified\n </Text>\n </Group>\n )}\n </Stack>\n\n <Stack gap={4}>\n <Text size=\"xs\" c=\"dimmed\">\n Created\n </Text>\n <Text size=\"sm\">{l(user.createdAt, { date: \"medium\" })}</Text>\n </Stack>\n\n <Stack gap={4}>\n <Text size=\"xs\" c=\"dimmed\">\n Updated\n </Text>\n <Text size=\"sm\">{l(user.updatedAt, { date: \"medium\" })}</Text>\n </Stack>\n </Group>\n </Stack>\n </Card>\n\n <Card withBorder p=\"lg\">\n <form {...form.props}>\n <Stack gap=\"md\">\n <Text size=\"lg\" fw={500}>\n Edit User\n </Text>\n\n <Group grow>\n <Control title=\"Email\" input={form.input.email} />\n <Control title=\"Phone Number\" input={form.input.phoneNumber} />\n </Group>\n\n <Group grow>\n <Control title=\"First Name\" input={form.input.firstName} />\n <Control title=\"Last Name\" input={form.input.lastName} />\n </Group>\n\n <Control title=\"Roles\" input={form.input.roles} />\n\n <Control title=\"Enabled\" input={form.input.enabled} />\n\n <Group>\n <ActionButton form={form}>Save Changes</ActionButton>\n </Group>\n </Stack>\n </form>\n </Card>\n </Flex>\n );\n};\n\nexport default AdminUserDetails;\n"],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { t as AdminUserDetails_default } from "./AdminUserDetails-BCFwOm9w.js";
2
+
3
+ export { AdminUserDetails_default as default };
@@ -0,0 +1,3 @@
1
+ import { t as AdminUserLayout_default } from "./AdminUserLayout-CR2OqV9Z.js";
2
+
3
+ export { AdminUserLayout_default as default };
@@ -0,0 +1,153 @@
1
+ import { ActionButton, Flex, Text } from "@alepha/ui";
2
+ import { NestedView, useClient, useRouter, useRouterState } from "@alepha/react";
3
+ import { IconDevices, IconSettings, IconUser } from "@tabler/icons-react";
4
+ import { Avatar, Badge, Card, Group, Loader, Stack, Tabs } from "@mantine/core";
5
+ import { jsx, jsxs } from "react/jsx-runtime";
6
+ import { useEffect, useState } from "react";
7
+
8
+ //#region src/admin/components/AdminUserLayout.tsx
9
+ const AdminUserLayout = (props) => {
10
+ const router = useRouter();
11
+ const state = useRouterState();
12
+ const client = useClient();
13
+ const userId = state.params.userId;
14
+ const [user, setUser] = useState(null);
15
+ const [loading, setLoading] = useState(true);
16
+ useEffect(() => {
17
+ const loadUser = async () => {
18
+ try {
19
+ setUser(await client.getUser({
20
+ params: { id: userId },
21
+ query: { userRealmName: props.userRealmName }
22
+ }));
23
+ } finally {
24
+ setLoading(false);
25
+ }
26
+ };
27
+ loadUser();
28
+ }, [userId]);
29
+ if (loading) return /* @__PURE__ */ jsx(Flex, {
30
+ flex: 1,
31
+ justify: "center",
32
+ align: "center",
33
+ children: /* @__PURE__ */ jsx(Loader, {})
34
+ });
35
+ if (!user) return /* @__PURE__ */ jsx(Flex, {
36
+ flex: 1,
37
+ justify: "center",
38
+ align: "center",
39
+ children: /* @__PURE__ */ jsx(Text, {
40
+ c: "dimmed",
41
+ children: "User not found"
42
+ })
43
+ });
44
+ const currentPath = state.url.pathname;
45
+ const detailsPath = router.path("adminUserDetails", { params: { userId } });
46
+ const sessionsPath = router.path("adminUserSessions", { params: { userId } });
47
+ const settingsPath = router.path("adminUserSettings", { params: { userId } });
48
+ const getActiveTab = () => {
49
+ if (currentPath.endsWith("/sessions")) return "sessions";
50
+ if (currentPath.endsWith("/settings")) return "settings";
51
+ return "details";
52
+ };
53
+ const activeTab = getActiveTab();
54
+ const displayName = user.firstName || user.lastName ? `${user.firstName ?? ""} ${user.lastName ?? ""}`.trim() : user.username || user.email || "User";
55
+ return /* @__PURE__ */ jsxs(Flex, {
56
+ flex: 1,
57
+ direction: "column",
58
+ gap: "md",
59
+ p: "md",
60
+ children: [
61
+ /* @__PURE__ */ jsx(Card, {
62
+ withBorder: true,
63
+ p: "md",
64
+ children: /* @__PURE__ */ jsxs(Group, { children: [/* @__PURE__ */ jsx(Avatar, {
65
+ size: "lg",
66
+ radius: "xl",
67
+ color: "blue",
68
+ children: displayName.charAt(0).toUpperCase()
69
+ }), /* @__PURE__ */ jsxs(Stack, {
70
+ gap: 4,
71
+ children: [
72
+ /* @__PURE__ */ jsxs(Group, {
73
+ gap: "xs",
74
+ children: [/* @__PURE__ */ jsx(Text, {
75
+ size: "lg",
76
+ fw: 500,
77
+ children: displayName
78
+ }), /* @__PURE__ */ jsx(Badge, {
79
+ size: "sm",
80
+ variant: "light",
81
+ color: user.enabled ? "green" : "red",
82
+ children: user.enabled ? "Active" : "Disabled"
83
+ })]
84
+ }),
85
+ /* @__PURE__ */ jsx(Text, {
86
+ size: "sm",
87
+ c: "dimmed",
88
+ children: user.email || user.username || user.id
89
+ }),
90
+ user.roles.length > 0 && /* @__PURE__ */ jsx(Group, {
91
+ gap: 4,
92
+ children: user.roles.map((role) => /* @__PURE__ */ jsx(Badge, {
93
+ size: "xs",
94
+ variant: "outline",
95
+ children: role
96
+ }, role))
97
+ })
98
+ ]
99
+ })] })
100
+ }),
101
+ /* @__PURE__ */ jsx(Tabs, {
102
+ value: activeTab,
103
+ children: /* @__PURE__ */ jsxs(Tabs.List, { children: [
104
+ /* @__PURE__ */ jsx(ActionButton, {
105
+ variant: "subtle",
106
+ href: detailsPath,
107
+ leftSection: /* @__PURE__ */ jsx(IconUser, { size: 16 }),
108
+ c: activeTab === "details" ? void 0 : "dimmed",
109
+ fw: activeTab === "details" ? 500 : 400,
110
+ style: {
111
+ borderBottom: activeTab === "details" ? "2px solid var(--mantine-primary-color-filled)" : "2px solid transparent",
112
+ borderRadius: 0
113
+ },
114
+ children: "Details"
115
+ }),
116
+ /* @__PURE__ */ jsx(ActionButton, {
117
+ variant: "subtle",
118
+ href: sessionsPath,
119
+ leftSection: /* @__PURE__ */ jsx(IconDevices, { size: 16 }),
120
+ c: activeTab === "sessions" ? void 0 : "dimmed",
121
+ fw: activeTab === "sessions" ? 500 : 400,
122
+ style: {
123
+ borderBottom: activeTab === "sessions" ? "2px solid var(--mantine-primary-color-filled)" : "2px solid transparent",
124
+ borderRadius: 0
125
+ },
126
+ children: "Sessions"
127
+ }),
128
+ /* @__PURE__ */ jsx(ActionButton, {
129
+ variant: "subtle",
130
+ href: settingsPath,
131
+ leftSection: /* @__PURE__ */ jsx(IconSettings, { size: 16 }),
132
+ c: activeTab === "settings" ? void 0 : "dimmed",
133
+ fw: activeTab === "settings" ? 500 : 400,
134
+ style: {
135
+ borderBottom: activeTab === "settings" ? "2px solid var(--mantine-primary-color-filled)" : "2px solid transparent",
136
+ borderRadius: 0
137
+ },
138
+ children: "Settings"
139
+ })
140
+ ] })
141
+ }),
142
+ /* @__PURE__ */ jsx(Flex, {
143
+ flex: 1,
144
+ children: /* @__PURE__ */ jsx(NestedView, {})
145
+ })
146
+ ]
147
+ });
148
+ };
149
+ var AdminUserLayout_default = AdminUserLayout;
150
+
151
+ //#endregion
152
+ export { AdminUserLayout_default as t };
153
+ //# sourceMappingURL=AdminUserLayout-CR2OqV9Z.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"AdminUserLayout-CR2OqV9Z.js","names":[],"sources":["../../src/admin/components/AdminUserLayout.tsx"],"sourcesContent":["import {\n NestedView,\n useClient,\n useRouter,\n useRouterState,\n} from \"@alepha/react\";\nimport { ActionButton, Flex, Text } from \"@alepha/ui\";\nimport { Avatar, Badge, Card, Group, Loader, Stack, Tabs } from \"@mantine/core\";\nimport { IconDevices, IconSettings, IconUser } from \"@tabler/icons-react\";\nimport type { UserController, UserEntity } from \"alepha/api/users\";\nimport { useEffect, useState } from \"react\";\nimport type { AdminRouter } from \"../AdminRouter.ts\";\n\nexport interface AdminUserLayoutProps {\n userRealmName?: string;\n}\n\nconst AdminUserLayout = (props: AdminUserLayoutProps) => {\n const router = useRouter<AdminRouter>();\n const state = useRouterState();\n const client = useClient<UserController>();\n const userId = state.params.userId as string;\n\n const [user, setUser] = useState<UserEntity | null>(null);\n const [loading, setLoading] = useState(true);\n\n useEffect(() => {\n const loadUser = async () => {\n try {\n const data = await client.getUser({\n params: { id: userId },\n query: { userRealmName: props.userRealmName },\n });\n setUser(data);\n } finally {\n setLoading(false);\n }\n };\n\n loadUser();\n }, [userId]);\n\n if (loading) {\n return (\n <Flex flex={1} justify=\"center\" align=\"center\">\n <Loader />\n </Flex>\n );\n }\n\n if (!user) {\n return (\n <Flex flex={1} justify=\"center\" align=\"center\">\n <Text c=\"dimmed\">User not found</Text>\n </Flex>\n );\n }\n\n const currentPath = state.url.pathname;\n const detailsPath = router.path(\"adminUserDetails\", { params: { userId } });\n const sessionsPath = router.path(\"adminUserSessions\", { params: { userId } });\n const settingsPath = router.path(\"adminUserSettings\", { params: { userId } });\n\n const getActiveTab = () => {\n if (currentPath.endsWith(\"/sessions\")) return \"sessions\";\n if (currentPath.endsWith(\"/settings\")) return \"settings\";\n return \"details\";\n };\n const activeTab = getActiveTab();\n\n const displayName =\n user.firstName || user.lastName\n ? `${user.firstName ?? \"\"} ${user.lastName ?? \"\"}`.trim()\n : user.username || user.email || \"User\";\n\n return (\n <Flex flex={1} direction=\"column\" gap=\"md\" p=\"md\">\n <Card withBorder p=\"md\">\n <Group>\n <Avatar size=\"lg\" radius=\"xl\" color=\"blue\">\n {displayName.charAt(0).toUpperCase()}\n </Avatar>\n <Stack gap={4}>\n <Group gap=\"xs\">\n <Text size=\"lg\" fw={500}>\n {displayName}\n </Text>\n <Badge\n size=\"sm\"\n variant=\"light\"\n color={user.enabled ? \"green\" : \"red\"}\n >\n {user.enabled ? \"Active\" : \"Disabled\"}\n </Badge>\n </Group>\n <Text size=\"sm\" c=\"dimmed\">\n {user.email || user.username || user.id}\n </Text>\n {user.roles.length > 0 && (\n <Group gap={4}>\n {user.roles.map((role: string) => (\n <Badge key={role} size=\"xs\" variant=\"outline\">\n {role}\n </Badge>\n ))}\n </Group>\n )}\n </Stack>\n </Group>\n </Card>\n\n <Tabs value={activeTab}>\n <Tabs.List>\n <ActionButton\n variant=\"subtle\"\n href={detailsPath}\n leftSection={<IconUser size={16} />}\n c={activeTab === \"details\" ? undefined : \"dimmed\"}\n fw={activeTab === \"details\" ? 500 : 400}\n style={{\n borderBottom:\n activeTab === \"details\"\n ? \"2px solid var(--mantine-primary-color-filled)\"\n : \"2px solid transparent\",\n borderRadius: 0,\n }}\n >\n Details\n </ActionButton>\n <ActionButton\n variant=\"subtle\"\n href={sessionsPath}\n leftSection={<IconDevices size={16} />}\n c={activeTab === \"sessions\" ? undefined : \"dimmed\"}\n fw={activeTab === \"sessions\" ? 500 : 400}\n style={{\n borderBottom:\n activeTab === \"sessions\"\n ? \"2px solid var(--mantine-primary-color-filled)\"\n : \"2px solid transparent\",\n borderRadius: 0,\n }}\n >\n Sessions\n </ActionButton>\n <ActionButton\n variant=\"subtle\"\n href={settingsPath}\n leftSection={<IconSettings size={16} />}\n c={activeTab === \"settings\" ? undefined : \"dimmed\"}\n fw={activeTab === \"settings\" ? 500 : 400}\n style={{\n borderBottom:\n activeTab === \"settings\"\n ? \"2px solid var(--mantine-primary-color-filled)\"\n : \"2px solid transparent\",\n borderRadius: 0,\n }}\n >\n Settings\n </ActionButton>\n </Tabs.List>\n </Tabs>\n\n <Flex flex={1}>\n <NestedView />\n </Flex>\n </Flex>\n );\n};\n\nexport default AdminUserLayout;\n"],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { t as AdminUserSessions_default } from "./AdminUserSessions-Bcf6-rjG.js";
2
+
3
+ export { AdminUserSessions_default as default };