@alepha/ui 0.17.2 → 0.18.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.
- package/dist/admin/{AdminApiKeys-CF_qOO3u.js → AdminApiKeys-C-6_Q-lH.js} +56 -192
- package/dist/admin/AdminApiKeys-C-6_Q-lH.js.map +1 -0
- package/dist/admin/{AdminAudits-BQno3hZG.js → AdminAudits-Bgbf04hO.js} +25 -61
- package/dist/admin/AdminAudits-Bgbf04hO.js.map +1 -0
- package/dist/admin/{AdminFiles-kvuUaASF.js → AdminFiles-B9a7G3cY.js} +6 -8
- package/dist/admin/AdminFiles-B9a7G3cY.js.map +1 -0
- package/dist/admin/{AdminJobDashboard-CrPxp0W1.js → AdminJobDashboard-DaTwf5OY.js} +55 -186
- package/dist/admin/AdminJobDashboard-DaTwf5OY.js.map +1 -0
- package/dist/admin/{AdminJobExecutions-D-b4Zt7W.js → AdminJobExecutions-B9cek5dl.js} +132 -168
- package/dist/admin/AdminJobExecutions-B9cek5dl.js.map +1 -0
- package/dist/admin/{AdminJobRegistry-CNX5cpDx.js → AdminJobRegistry-DFgV3oqx.js} +60 -83
- package/dist/admin/AdminJobRegistry-DFgV3oqx.js.map +1 -0
- package/dist/admin/AdminLayout-DHsvWxVB.js +70 -0
- package/dist/admin/AdminLayout-DHsvWxVB.js.map +1 -0
- package/dist/admin/{AdminParameters-DCGbpt2c.js → AdminParameters-DHw9ATgl.js} +53 -53
- package/dist/admin/AdminParameters-DHw9ATgl.js.map +1 -0
- package/dist/admin/{AdminSessions-DyhW6RZv.js → AdminSessions-BhGJPI3z.js} +11 -18
- package/dist/admin/AdminSessions-BhGJPI3z.js.map +1 -0
- package/dist/admin/{AdminUserLayout-CrBj4UuI.js → AdminUserLayout-BdC4Te8m.js} +112 -151
- package/dist/admin/AdminUserLayout-BdC4Te8m.js.map +1 -0
- package/dist/admin/AdminUserProfile-DAt23fqY.js +69 -0
- package/dist/admin/AdminUserProfile-DAt23fqY.js.map +1 -0
- package/dist/admin/AdminUserSessions-1uzcx02z.js +109 -0
- package/dist/admin/AdminUserSessions-1uzcx02z.js.map +1 -0
- package/dist/admin/AdminUsers-C85c3eiQ.js +121 -0
- package/dist/admin/AdminUsers-C85c3eiQ.js.map +1 -0
- package/dist/{auth/AuthLayout-CdJcrPs4.js → admin/AuthLayout-DFJvCvzw.js} +3 -3
- package/dist/{auth/AuthLayout-CdJcrPs4.js.map → admin/AuthLayout-DFJvCvzw.js.map} +1 -1
- package/dist/{auth/IconGoogle-Bm18QD2q.js → admin/IconGoogle-CSQLPYwX.js} +1 -1
- package/dist/{auth/IconGoogle-Bm18QD2q.js.map → admin/IconGoogle-CSQLPYwX.js.map} +1 -1
- package/dist/{demo/DemoLogin-DjJ9314c.js → admin/Login-BGheURrg.js} +15 -129
- package/dist/{auth/Login-BS_FYTy0.js.map → admin/Login-BGheURrg.js.map} +1 -1
- package/dist/{auth/Profile-CjDsW378.js → admin/Profile-B-c9pCPf.js} +5 -5
- package/dist/{auth/Profile-CjDsW378.js.map → admin/Profile-B-c9pCPf.js.map} +1 -1
- package/dist/{demo/DemoRegister-DzkJ5M83.js → admin/Register-Cs10l8vX.js} +20 -146
- package/dist/{auth/Register-C5eqzAaD.js.map → admin/Register-Cs10l8vX.js.map} +1 -1
- package/dist/{demo/DemoResetPassword-DWh4_BpQ.js → admin/ResetPassword-BwDdfkGH.js} +20 -82
- package/dist/{auth/ResetPassword-XifinVao.js.map → admin/ResetPassword-BwDdfkGH.js.map} +1 -1
- package/dist/{demo/DemoVerifyEmail-DbU_tCj8.js → admin/VerifyEmail-DfXHAiQl.js} +15 -32
- package/dist/{auth/VerifyEmail-DTgbeJOO.js.map → admin/VerifyEmail-DfXHAiQl.js.map} +1 -1
- package/dist/admin/auth-Dr0Cf8I7.js +319 -0
- package/dist/admin/auth-Dr0Cf8I7.js.map +1 -0
- package/dist/admin/core-2xoLiT0o.js +4031 -0
- package/dist/admin/core-2xoLiT0o.js.map +1 -0
- package/dist/admin/index.d.ts +739 -13
- package/dist/admin/index.d.ts.map +1 -1
- package/dist/admin/index.js +79 -111
- package/dist/admin/index.js.map +1 -1
- package/dist/admin/rolldown-runtime-CjeV3_4I.js +18 -0
- package/dist/auth/AuthLayout-CAE1pX9s.js +22 -0
- package/dist/auth/AuthLayout-CAE1pX9s.js.map +1 -0
- package/dist/auth/{Login-BS_FYTy0.js → Login-Denw_UGy.js} +8 -8
- package/dist/auth/Login-Denw_UGy.js.map +1 -0
- package/dist/auth/Profile-BMX_Ar_s.js +155 -0
- package/dist/auth/Profile-BMX_Ar_s.js.map +1 -0
- package/dist/auth/{Register-C5eqzAaD.js → Register-6hi_cpfF.js} +8 -8
- package/dist/auth/Register-6hi_cpfF.js.map +1 -0
- package/dist/auth/{ResetPassword-XifinVao.js → ResetPassword-CqfTk1FI.js} +6 -6
- package/dist/auth/ResetPassword-CqfTk1FI.js.map +1 -0
- package/dist/auth/{VerifyEmail-DTgbeJOO.js → VerifyEmail-nWiSTMjF.js} +5 -5
- package/dist/auth/VerifyEmail-nWiSTMjF.js.map +1 -0
- package/dist/auth/core-niW0sFLv.js +2264 -0
- package/dist/auth/core-niW0sFLv.js.map +1 -0
- package/dist/auth/index.d.ts +336 -8
- package/dist/auth/index.d.ts.map +1 -1
- package/dist/auth/index.js +18 -22
- package/dist/auth/index.js.map +1 -1
- package/dist/core/index.d.ts +1033 -843
- package/dist/core/index.d.ts.map +1 -1
- package/dist/core/index.js +1626 -1354
- package/dist/core/index.js.map +1 -1
- package/dist/demo/AuthLayout-jLa0aKsI.js +22 -0
- package/dist/demo/AuthLayout-jLa0aKsI.js.map +1 -0
- package/dist/demo/DemoButton-BmaWZVwf.js +178 -0
- package/dist/demo/DemoButton-BmaWZVwf.js.map +1 -0
- package/dist/demo/{DemoDataTable-lnBKWBf8.js → DemoDataTable-Z9xyV221.js} +18 -18
- package/dist/demo/DemoDataTable-Z9xyV221.js.map +1 -0
- package/dist/demo/DemoDialog-4ItHLf9t.js +101 -0
- package/dist/demo/DemoDialog-4ItHLf9t.js.map +1 -0
- package/dist/demo/DemoFlex-EtVq8QfX.js +105 -0
- package/dist/demo/DemoFlex-EtVq8QfX.js.map +1 -0
- package/dist/demo/DemoHeading-BS-vGfkI.js +18 -0
- package/dist/demo/DemoHeading-BS-vGfkI.js.map +1 -0
- package/dist/demo/{DemoHome-CUMZsYaH.js → DemoHome-Clbn8AmS.js} +9 -12
- package/dist/demo/DemoHome-Clbn8AmS.js.map +1 -0
- package/dist/demo/DemoJsonViewer-DkIX_ky2.js +109 -0
- package/dist/demo/DemoJsonViewer-DkIX_ky2.js.map +1 -0
- package/dist/demo/DemoLayout-C56xb5EE.js +73 -0
- package/dist/demo/DemoLayout-C56xb5EE.js.map +1 -0
- package/dist/demo/DemoLogin-BZwpicOS.js +128 -0
- package/dist/demo/DemoLogin-BZwpicOS.js.map +1 -0
- package/dist/demo/DemoRegister-C7_qc4MJ.js +140 -0
- package/dist/demo/DemoRegister-C7_qc4MJ.js.map +1 -0
- package/dist/demo/DemoResetPassword-BI1Ct4Dw.js +76 -0
- package/dist/demo/DemoResetPassword-BI1Ct4Dw.js.map +1 -0
- package/dist/demo/{DemoSidebar-C1csnGhX.js → DemoSidebar-CcBo4ltC.js} +6 -9
- package/dist/demo/DemoSidebar-CcBo4ltC.js.map +1 -0
- package/dist/demo/DemoText-CzXuUn3g.js +124 -0
- package/dist/demo/DemoText-CzXuUn3g.js.map +1 -0
- package/dist/demo/DemoToast-BgHDhWrX.js +95 -0
- package/dist/demo/DemoToast-BgHDhWrX.js.map +1 -0
- package/dist/demo/{DemoTypeForm-CWz6fJrJ.js → DemoTypeForm-DDzWoMSV.js} +4 -4
- package/dist/demo/{DemoTypeForm-CWz6fJrJ.js.map → DemoTypeForm-DDzWoMSV.js.map} +1 -1
- package/dist/demo/DemoVerifyEmail-C_Irdnov.js +30 -0
- package/dist/demo/DemoVerifyEmail-C_Irdnov.js.map +1 -0
- package/dist/demo/IconGoogle-CSQLPYwX.js +56 -0
- package/dist/demo/IconGoogle-CSQLPYwX.js.map +1 -0
- package/dist/demo/Login-hSOU3jZc.js +219 -0
- package/dist/demo/Login-hSOU3jZc.js.map +1 -0
- package/dist/demo/Profile-CWqti7FB.js +155 -0
- package/dist/demo/Profile-CWqti7FB.js.map +1 -0
- package/dist/demo/Register-a70LPgs2.js +375 -0
- package/dist/demo/Register-a70LPgs2.js.map +1 -0
- package/dist/demo/ResetPassword-DWN0lzr5.js +286 -0
- package/dist/demo/ResetPassword-DWN0lzr5.js.map +1 -0
- package/dist/demo/Showcase-Dq3MISpd.js +232 -0
- package/dist/demo/Showcase-Dq3MISpd.js.map +1 -0
- package/dist/demo/VerifyEmail-DZWL72K4.js +135 -0
- package/dist/demo/VerifyEmail-DZWL72K4.js.map +1 -0
- package/dist/demo/auth-d6n3xbug.js +257 -0
- package/dist/demo/auth-d6n3xbug.js.map +1 -0
- package/dist/demo/core-RCUw1Q-a.js +4217 -0
- package/dist/demo/core-RCUw1Q-a.js.map +1 -0
- package/dist/demo/index.d.ts +17 -6
- package/dist/demo/index.d.ts.map +1 -1
- package/dist/demo/index.js +92 -24
- package/dist/demo/index.js.map +1 -1
- package/dist/demo/rolldown-runtime-CjeV3_4I.js +18 -0
- package/package.json +16 -20
- package/src/admin/AdminRouter.ts +10 -39
- package/src/admin/components/AdminLayout.tsx +42 -10
- package/src/admin/components/audits/AdminAudits.tsx +10 -64
- package/src/admin/components/files/AdminFiles.tsx +2 -3
- package/src/admin/components/jobs/AdminJobDashboard.tsx +36 -142
- package/src/admin/components/jobs/AdminJobExecutions.tsx +117 -175
- package/src/admin/components/jobs/AdminJobRegistry.tsx +58 -73
- package/src/admin/components/keys/AdminApiKeys.tsx +21 -169
- package/src/admin/components/parameters/AdminParameters.tsx +4 -4
- package/src/admin/components/parameters/ParameterEmptyState.tsx +1 -2
- package/src/admin/components/parameters/ParameterHistory.tsx +3 -3
- package/src/admin/components/parameters/ParameterTree.tsx +2 -8
- package/src/admin/components/parameters/types.ts +3 -3
- package/src/admin/components/sessions/AdminSessions.tsx +8 -16
- package/src/admin/components/users/AdminUserLayout.tsx +113 -150
- package/src/admin/components/users/AdminUserProfile.tsx +50 -0
- package/src/admin/components/users/AdminUserSessions.tsx +106 -126
- package/src/admin/components/users/AdminUsers.tsx +46 -62
- package/src/admin/index.ts +0 -4
- package/src/auth/components/buttons/UserButton.tsx +1 -1
- package/src/auth/index.ts +0 -4
- package/src/core/UiRouter.ts +1 -1
- package/src/core/atoms/alephaSidebarAtom.ts +7 -31
- package/src/core/components/{layout/AlephaMantineProvider.tsx → AlephaMantineProvider.tsx} +3 -4
- package/src/core/components/Flex.tsx +63 -0
- package/src/core/components/Heading.tsx +19 -0
- package/src/core/components/Text.tsx +140 -0
- package/src/core/components/buttons/ActionButton.tsx +12 -1
- package/src/core/components/buttons/BurgerButton.tsx +3 -3
- package/src/core/components/buttons/LanguageButton.tsx +1 -1
- package/src/core/components/buttons/ToggleSidebarButton.tsx +1 -4
- package/src/core/components/data/DetailDrawer.tsx +144 -0
- package/src/core/components/data/DetailList.tsx +64 -0
- package/src/core/components/data/StatCards.tsx +50 -0
- package/src/core/components/layout/AppBar.tsx +11 -10
- package/src/core/components/layout/Breadcrumb.tsx +8 -8
- package/src/core/components/layout/Container.tsx +15 -0
- package/src/core/components/layout/DashboardShell.tsx +23 -238
- package/src/core/components/layout/Omnibar.tsx +1 -2
- package/src/core/components/layout/Sidebar.tsx +103 -71
- package/src/core/components/layout/index.ts +65 -0
- package/src/core/{components/form → form/components}/Control.tsx +32 -14
- package/src/core/{components/form → form/components}/ControlArray.tsx +2 -5
- package/src/core/{components/form → form/components}/ControlDate.tsx +1 -4
- package/src/core/{components/form → form/components}/ControlNumber.tsx +1 -4
- package/src/core/{components/form → form/components}/ControlObject.tsx +1 -4
- package/src/core/{components/form → form/components}/ControlQueryBuilder.tsx +7 -7
- package/src/core/{components/form → form/components}/ControlSelect.tsx +2 -4
- package/src/core/{components/form → form/components}/TypeForm.browser.spec.tsx +22 -64
- package/src/core/{components/form → form/components}/TypeForm.tsx +1 -3
- package/src/core/form/factories/dialogForm.tsx +31 -0
- package/src/core/form/index.ts +23 -0
- package/src/core/{utils → form/utils}/parseInput.ts +2 -4
- package/src/core/index.ts +43 -51
- package/src/core/interfaces/AlephaIntent.ts +6 -0
- package/src/core/interfaces/AlephaTheme.ts +0 -1
- package/src/core/json/factories/dialogJson.tsx +24 -0
- package/src/core/json/index.ts +2 -0
- package/src/core/primitives/$ui.ts +17 -0
- package/src/core/services/DialogService.tsx +1 -48
- package/src/core/styles.css +1 -8
- package/src/core/{components/table → table/components}/ColumnPicker.tsx +2 -3
- package/src/core/{components/table → table/components}/DataTable.tsx +8 -9
- package/src/core/{components/table → table/components}/DataTableFilters.tsx +6 -3
- package/src/core/{components/table → table/components}/DataTableToolbar.tsx +4 -5
- package/src/core/{components/table → table/components}/FilterPicker.tsx +2 -3
- package/src/core/table/index.ts +12 -0
- package/src/core/{components/table → table/interfaces}/types.ts +2 -2
- package/src/demo/DemoRouter.ts +87 -6
- package/src/demo/components/DemoHome.tsx +6 -10
- package/src/demo/components/DemoLayout.tsx +38 -8
- package/src/demo/components/auth/DemoLogin.tsx +1 -1
- package/src/demo/components/auth/DemoRegister.tsx +1 -1
- package/src/demo/components/auth/DemoResetPassword.tsx +1 -1
- package/src/demo/components/auth/DemoVerifyEmail.tsx +1 -1
- package/src/demo/components/core/DemoButton.tsx +160 -0
- package/src/demo/components/core/DemoFlex.tsx +101 -0
- package/src/demo/components/core/DemoHeading.tsx +13 -0
- package/src/demo/components/core/DemoText.tsx +110 -0
- package/src/demo/components/json/DemoJsonViewer.tsx +1 -1
- package/src/demo/components/layout/DemoDialog.tsx +103 -0
- package/src/demo/components/{core → layout}/DemoSidebar.tsx +0 -1
- package/src/demo/components/layout/DemoToast.tsx +96 -0
- package/src/demo/components/shared/MacWindow.tsx +149 -74
- package/src/demo/components/shared/Showcase.tsx +4 -8
- package/src/demo/index.ts +1 -4
- package/src/demo/primitives/$uiDemo.ts +10 -0
- package/dist/admin/AdminApiKeys-CF_qOO3u.js.map +0 -1
- package/dist/admin/AdminAudits-BQno3hZG.js.map +0 -1
- package/dist/admin/AdminFiles-kvuUaASF.js.map +0 -1
- package/dist/admin/AdminJobDashboard-CrPxp0W1.js.map +0 -1
- package/dist/admin/AdminJobExecutions-D-b4Zt7W.js.map +0 -1
- package/dist/admin/AdminJobRegistry-CNX5cpDx.js.map +0 -1
- package/dist/admin/AdminLayout-e-ZP5nWw.js +0 -37
- package/dist/admin/AdminLayout-e-ZP5nWw.js.map +0 -1
- package/dist/admin/AdminParameters-DCGbpt2c.js.map +0 -1
- package/dist/admin/AdminSessions-DyhW6RZv.js.map +0 -1
- package/dist/admin/AdminUserAudits-D1GcREEE.js +0 -177
- package/dist/admin/AdminUserAudits-D1GcREEE.js.map +0 -1
- package/dist/admin/AdminUserCreate-DR8LA0tv.js +0 -104
- package/dist/admin/AdminUserCreate-DR8LA0tv.js.map +0 -1
- package/dist/admin/AdminUserDetails-CDkZNHQD.js +0 -477
- package/dist/admin/AdminUserDetails-CDkZNHQD.js.map +0 -1
- package/dist/admin/AdminUserLayout-CrBj4UuI.js.map +0 -1
- package/dist/admin/AdminUserSessions-srgFHrqy.js +0 -129
- package/dist/admin/AdminUserSessions-srgFHrqy.js.map +0 -1
- package/dist/admin/AdminUserSettings-BFuxl-xT.js +0 -167
- package/dist/admin/AdminUserSettings-BFuxl-xT.js.map +0 -1
- package/dist/admin/AdminUsers-D1pDpiwK.js +0 -118
- package/dist/admin/AdminUsers-D1pDpiwK.js.map +0 -1
- package/dist/demo/DemoDataTable-lnBKWBf8.js.map +0 -1
- package/dist/demo/DemoHome-CUMZsYaH.js.map +0 -1
- package/dist/demo/DemoJsonViewer-_uokbGaW.js +0 -429
- package/dist/demo/DemoJsonViewer-_uokbGaW.js.map +0 -1
- package/dist/demo/DemoLayout-DHVoacE6.js +0 -46
- package/dist/demo/DemoLayout-DHVoacE6.js.map +0 -1
- package/dist/demo/DemoLogin-DjJ9314c.js.map +0 -1
- package/dist/demo/DemoRegister-DzkJ5M83.js.map +0 -1
- package/dist/demo/DemoResetPassword-DWh4_BpQ.js.map +0 -1
- package/dist/demo/DemoSidebar-C1csnGhX.js.map +0 -1
- package/dist/demo/DemoVerifyEmail-DbU_tCj8.js.map +0 -1
- package/dist/demo/Showcase-BzoXNlCn.js +0 -185
- package/dist/demo/Showcase-BzoXNlCn.js.map +0 -1
- package/dist/json/index.d.ts +0 -57
- package/dist/json/index.d.ts.map +0 -1
- package/dist/json/index.js +0 -325
- package/dist/json/index.js.map +0 -1
- package/src/admin/components/users/AdminUserAudits.tsx +0 -184
- package/src/admin/components/users/AdminUserCreate.tsx +0 -85
- package/src/admin/components/users/AdminUserDetails.tsx +0 -431
- package/src/admin/components/users/AdminUserSettings.tsx +0 -171
- package/src/core/components/data/ErrorViewer.tsx +0 -171
- package/src/json/extensions/DialogService.tsx +0 -31
- package/src/json/index.ts +0 -18
- package/src/json/styles.css +0 -1
- /package/dist/{demo → auth}/IconGoogle-Ch1m3Uzl.js +0 -0
- /package/dist/{demo → auth}/IconGoogle-Ch1m3Uzl.js.map +0 -0
- /package/src/{json → core/json}/components/JsonViewer.css +0 -0
- /package/src/{json → core/json}/components/JsonViewer.tsx +0 -0
- /package/src/core/{components/table → table/components}/DataTablePagination.tsx +0 -0
- /package/src/core/{components/table → table/components}/useTableSelection.ts +0 -0
|
@@ -1,10 +1,11 @@
|
|
|
1
|
-
import { ActionButton } from "
|
|
2
|
-
import {
|
|
3
|
-
import {
|
|
4
|
-
import { ActionIcon, Avatar, Badge, Button, Center, Flex as Flex$1, Loader, Menu, Tabs, Text as Text$1, Tooltip } from "@mantine/core";
|
|
1
|
+
import { _ as ActionButton, b as useToast, i as useDialog, s as Text$1, u as Flex$1 } from "./core-2xoLiT0o.js";
|
|
2
|
+
import { t } from "alepha";
|
|
3
|
+
import { ActionIcon, Avatar, Badge, Button, Flex, Loader, Menu, Tabs, Text, Tooltip } from "@mantine/core";
|
|
5
4
|
import { jsx, jsxs } from "react/jsx-runtime";
|
|
5
|
+
import { createContext, useCallback, useContext, useEffect, useState } from "react";
|
|
6
|
+
import { IconBan, IconChevronDown, IconChevronLeft, IconExternalLink, IconShieldCheck, IconTrash } from "@tabler/icons-react";
|
|
7
|
+
import { NestedView, useActive, useRouter, useRouterState } from "alepha/react/router";
|
|
6
8
|
import { useClient } from "alepha/react";
|
|
7
|
-
import { useEffect, useState } from "react";
|
|
8
9
|
|
|
9
10
|
//#region ../../src/admin/components/shared/AdminResourceHeader.tsx
|
|
10
11
|
const ActionMenuItem = (props) => {
|
|
@@ -46,32 +47,32 @@ const AdminResourceHeader = (props) => {
|
|
|
46
47
|
children: title.charAt(0).toUpperCase()
|
|
47
48
|
});
|
|
48
49
|
};
|
|
49
|
-
return /* @__PURE__ */ jsxs(Flex
|
|
50
|
+
return /* @__PURE__ */ jsxs(Flex, {
|
|
50
51
|
direction: "column",
|
|
51
52
|
gap: "xs",
|
|
52
|
-
children: [backHref && /* @__PURE__ */ jsx(Flex
|
|
53
|
+
children: [backHref && /* @__PURE__ */ jsx(Flex, { children: /* @__PURE__ */ jsx(ActionButton, {
|
|
53
54
|
variant: "subtle",
|
|
54
55
|
size: "xs",
|
|
55
56
|
href: backHref,
|
|
56
57
|
leftSection: /* @__PURE__ */ jsx(IconChevronLeft, { size: 14 }),
|
|
57
58
|
c: "dimmed",
|
|
58
59
|
children: backLabel
|
|
59
|
-
}) }), /* @__PURE__ */ jsxs(Flex
|
|
60
|
+
}) }), /* @__PURE__ */ jsxs(Flex, {
|
|
60
61
|
justify: "space-between",
|
|
61
62
|
align: "flex-start",
|
|
62
63
|
wrap: "nowrap",
|
|
63
|
-
children: [/* @__PURE__ */ jsxs(Flex
|
|
64
|
+
children: [/* @__PURE__ */ jsxs(Flex, {
|
|
64
65
|
gap: "md",
|
|
65
66
|
wrap: "nowrap",
|
|
66
|
-
children: [renderAvatar(), /* @__PURE__ */ jsxs(Flex
|
|
67
|
+
children: [renderAvatar(), /* @__PURE__ */ jsxs(Flex, {
|
|
67
68
|
direction: "column",
|
|
68
69
|
gap: 2,
|
|
69
70
|
justify: "center",
|
|
70
71
|
style: { minHeight: 56 },
|
|
71
|
-
children: [/* @__PURE__ */ jsxs(Flex
|
|
72
|
+
children: [/* @__PURE__ */ jsxs(Flex, {
|
|
72
73
|
gap: "xs",
|
|
73
74
|
align: "center",
|
|
74
|
-
children: [/* @__PURE__ */ jsx(Text
|
|
75
|
+
children: [/* @__PURE__ */ jsx(Text, {
|
|
75
76
|
size: "md",
|
|
76
77
|
fw: 600,
|
|
77
78
|
lh: 1.2,
|
|
@@ -83,13 +84,13 @@ const AdminResourceHeader = (props) => {
|
|
|
83
84
|
tt: "lowercase",
|
|
84
85
|
children: status.label
|
|
85
86
|
})]
|
|
86
|
-
}), subtitle && /* @__PURE__ */ jsx(Text
|
|
87
|
+
}), subtitle && /* @__PURE__ */ jsx(Text, {
|
|
87
88
|
size: "xs",
|
|
88
89
|
c: "dimmed",
|
|
89
90
|
children: subtitle
|
|
90
91
|
})]
|
|
91
92
|
})]
|
|
92
|
-
}), /* @__PURE__ */ jsxs(Flex
|
|
93
|
+
}), /* @__PURE__ */ jsxs(Flex, {
|
|
93
94
|
gap: "xs",
|
|
94
95
|
children: [
|
|
95
96
|
externalUrl && /* @__PURE__ */ jsx(Tooltip, {
|
|
@@ -159,171 +160,131 @@ const AdminResourceTabs = (props) => {
|
|
|
159
160
|
|
|
160
161
|
//#endregion
|
|
161
162
|
//#region ../../src/admin/components/users/AdminUserLayout.tsx
|
|
163
|
+
const UserContext = createContext(null);
|
|
164
|
+
const useUser = () => {
|
|
165
|
+
const ctx = useContext(UserContext);
|
|
166
|
+
if (!ctx) throw new Error("useUser must be used within AdminUserLayout");
|
|
167
|
+
return ctx;
|
|
168
|
+
};
|
|
169
|
+
t.object({
|
|
170
|
+
email: t.optional(t.email()),
|
|
171
|
+
phoneNumber: t.optional(t.e164()),
|
|
172
|
+
firstName: t.optional(t.string()),
|
|
173
|
+
lastName: t.optional(t.string()),
|
|
174
|
+
roles: t.optional(t.array(t.string())),
|
|
175
|
+
enabled: t.optional(t.boolean())
|
|
176
|
+
});
|
|
177
|
+
const displayName = (u) => u.firstName || u.lastName ? `${u.firstName ?? ""} ${u.lastName ?? ""}`.trim() : u.username || u.email || "User";
|
|
162
178
|
const AdminUserLayout = (props) => {
|
|
163
179
|
const router = useRouter();
|
|
164
180
|
const state = useRouterState();
|
|
165
181
|
const client = useClient();
|
|
182
|
+
const dialog = useDialog();
|
|
183
|
+
const toast = useToast();
|
|
166
184
|
const userId = state.params.userId;
|
|
167
185
|
const [user, setUser] = useState(null);
|
|
168
186
|
const [loading, setLoading] = useState(true);
|
|
169
|
-
const
|
|
187
|
+
const realmQuery = { userRealmName: props.userRealmName };
|
|
188
|
+
const loadUser = useCallback(async () => {
|
|
189
|
+
setLoading(true);
|
|
190
|
+
try {
|
|
191
|
+
setUser(await client.getUser({
|
|
192
|
+
params: { id: userId },
|
|
193
|
+
query: realmQuery
|
|
194
|
+
}));
|
|
195
|
+
} finally {
|
|
196
|
+
setLoading(false);
|
|
197
|
+
}
|
|
198
|
+
}, [userId, client]);
|
|
170
199
|
useEffect(() => {
|
|
171
|
-
const loadUser = async () => {
|
|
172
|
-
try {
|
|
173
|
-
setUser(await client.getUser({
|
|
174
|
-
params: { id: userId },
|
|
175
|
-
query: { userRealmName: props.userRealmName }
|
|
176
|
-
}));
|
|
177
|
-
} finally {
|
|
178
|
-
setLoading(false);
|
|
179
|
-
}
|
|
180
|
-
};
|
|
181
200
|
loadUser();
|
|
182
|
-
}, [
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
direction: "column",
|
|
191
|
-
align: "center",
|
|
192
|
-
gap: "xs",
|
|
193
|
-
children: [/* @__PURE__ */ jsx(IconUser, {
|
|
194
|
-
size: 48,
|
|
195
|
-
opacity: .3
|
|
196
|
-
}), /* @__PURE__ */ jsx(Text$1, {
|
|
197
|
-
c: "dimmed",
|
|
198
|
-
children: "User not found"
|
|
199
|
-
})]
|
|
200
|
-
})
|
|
201
|
-
});
|
|
202
|
-
const displayName = user.firstName || user.lastName ? `${user.firstName ?? ""} ${user.lastName ?? ""}`.trim() : user.username || user.email || "User";
|
|
203
|
-
const currentPath = state.url.pathname;
|
|
204
|
-
const getActiveTab = () => {
|
|
205
|
-
if (currentPath.endsWith("/sessions")) return "sessions";
|
|
206
|
-
if (currentPath.endsWith("/settings")) return "settings";
|
|
207
|
-
if (currentPath.endsWith("/audits")) return "audits";
|
|
208
|
-
return "profile";
|
|
209
|
-
};
|
|
210
|
-
const handleBlockUser = async () => {
|
|
211
|
-
setActionLoading("block");
|
|
212
|
-
try {
|
|
201
|
+
}, [loadUser]);
|
|
202
|
+
const handleToggleEnabled = async () => {
|
|
203
|
+
if (!user) return;
|
|
204
|
+
const action = user.enabled ? "disable" : "enable";
|
|
205
|
+
if (await dialog.confirm({
|
|
206
|
+
title: `${user.enabled ? "Disable" : "Enable"} User`,
|
|
207
|
+
message: `Are you sure you want to ${action} ${displayName(user)}?`
|
|
208
|
+
})) {
|
|
213
209
|
setUser(await client.updateUser({
|
|
214
|
-
params: { id:
|
|
215
|
-
query:
|
|
210
|
+
params: { id: user.id },
|
|
211
|
+
query: realmQuery,
|
|
216
212
|
body: { enabled: !user.enabled }
|
|
217
213
|
}));
|
|
218
|
-
|
|
219
|
-
setActionLoading(null);
|
|
214
|
+
toast.success({ title: `User ${action}d` });
|
|
220
215
|
}
|
|
221
216
|
};
|
|
222
|
-
const
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
setActionLoading(null);
|
|
231
|
-
};
|
|
232
|
-
const handleDeleteUser = async () => {
|
|
233
|
-
if (!confirm("Are you sure you want to delete this user? This action cannot be undone.")) return;
|
|
234
|
-
setActionLoading("delete");
|
|
235
|
-
try {
|
|
217
|
+
const handleDelete = async () => {
|
|
218
|
+
if (!user) return;
|
|
219
|
+
if (await dialog.confirm({
|
|
220
|
+
title: "Delete User",
|
|
221
|
+
message: `Are you sure you want to delete ${displayName(user)}? This cannot be undone.`,
|
|
222
|
+
confirmColor: "red",
|
|
223
|
+
confirmLabel: "Delete"
|
|
224
|
+
})) {
|
|
236
225
|
await client.deleteUser({
|
|
237
|
-
params: { id:
|
|
238
|
-
query:
|
|
226
|
+
params: { id: user.id },
|
|
227
|
+
query: realmQuery
|
|
239
228
|
});
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
setActionLoading(null);
|
|
229
|
+
toast.success({ title: "User deleted" });
|
|
230
|
+
router.push("adminUsers");
|
|
243
231
|
}
|
|
244
232
|
};
|
|
245
|
-
return /* @__PURE__ */ jsx(Flex$1, {
|
|
246
|
-
|
|
247
|
-
|
|
233
|
+
if (loading) return /* @__PURE__ */ jsx(Flex$1, {
|
|
234
|
+
flex: 1,
|
|
235
|
+
justify: "center",
|
|
236
|
+
align: "center",
|
|
237
|
+
children: /* @__PURE__ */ jsx(Loader, {})
|
|
238
|
+
});
|
|
239
|
+
if (!user) return /* @__PURE__ */ jsx(Flex$1, {
|
|
248
240
|
flex: 1,
|
|
241
|
+
justify: "center",
|
|
242
|
+
align: "center",
|
|
243
|
+
children: /* @__PURE__ */ jsx(Text$1, {
|
|
244
|
+
c: "dimmed",
|
|
245
|
+
children: "User not found"
|
|
246
|
+
})
|
|
247
|
+
});
|
|
248
|
+
return /* @__PURE__ */ jsx(UserContext.Provider, {
|
|
249
|
+
value: {
|
|
250
|
+
user,
|
|
251
|
+
reload: loadUser
|
|
252
|
+
},
|
|
249
253
|
children: /* @__PURE__ */ jsxs(Flex$1, {
|
|
254
|
+
flex: 1,
|
|
250
255
|
direction: "column",
|
|
251
256
|
gap: "lg",
|
|
257
|
+
p: "md",
|
|
252
258
|
children: [
|
|
253
259
|
/* @__PURE__ */ jsx(AdminResourceHeader, {
|
|
254
260
|
backHref: router.path("adminUsers"),
|
|
255
261
|
backLabel: "Users",
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
title: displayName,
|
|
259
|
-
subtitle: user.email || user.username || void 0,
|
|
262
|
+
title: displayName(user),
|
|
263
|
+
subtitle: user.email || user.username,
|
|
260
264
|
status: {
|
|
261
265
|
label: user.enabled ? "Active" : "Disabled",
|
|
262
|
-
color: user.enabled ? "green" : "
|
|
266
|
+
color: user.enabled ? "green" : "gray"
|
|
263
267
|
},
|
|
264
|
-
menuActions: [
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
|
|
272
|
-
|
|
273
|
-
|
|
274
|
-
onClick: handleBlockUser,
|
|
275
|
-
loading: actionLoading === "block"
|
|
276
|
-
},
|
|
277
|
-
...user.email && !user.emailVerified ? [{
|
|
278
|
-
label: "Send Verification Email",
|
|
279
|
-
icon: IconMail,
|
|
280
|
-
onClick: handleSendVerification,
|
|
281
|
-
loading: actionLoading === "verify"
|
|
282
|
-
}] : [],
|
|
283
|
-
{
|
|
284
|
-
label: "Reset Password",
|
|
285
|
-
icon: IconLock,
|
|
286
|
-
onClick: handleResetPassword,
|
|
287
|
-
loading: actionLoading === "reset"
|
|
288
|
-
},
|
|
289
|
-
{
|
|
290
|
-
label: "Delete User",
|
|
291
|
-
icon: IconTrash,
|
|
292
|
-
color: "red",
|
|
293
|
-
onClick: handleDeleteUser,
|
|
294
|
-
loading: actionLoading === "delete"
|
|
295
|
-
}
|
|
296
|
-
]
|
|
297
|
-
}),
|
|
298
|
-
/* @__PURE__ */ jsx(AdminResourceTabs, {
|
|
299
|
-
activeTab: getActiveTab(),
|
|
300
|
-
tabs: [
|
|
301
|
-
{
|
|
302
|
-
value: "profile",
|
|
303
|
-
label: "Profile",
|
|
304
|
-
icon: IconUser,
|
|
305
|
-
href: router.path("adminUserDetails", { params: { userId } })
|
|
306
|
-
},
|
|
307
|
-
{
|
|
308
|
-
value: "sessions",
|
|
309
|
-
label: "Sessions",
|
|
310
|
-
icon: IconDevices,
|
|
311
|
-
href: router.path("adminUserSessions", { params: { userId } })
|
|
312
|
-
},
|
|
313
|
-
{
|
|
314
|
-
value: "audits",
|
|
315
|
-
label: "Activity",
|
|
316
|
-
icon: IconHistory,
|
|
317
|
-
href: router.path("adminUserAudits", { params: { userId } })
|
|
318
|
-
},
|
|
319
|
-
{
|
|
320
|
-
value: "settings",
|
|
321
|
-
label: "Settings",
|
|
322
|
-
icon: IconSettings,
|
|
323
|
-
href: router.path("adminUserSettings", { params: { userId } })
|
|
324
|
-
}
|
|
325
|
-
]
|
|
268
|
+
menuActions: [{
|
|
269
|
+
label: user.enabled ? "Disable" : "Enable",
|
|
270
|
+
icon: user.enabled ? IconBan : IconShieldCheck,
|
|
271
|
+
onClick: handleToggleEnabled
|
|
272
|
+
}, {
|
|
273
|
+
label: "Delete",
|
|
274
|
+
icon: IconTrash,
|
|
275
|
+
color: "red",
|
|
276
|
+
onClick: handleDelete
|
|
277
|
+
}]
|
|
326
278
|
}),
|
|
279
|
+
/* @__PURE__ */ jsx(AdminResourceTabs, { tabs: [{
|
|
280
|
+
value: "profile",
|
|
281
|
+
label: "Profile",
|
|
282
|
+
href: router.path("adminUserProfile", { params: { userId } })
|
|
283
|
+
}, {
|
|
284
|
+
value: "sessions",
|
|
285
|
+
label: "Sessions",
|
|
286
|
+
href: router.path("adminUserSessions", { params: { userId } })
|
|
287
|
+
}] }),
|
|
327
288
|
/* @__PURE__ */ jsx(NestedView, {})
|
|
328
289
|
]
|
|
329
290
|
})
|
|
@@ -331,5 +292,5 @@ const AdminUserLayout = (props) => {
|
|
|
331
292
|
};
|
|
332
293
|
|
|
333
294
|
//#endregion
|
|
334
|
-
export { AdminUserLayout as default };
|
|
335
|
-
//# sourceMappingURL=AdminUserLayout-
|
|
295
|
+
export { AdminUserLayout as default, useUser };
|
|
296
|
+
//# sourceMappingURL=AdminUserLayout-BdC4Te8m.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"AdminUserLayout-BdC4Te8m.js","names":["Flex","Text"],"sources":["../../src/admin/components/shared/AdminResourceHeader.tsx","../../src/admin/components/shared/AdminResourceTabs.tsx","../../src/admin/components/users/AdminUserLayout.tsx"],"sourcesContent":["import { ActionButton } from \"@alepha/ui\";\nimport {\n ActionIcon,\n Avatar,\n Badge,\n Button,\n Flex,\n Menu,\n Text,\n Tooltip,\n} from \"@mantine/core\";\nimport {\n IconChevronDown,\n IconChevronLeft,\n IconExternalLink,\n} from \"@tabler/icons-react\";\nimport { useRouter } from \"alepha/react/router\";\nimport type { ComponentType, ReactNode } from \"react\";\n\nexport interface AdminResourceAction {\n label: string;\n icon?: ComponentType<{ size?: number }>;\n onClick?: () => void;\n href?: string;\n color?: string;\n disabled?: boolean;\n loading?: boolean;\n variant?: \"filled\" | \"light\" | \"outline\" | \"subtle\";\n}\n\nexport interface AdminResourceHeaderProps {\n /**\n * Back navigation URL\n */\n backHref?: string;\n\n /**\n * Back navigation label\n */\n backLabel?: string;\n\n /**\n * Avatar content (letter, image URL, or custom node)\n */\n avatar?: string | ReactNode;\n\n /**\n * Avatar color\n */\n avatarColor?: string;\n\n /**\n * Resource title (e.g., user name)\n */\n title: string;\n\n /**\n * Secondary text (e.g., email)\n */\n subtitle?: string;\n\n /**\n * Tertiary identifier to copy (e.g., user ID)\n */\n identifier?: string;\n\n /**\n * Label for the identifier tooltip\n */\n identifierLabel?: string;\n\n /**\n * Status badge\n */\n status?: {\n label: string;\n color: \"green\" | \"red\" | \"yellow\" | \"blue\" | \"gray\";\n };\n\n /**\n * Additional badges (e.g., roles)\n */\n badges?: Array<{\n label: string;\n color?: string;\n variant?: \"filled\" | \"light\" | \"outline\" | \"dot\";\n }>;\n\n /**\n * Primary action button\n */\n primaryAction?: AdminResourceAction;\n\n /**\n * Menu actions (shown in dropdown)\n */\n menuActions?: AdminResourceAction[];\n\n /**\n * External link URL\n */\n externalUrl?: string;\n\n /**\n * Loading state\n */\n loading?: boolean;\n}\n\nconst ActionMenuItem = (props: { action: AdminResourceAction }) => {\n const { action } = props;\n const router = useRouter();\n\n const menuItemProps: Record<string, unknown> = {};\n if (action.href) {\n Object.assign(menuItemProps, router.anchor(action.href));\n } else if (action.onClick) {\n menuItemProps.onClick = action.onClick;\n }\n\n return (\n <Menu.Item\n leftSection={action.icon ? <action.icon size={16} /> : undefined}\n color={action.color}\n disabled={action.disabled}\n {...menuItemProps}\n >\n {action.label}\n </Menu.Item>\n );\n};\n\nconst AdminResourceHeader = (props: AdminResourceHeaderProps) => {\n const {\n backHref,\n backLabel = \"Back\",\n avatar,\n avatarColor = \"blue\",\n title,\n subtitle,\n identifier,\n identifierLabel = \"ID\",\n status,\n badges = [],\n primaryAction,\n menuActions = [],\n externalUrl,\n } = props;\n\n const renderAvatar = () => {\n if (typeof avatar === \"string\") {\n if (avatar.startsWith(\"http\") || avatar.startsWith(\"/\")) {\n return (\n <Avatar src={avatar} size={56} radius=\"md\" color={avatarColor} />\n );\n }\n return (\n <Avatar size={56} radius=\"md\" color={avatarColor}>\n {avatar}\n </Avatar>\n );\n }\n if (avatar) {\n return avatar;\n }\n return (\n <Avatar size={56} radius=\"md\" color={avatarColor}>\n {title.charAt(0).toUpperCase()}\n </Avatar>\n );\n };\n\n return (\n <Flex direction=\"column\" gap=\"xs\">\n {/* Breadcrumb / Back navigation */}\n {backHref && (\n <Flex>\n <ActionButton\n variant=\"subtle\"\n size=\"xs\"\n href={backHref}\n leftSection={<IconChevronLeft size={14} />}\n c=\"dimmed\"\n >\n {backLabel}\n </ActionButton>\n </Flex>\n )}\n\n {/* Main header */}\n <Flex justify=\"space-between\" align=\"flex-start\" wrap=\"nowrap\">\n {/* Left: Avatar + Info */}\n <Flex gap=\"md\" wrap=\"nowrap\">\n {renderAvatar()}\n\n <Flex\n direction=\"column\"\n gap={2}\n justify=\"center\"\n style={{ minHeight: 56 }}\n >\n {/* Title row */}\n <Flex gap=\"xs\" align=\"center\">\n <Text size=\"md\" fw={600} lh={1.2}>\n {title}\n </Text>\n {status && (\n <Badge\n size=\"xs\"\n variant=\"light\"\n color={status.color}\n tt=\"lowercase\"\n >\n {status.label}\n </Badge>\n )}\n </Flex>\n\n {/* Subtitle */}\n {subtitle && (\n <Text size=\"xs\" c=\"dimmed\">\n {subtitle}\n </Text>\n )}\n </Flex>\n </Flex>\n\n {/* Right: Actions */}\n <Flex gap=\"xs\">\n {externalUrl && (\n <Tooltip label=\"Open in new tab\" openDelay={500}>\n <ActionIcon\n variant=\"subtle\"\n color=\"gray\"\n component=\"a\"\n href={externalUrl}\n target=\"_blank\"\n >\n <IconExternalLink size={18} />\n </ActionIcon>\n </Tooltip>\n )}\n\n {primaryAction && (\n <ActionButton\n variant={primaryAction.variant ?? \"light\"}\n color={primaryAction.color}\n onClick={primaryAction.onClick}\n href={primaryAction.href}\n loading={primaryAction.loading}\n disabled={primaryAction.disabled}\n leftSection={\n primaryAction.icon ? (\n <primaryAction.icon size={16} />\n ) : undefined\n }\n >\n {primaryAction.label}\n </ActionButton>\n )}\n\n {menuActions.length > 0 && (\n <Menu position=\"bottom-end\" shadow=\"md\" width={220}>\n <Menu.Target>\n <Button\n variant=\"default\"\n rightSection={<IconChevronDown size={16} />}\n >\n Actions\n </Button>\n </Menu.Target>\n <Menu.Dropdown>\n {menuActions.map((action, index) => (\n <ActionMenuItem key={index} action={action} />\n ))}\n </Menu.Dropdown>\n </Menu>\n )}\n </Flex>\n </Flex>\n </Flex>\n );\n};\n\nexport default AdminResourceHeader;\n","import { Tabs } from \"@mantine/core\";\nimport { useActive, useRouter } from \"alepha/react/router\";\nimport type { ComponentType, ReactNode } from \"react\";\n\nexport interface AdminResourceTab {\n /**\n * Tab key/value\n */\n value: string;\n\n /**\n * Tab label\n */\n label: string;\n\n /**\n * Tab icon\n */\n icon?: ComponentType<{ size?: number }>;\n\n /**\n * Navigation href\n */\n href: string;\n\n /**\n * Whether tab is disabled\n */\n disabled?: boolean;\n\n /**\n * Badge count to show\n */\n count?: number;\n}\n\nexport interface AdminResourceTabsProps {\n /**\n * Array of tab configurations\n */\n tabs: AdminResourceTab[];\n\n /**\n * Currently active tab value\n */\n activeTab?: string;\n\n /**\n * Content to render below tabs\n */\n children?: ReactNode;\n}\n\nconst TabItem = (props: { tab: AdminResourceTab }) => {\n const { tab } = props;\n const router = useRouter();\n const { isActive, isPending } = useActive({ href: tab.href });\n const anchorProps = router.anchor(tab.href);\n\n return (\n <Tabs.Tab\n value={tab.value}\n component=\"a\"\n leftSection={tab.icon ? <tab.icon size={16} /> : undefined}\n disabled={tab.disabled}\n data-active={isActive || undefined}\n style={{\n opacity: isPending ? 0.6 : 1,\n }}\n {...anchorProps}\n >\n {tab.label}\n {tab.count !== undefined && tab.count > 0 && ` (${tab.count})`}\n </Tabs.Tab>\n );\n};\n\nconst AdminResourceTabs = (props: AdminResourceTabsProps) => {\n const { tabs, activeTab, children } = props;\n\n return (\n <Tabs value={activeTab} variant=\"default\">\n <Tabs.List>\n {tabs.map((tab) => (\n <TabItem key={tab.value} tab={tab} />\n ))}\n </Tabs.List>\n\n {children}\n </Tabs>\n );\n};\n\nexport default AdminResourceTabs;\n","import { Flex, Text, useDialog, useToast } from \"@alepha/ui\";\nimport { Loader } from \"@mantine/core\";\nimport { IconBan, IconShieldCheck, IconTrash } from \"@tabler/icons-react\";\nimport { t } from \"alepha\";\nimport type { AdminUserController, UserEntity } from \"alepha/api/users\";\nimport { useClient } from \"alepha/react\";\nimport { NestedView, useRouter, useRouterState } from \"alepha/react/router\";\nimport {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useState,\n} from \"react\";\nimport type { AdminRouter } from \"../../AdminRouter.ts\";\nimport AdminResourceHeader from \"../shared/AdminResourceHeader.tsx\";\nimport AdminResourceTabs from \"../shared/AdminResourceTabs.tsx\";\n\nexport interface AdminUserLayoutProps {\n userRealmName?: string;\n}\n\ninterface UserContextValue {\n user: UserEntity;\n reload: () => void;\n}\n\nconst UserContext = createContext<UserContextValue | null>(null);\n\nexport const useUser = () => {\n const ctx = useContext(UserContext);\n if (!ctx) throw new Error(\"useUser must be used within AdminUserLayout\");\n return ctx;\n};\n\nconst updateUserSchema = 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\nconst displayName = (u: UserEntity) =>\n u.firstName || u.lastName\n ? `${u.firstName ?? \"\"} ${u.lastName ?? \"\"}`.trim()\n : u.username || u.email || \"User\";\n\nconst AdminUserLayout = (props: AdminUserLayoutProps) => {\n const router = useRouter<AdminRouter>();\n const state = useRouterState();\n const client = useClient<AdminUserController>();\n const dialog = useDialog();\n const toast = useToast();\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 const realmQuery = { userRealmName: props.userRealmName };\n\n const loadUser = useCallback(async () => {\n setLoading(true);\n try {\n const data = await client.getUser({\n params: { id: userId },\n query: realmQuery,\n });\n setUser(data);\n } finally {\n setLoading(false);\n }\n }, [userId, client]);\n\n useEffect(() => {\n loadUser();\n }, [loadUser]);\n\n const handleToggleEnabled = async () => {\n if (!user) return;\n const action = user.enabled ? \"disable\" : \"enable\";\n const confirmed = await dialog.confirm({\n title: `${user.enabled ? \"Disable\" : \"Enable\"} User`,\n message: `Are you sure you want to ${action} ${displayName(user)}?`,\n });\n if (confirmed) {\n const updated = await client.updateUser({\n params: { id: user.id },\n query: realmQuery,\n body: { enabled: !user.enabled },\n });\n setUser(updated);\n toast.success({ title: `User ${action}d` });\n }\n };\n\n const handleDelete = async () => {\n if (!user) return;\n const confirmed = await dialog.confirm({\n title: \"Delete User\",\n message: `Are you sure you want to delete ${displayName(user)}? This cannot be undone.`,\n confirmColor: \"red\",\n confirmLabel: \"Delete\",\n });\n if (confirmed) {\n await client.deleteUser({\n params: { id: user.id },\n query: realmQuery,\n });\n toast.success({ title: \"User deleted\" });\n router.push(\"adminUsers\");\n }\n };\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 <UserContext.Provider value={{ user, reload: loadUser }}>\n <Flex flex={1} direction=\"column\" gap=\"lg\" p=\"md\">\n <AdminResourceHeader\n backHref={router.path(\"adminUsers\")}\n backLabel=\"Users\"\n title={displayName(user)}\n subtitle={user.email || user.username}\n status={{\n label: user.enabled ? \"Active\" : \"Disabled\",\n color: user.enabled ? \"green\" : \"gray\",\n }}\n menuActions={[\n {\n label: user.enabled ? \"Disable\" : \"Enable\",\n icon: user.enabled ? IconBan : IconShieldCheck,\n onClick: handleToggleEnabled,\n },\n {\n label: \"Delete\",\n icon: IconTrash,\n color: \"red\",\n onClick: handleDelete,\n },\n ]}\n />\n\n <AdminResourceTabs\n tabs={[\n {\n value: \"profile\",\n label: \"Profile\",\n href: router.path(\"adminUserProfile\", {\n params: { userId },\n }),\n },\n {\n value: \"sessions\",\n label: \"Sessions\",\n href: router.path(\"adminUserSessions\", {\n params: { userId },\n }),\n },\n ]}\n />\n\n <NestedView />\n </Flex>\n </UserContext.Provider>\n );\n};\n\nexport default AdminUserLayout;\n"],"mappings":"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"}
|
|
@@ -0,0 +1,69 @@
|
|
|
1
|
+
import { f as DetailList, u as Flex$1 } from "./core-2xoLiT0o.js";
|
|
2
|
+
import { useUser } from "./AdminUserLayout-BdC4Te8m.js";
|
|
3
|
+
import { useI18n } from "alepha/react/i18n";
|
|
4
|
+
import { Badge } from "@mantine/core";
|
|
5
|
+
import { jsx } from "react/jsx-runtime";
|
|
6
|
+
|
|
7
|
+
//#region ../../src/admin/components/users/AdminUserProfile.tsx
|
|
8
|
+
const AdminUserProfile = () => {
|
|
9
|
+
const { user } = useUser();
|
|
10
|
+
const { l } = useI18n();
|
|
11
|
+
return /* @__PURE__ */ jsx(DetailList, { items: [
|
|
12
|
+
{
|
|
13
|
+
label: "ID",
|
|
14
|
+
value: user.id,
|
|
15
|
+
copyable: user.id
|
|
16
|
+
},
|
|
17
|
+
{
|
|
18
|
+
label: "Username",
|
|
19
|
+
value: user.username
|
|
20
|
+
},
|
|
21
|
+
{
|
|
22
|
+
label: "Email",
|
|
23
|
+
value: user.email
|
|
24
|
+
},
|
|
25
|
+
{
|
|
26
|
+
label: "Email Verified",
|
|
27
|
+
value: user.emailVerified ? "Yes" : "No"
|
|
28
|
+
},
|
|
29
|
+
{
|
|
30
|
+
label: "Phone",
|
|
31
|
+
value: user.phoneNumber
|
|
32
|
+
},
|
|
33
|
+
{
|
|
34
|
+
label: "First Name",
|
|
35
|
+
value: user.firstName
|
|
36
|
+
},
|
|
37
|
+
{
|
|
38
|
+
label: "Last Name",
|
|
39
|
+
value: user.lastName
|
|
40
|
+
},
|
|
41
|
+
{
|
|
42
|
+
label: "Realm",
|
|
43
|
+
value: user.realm
|
|
44
|
+
},
|
|
45
|
+
{
|
|
46
|
+
label: "Roles",
|
|
47
|
+
value: user.roles.length > 0 ? /* @__PURE__ */ jsx(Flex$1, {
|
|
48
|
+
gap: 4,
|
|
49
|
+
children: user.roles.map((role) => /* @__PURE__ */ jsx(Badge, {
|
|
50
|
+
size: "xs",
|
|
51
|
+
variant: "default",
|
|
52
|
+
children: role
|
|
53
|
+
}, role))
|
|
54
|
+
}) : null
|
|
55
|
+
},
|
|
56
|
+
{
|
|
57
|
+
label: "Created",
|
|
58
|
+
value: String(l(user.createdAt, { date: "lll" }))
|
|
59
|
+
},
|
|
60
|
+
{
|
|
61
|
+
label: "Updated",
|
|
62
|
+
value: String(l(user.updatedAt, { date: "lll" }))
|
|
63
|
+
}
|
|
64
|
+
] });
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
//#endregion
|
|
68
|
+
export { AdminUserProfile as default };
|
|
69
|
+
//# sourceMappingURL=AdminUserProfile-DAt23fqY.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"AdminUserProfile-DAt23fqY.js","names":["Flex"],"sources":["../../src/admin/components/users/AdminUserProfile.tsx"],"sourcesContent":["import { DetailList, Flex } from \"@alepha/ui\";\nimport { Badge } from \"@mantine/core\";\nimport { useI18n } from \"alepha/react/i18n\";\nimport { useUser } from \"./AdminUserLayout.tsx\";\n\nconst AdminUserProfile = () => {\n const { user } = useUser();\n const { l } = useI18n();\n\n return (\n <DetailList\n items={[\n { label: \"ID\", value: user.id, copyable: user.id },\n { label: \"Username\", value: user.username },\n { label: \"Email\", value: user.email },\n {\n label: \"Email Verified\",\n value: user.emailVerified ? \"Yes\" : \"No\",\n },\n { label: \"Phone\", value: user.phoneNumber },\n { label: \"First Name\", value: user.firstName },\n { label: \"Last Name\", value: user.lastName },\n { label: \"Realm\", value: user.realm },\n {\n label: \"Roles\",\n value:\n user.roles.length > 0 ? (\n <Flex gap={4}>\n {user.roles.map((role) => (\n <Badge key={role} size=\"xs\" variant=\"default\">\n {role}\n </Badge>\n ))}\n </Flex>\n ) : null,\n },\n {\n label: \"Created\",\n value: String(l(user.createdAt, { date: \"lll\" })),\n },\n {\n label: \"Updated\",\n value: String(l(user.updatedAt, { date: \"lll\" })),\n },\n ]}\n />\n );\n};\n\nexport default AdminUserProfile;\n"],"mappings":";;;;;;;AAKA,MAAM,yBAAyB;CAC7B,MAAM,EAAE,SAAS,SAAS;CAC1B,MAAM,EAAE,MAAM,SAAS;AAEvB,QACE,oBAAC,cACC,OAAO;EACL;GAAE,OAAO;GAAM,OAAO,KAAK;GAAI,UAAU,KAAK;GAAI;EAClD;GAAE,OAAO;GAAY,OAAO,KAAK;GAAU;EAC3C;GAAE,OAAO;GAAS,OAAO,KAAK;GAAO;EACrC;GACE,OAAO;GACP,OAAO,KAAK,gBAAgB,QAAQ;GACrC;EACD;GAAE,OAAO;GAAS,OAAO,KAAK;GAAa;EAC3C;GAAE,OAAO;GAAc,OAAO,KAAK;GAAW;EAC9C;GAAE,OAAO;GAAa,OAAO,KAAK;GAAU;EAC5C;GAAE,OAAO;GAAS,OAAO,KAAK;GAAO;EACrC;GACE,OAAO;GACP,OACE,KAAK,MAAM,SAAS,IAClB,oBAACA;IAAK,KAAK;cACR,KAAK,MAAM,KAAK,SACf,oBAAC;KAAiB,MAAK;KAAK,SAAQ;eACjC;OADS,KAEJ,CACR;KACG,GACL;GACP;EACD;GACE,OAAO;GACP,OAAO,OAAO,EAAE,KAAK,WAAW,EAAE,MAAM,OAAO,CAAC,CAAC;GAClD;EACD;GACE,OAAO;GACP,OAAO,OAAO,EAAE,KAAK,WAAW,EAAE,MAAM,OAAO,CAAC,CAAC;GAClD;EACF,GACD"}
|
|
@@ -0,0 +1,109 @@
|
|
|
1
|
+
import { i as useDialog, r as DataTable, s as Text, u as Flex } from "./core-2xoLiT0o.js";
|
|
2
|
+
import { t } from "alepha";
|
|
3
|
+
import { useI18n } from "alepha/react/i18n";
|
|
4
|
+
import { jsx, jsxs } from "react/jsx-runtime";
|
|
5
|
+
import { useState } from "react";
|
|
6
|
+
import { IconDeviceDesktop, IconDeviceMobile, IconDeviceTablet, IconTrash } from "@tabler/icons-react";
|
|
7
|
+
import { useRouterState } from "alepha/react/router";
|
|
8
|
+
import { useClient } from "alepha/react";
|
|
9
|
+
|
|
10
|
+
//#region ../../src/admin/components/users/AdminUserSessions.tsx
|
|
11
|
+
const emptyFilters = t.object({});
|
|
12
|
+
const getDeviceIcon = (device) => {
|
|
13
|
+
switch (device) {
|
|
14
|
+
case "MOBILE": return /* @__PURE__ */ jsx(IconDeviceMobile, { size: 14 });
|
|
15
|
+
case "TABLET": return /* @__PURE__ */ jsx(IconDeviceTablet, { size: 14 });
|
|
16
|
+
default: return /* @__PURE__ */ jsx(IconDeviceDesktop, { size: 14 });
|
|
17
|
+
}
|
|
18
|
+
};
|
|
19
|
+
const isExpired = (expiresAt) => new Date(expiresAt) < /* @__PURE__ */ new Date();
|
|
20
|
+
const AdminUserSessions = (props) => {
|
|
21
|
+
const userId = useRouterState().params.userId;
|
|
22
|
+
const client = useClient();
|
|
23
|
+
const { l } = useI18n();
|
|
24
|
+
const dialog = useDialog();
|
|
25
|
+
const [refreshKey, setRefreshKey] = useState(0);
|
|
26
|
+
const handleDeleteSession = async (sessionId) => {
|
|
27
|
+
if (await dialog.confirm({
|
|
28
|
+
title: "Revoke Session",
|
|
29
|
+
message: "Are you sure you want to revoke this session?"
|
|
30
|
+
})) {
|
|
31
|
+
await client.deleteSession({
|
|
32
|
+
params: { id: sessionId },
|
|
33
|
+
query: { userRealmName: props.userRealmName }
|
|
34
|
+
});
|
|
35
|
+
setRefreshKey((k) => k + 1);
|
|
36
|
+
}
|
|
37
|
+
};
|
|
38
|
+
return /* @__PURE__ */ jsx(DataTable, {
|
|
39
|
+
submitOnInit: true,
|
|
40
|
+
defaultSize: 10,
|
|
41
|
+
filters: emptyFilters,
|
|
42
|
+
tableProps: {
|
|
43
|
+
horizontalSpacing: "xs",
|
|
44
|
+
verticalSpacing: "xs"
|
|
45
|
+
},
|
|
46
|
+
tableTrProps: (item) => ({ style: { opacity: isExpired(item.expiresAt) ? .5 : 1 } }),
|
|
47
|
+
items: async (filters) => {
|
|
48
|
+
return await client.findSessions({ query: {
|
|
49
|
+
...filters,
|
|
50
|
+
userId,
|
|
51
|
+
userRealmName: props.userRealmName
|
|
52
|
+
} });
|
|
53
|
+
},
|
|
54
|
+
columns: {
|
|
55
|
+
device: {
|
|
56
|
+
label: "Device",
|
|
57
|
+
value: (item) => /* @__PURE__ */ jsxs(Flex, {
|
|
58
|
+
gap: 4,
|
|
59
|
+
align: "center",
|
|
60
|
+
children: [getDeviceIcon(item.userAgent?.device), /* @__PURE__ */ jsx(Text, {
|
|
61
|
+
size: "xs",
|
|
62
|
+
children: item.userAgent ? `${item.userAgent.browser} / ${item.userAgent.os}` : "—"
|
|
63
|
+
})]
|
|
64
|
+
})
|
|
65
|
+
},
|
|
66
|
+
ip: {
|
|
67
|
+
label: "IP",
|
|
68
|
+
fit: true,
|
|
69
|
+
value: (item) => /* @__PURE__ */ jsx(Text, {
|
|
70
|
+
size: "xs",
|
|
71
|
+
ff: "monospace",
|
|
72
|
+
c: "dimmed",
|
|
73
|
+
children: item.ip || "—"
|
|
74
|
+
})
|
|
75
|
+
},
|
|
76
|
+
status: {
|
|
77
|
+
label: "Status",
|
|
78
|
+
fit: true,
|
|
79
|
+
value: (item) => /* @__PURE__ */ jsx(Text, {
|
|
80
|
+
size: "xs",
|
|
81
|
+
c: "dimmed",
|
|
82
|
+
children: isExpired(item.expiresAt) ? "Expired" : "Active"
|
|
83
|
+
})
|
|
84
|
+
},
|
|
85
|
+
createdAt: {
|
|
86
|
+
label: "Created",
|
|
87
|
+
fit: true,
|
|
88
|
+
value: (item) => /* @__PURE__ */ jsx(Text, {
|
|
89
|
+
size: "xs",
|
|
90
|
+
c: "dimmed",
|
|
91
|
+
children: l(item.createdAt, { date: "fromNow" })
|
|
92
|
+
})
|
|
93
|
+
},
|
|
94
|
+
actions: {
|
|
95
|
+
label: "",
|
|
96
|
+
fit: true,
|
|
97
|
+
actions: (item) => [{
|
|
98
|
+
icon: /* @__PURE__ */ jsx(IconTrash, { size: 14 }),
|
|
99
|
+
onClick: () => handleDeleteSession(item.id),
|
|
100
|
+
tooltip: "Revoke session"
|
|
101
|
+
}]
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
}, refreshKey);
|
|
105
|
+
};
|
|
106
|
+
|
|
107
|
+
//#endregion
|
|
108
|
+
export { AdminUserSessions as default };
|
|
109
|
+
//# sourceMappingURL=AdminUserSessions-1uzcx02z.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"AdminUserSessions-1uzcx02z.js","names":[],"sources":["../../src/admin/components/users/AdminUserSessions.tsx"],"sourcesContent":["import { DataTable, Flex, Text, useDialog } from \"@alepha/ui\";\nimport {\n IconDeviceDesktop,\n IconDeviceMobile,\n IconDeviceTablet,\n IconTrash,\n} from \"@tabler/icons-react\";\nimport { type Page, t } from \"alepha\";\nimport type { AdminSessionController, SessionEntity } from \"alepha/api/users\";\nimport { useClient } from \"alepha/react\";\nimport { useI18n } from \"alepha/react/i18n\";\nimport { useRouterState } from \"alepha/react/router\";\nimport { useState } from \"react\";\n\nexport interface AdminUserSessionsProps {\n userRealmName?: string;\n}\n\nconst emptyFilters = t.object({});\n\nconst 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\nconst isExpired = (expiresAt: Date | string) =>\n new Date(expiresAt) < new Date();\n\nconst AdminUserSessions = (props: AdminUserSessionsProps) => {\n const state = useRouterState();\n const userId = state.params.userId as string;\n const client = useClient<AdminSessionController>();\n const { l } = useI18n();\n const dialog = useDialog();\n const [refreshKey, setRefreshKey] = useState(0);\n\n const handleDeleteSession = async (sessionId: string) => {\n const confirmed = await dialog.confirm({\n title: \"Revoke Session\",\n message: \"Are you sure you want to revoke this session?\",\n });\n if (confirmed) {\n await client.deleteSession({\n params: { id: sessionId },\n query: { userRealmName: props.userRealmName },\n });\n setRefreshKey((k) => k + 1);\n }\n };\n\n return (\n <DataTable<SessionEntity, typeof emptyFilters>\n key={refreshKey}\n submitOnInit\n defaultSize={10}\n filters={emptyFilters}\n tableProps={{\n horizontalSpacing: \"xs\",\n verticalSpacing: \"xs\",\n }}\n tableTrProps={(item) => ({\n style: {\n opacity: isExpired(item.expiresAt) ? 0.5 : 1,\n },\n })}\n items={async (filters) => {\n const response = await client.findSessions({\n query: {\n ...filters,\n userId,\n userRealmName: props.userRealmName,\n },\n });\n return response as Page<SessionEntity>;\n }}\n columns={{\n device: {\n label: \"Device\",\n value: (item) => (\n <Flex gap={4} align=\"center\">\n {getDeviceIcon(item.userAgent?.device)}\n <Text size=\"xs\">\n {item.userAgent\n ? `${item.userAgent.browser} / ${item.userAgent.os}`\n : \"\\u2014\"}\n </Text>\n </Flex>\n ),\n },\n ip: {\n label: \"IP\",\n fit: true,\n value: (item) => (\n <Text size=\"xs\" ff=\"monospace\" c=\"dimmed\">\n {item.ip || \"\\u2014\"}\n </Text>\n ),\n },\n status: {\n label: \"Status\",\n fit: true,\n value: (item) => (\n <Text size=\"xs\" c=\"dimmed\">\n {isExpired(item.expiresAt) ? \"Expired\" : \"Active\"}\n </Text>\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 actions: (item) => [\n {\n icon: <IconTrash size={14} />,\n onClick: () => handleDeleteSession(item.id),\n tooltip: \"Revoke session\",\n },\n ],\n },\n }}\n />\n );\n};\n\nexport default AdminUserSessions;\n"],"mappings":";;;;;;;;;;AAkBA,MAAM,eAAe,EAAE,OAAO,EAAE,CAAC;AAEjC,MAAM,iBAAiB,WAAoB;AACzC,SAAQ,QAAR;EACE,KAAK,SACH,QAAO,oBAAC,oBAAiB,MAAM,KAAM;EACvC,KAAK,SACH,QAAO,oBAAC,oBAAiB,MAAM,KAAM;EACvC,QACE,QAAO,oBAAC,qBAAkB,MAAM,KAAM;;;AAI5C,MAAM,aAAa,cACjB,IAAI,KAAK,UAAU,mBAAG,IAAI,MAAM;AAElC,MAAM,qBAAqB,UAAkC;CAE3D,MAAM,SADQ,gBAAgB,CACT,OAAO;CAC5B,MAAM,SAAS,WAAmC;CAClD,MAAM,EAAE,MAAM,SAAS;CACvB,MAAM,SAAS,WAAW;CAC1B,MAAM,CAAC,YAAY,iBAAiB,SAAS,EAAE;CAE/C,MAAM,sBAAsB,OAAO,cAAsB;AAKvD,MAJkB,MAAM,OAAO,QAAQ;GACrC,OAAO;GACP,SAAS;GACV,CAAC,EACa;AACb,SAAM,OAAO,cAAc;IACzB,QAAQ,EAAE,IAAI,WAAW;IACzB,OAAO,EAAE,eAAe,MAAM,eAAe;IAC9C,CAAC;AACF,kBAAe,MAAM,IAAI,EAAE;;;AAI/B,QACE,oBAAC;EAEC;EACA,aAAa;EACb,SAAS;EACT,YAAY;GACV,mBAAmB;GACnB,iBAAiB;GAClB;EACD,eAAe,UAAU,EACvB,OAAO,EACL,SAAS,UAAU,KAAK,UAAU,GAAG,KAAM,GAC5C,EACF;EACD,OAAO,OAAO,YAAY;AAQxB,UAPiB,MAAM,OAAO,aAAa,EACzC,OAAO;IACL,GAAG;IACH;IACA,eAAe,MAAM;IACtB,EACF,CAAC;;EAGJ,SAAS;GACP,QAAQ;IACN,OAAO;IACP,QAAQ,SACN,qBAAC;KAAK,KAAK;KAAG,OAAM;gBACjB,cAAc,KAAK,WAAW,OAAO,EACtC,oBAAC;MAAK,MAAK;gBACR,KAAK,YACF,GAAG,KAAK,UAAU,QAAQ,KAAK,KAAK,UAAU,OAC9C;OACC;MACF;IAEV;GACD,IAAI;IACF,OAAO;IACP,KAAK;IACL,QAAQ,SACN,oBAAC;KAAK,MAAK;KAAK,IAAG;KAAY,GAAE;eAC9B,KAAK,MAAM;MACP;IAEV;GACD,QAAQ;IACN,OAAO;IACP,KAAK;IACL,QAAQ,SACN,oBAAC;KAAK,MAAK;KAAK,GAAE;eACf,UAAU,KAAK,UAAU,GAAG,YAAY;MACpC;IAEV;GACD,WAAW;IACT,OAAO;IACP,KAAK;IACL,QAAQ,SACN,oBAAC;KAAK,MAAK;KAAK,GAAE;eACf,EAAE,KAAK,WAAW,EAAE,MAAM,WAAW,CAAC;MAClC;IAEV;GACD,SAAS;IACP,OAAO;IACP,KAAK;IACL,UAAU,SAAS,CACjB;KACE,MAAM,oBAAC,aAAU,MAAM,KAAM;KAC7B,eAAe,oBAAoB,KAAK,GAAG;KAC3C,SAAS;KACV,CACF;IACF;GACF;IA3EI,WA4EL"}
|