@mesob/auth-react 0.5.10 → 0.6.0
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/{chunk-ORQZZUVL.js → chunk-2SYBZ6TR.js} +2 -4
- package/dist/chunk-2SYBZ6TR.js.map +1 -0
- package/dist/{chunk-LI7WPOVY.js → chunk-323NYGKW.js} +3 -6
- package/dist/chunk-323NYGKW.js.map +1 -0
- package/dist/{chunk-Y6KURGWG.js → chunk-34AJJ2CI.js} +2 -4
- package/dist/chunk-34AJJ2CI.js.map +1 -0
- package/dist/{chunk-WAMZL5CS.js → chunk-4O3LAHTY.js} +4 -7
- package/dist/chunk-4O3LAHTY.js.map +1 -0
- package/dist/{chunk-4YPLJ2P6.js → chunk-57G5Z44Y.js} +2 -6
- package/dist/chunk-57G5Z44Y.js.map +1 -0
- package/dist/{chunk-NJGVOQIU.js → chunk-5O6VWABF.js} +5 -8
- package/dist/chunk-5O6VWABF.js.map +1 -0
- package/dist/{chunk-UV7JR3YU.js → chunk-5PGLBU4A.js} +4 -6
- package/dist/chunk-5PGLBU4A.js.map +1 -0
- package/dist/{chunk-IUCTHMVY.js → chunk-5YAYZDKX.js} +2 -5
- package/dist/chunk-5YAYZDKX.js.map +1 -0
- package/dist/{chunk-M2K6O5CN.js → chunk-6BLYK6D6.js} +85 -29
- package/dist/chunk-6BLYK6D6.js.map +1 -0
- package/dist/{chunk-G2RWFKGF.js → chunk-6HVUVXQB.js} +2 -4
- package/dist/chunk-6HVUVXQB.js.map +1 -0
- package/dist/chunk-6YHUCPJ4.js +28 -0
- package/dist/chunk-6YHUCPJ4.js.map +1 -0
- package/dist/{chunk-B7BS57X7.js → chunk-7FSFDTQR.js} +10 -11
- package/dist/chunk-7FSFDTQR.js.map +1 -0
- package/dist/{chunk-22KICA5N.js → chunk-ABIQ2NM3.js} +3 -5
- package/dist/{chunk-22KICA5N.js.map → chunk-ABIQ2NM3.js.map} +1 -1
- package/dist/{chunk-TT3V6PC7.js → chunk-AHWUP6LB.js} +4 -6
- package/dist/chunk-AHWUP6LB.js.map +1 -0
- package/dist/{chunk-M677DPBR.js → chunk-AKJ3EHND.js} +2 -6
- package/dist/chunk-AKJ3EHND.js.map +1 -0
- package/dist/{chunk-J7NROVB4.js → chunk-CY3MODZU.js} +2 -2
- package/dist/{chunk-QNPK2H5A.js → chunk-F5SAYP67.js} +3 -5
- package/dist/{chunk-QNPK2H5A.js.map → chunk-F5SAYP67.js.map} +1 -1
- package/dist/{chunk-I46PN4JU.js → chunk-F6WCSBHX.js} +2 -5
- package/dist/chunk-F6WCSBHX.js.map +1 -0
- package/dist/{chunk-25DJGLNU.js → chunk-FWYDZVFS.js} +9 -10
- package/dist/chunk-FWYDZVFS.js.map +1 -0
- package/dist/{chunk-75K2SCNC.js → chunk-HXAIYFI4.js} +3 -6
- package/dist/chunk-HXAIYFI4.js.map +1 -0
- package/dist/{chunk-7QJBDRTL.js → chunk-IFXBZY6Q.js} +3 -6
- package/dist/chunk-IFXBZY6Q.js.map +1 -0
- package/dist/{chunk-UIXR5GF3.js → chunk-ITKSPHYO.js} +2 -2
- package/dist/{chunk-HOROLWBY.js → chunk-J4X3CA3V.js} +2 -4
- package/dist/chunk-J4X3CA3V.js.map +1 -0
- package/dist/{chunk-GP4XI5KB.js → chunk-L5UKZAA4.js} +3 -3
- package/dist/{chunk-VTANFZKG.js → chunk-LGAHTK5V.js} +5 -7
- package/dist/chunk-LGAHTK5V.js.map +1 -0
- package/dist/{chunk-EWXK56WQ.js → chunk-LNG736CV.js} +1 -1
- package/dist/{chunk-DQB4WY5T.js → chunk-LVA77T4L.js} +4 -6
- package/dist/chunk-LVA77T4L.js.map +1 -0
- package/dist/{chunk-WY2F7475.js → chunk-NNYLSQ2X.js} +9 -10
- package/dist/chunk-NNYLSQ2X.js.map +1 -0
- package/dist/{chunk-CBR5NTFM.js → chunk-OCFX4MBQ.js} +10 -11
- package/dist/chunk-OCFX4MBQ.js.map +1 -0
- package/dist/{chunk-XTLFZ77E.js → chunk-OMAJKLVW.js} +18 -20
- package/dist/chunk-OMAJKLVW.js.map +1 -0
- package/dist/{chunk-DRAUYDZ5.js → chunk-OUBYTCTD.js} +2 -4
- package/dist/chunk-OUBYTCTD.js.map +1 -0
- package/dist/{chunk-KL2XZKDU.js → chunk-Q5RYLX6Z.js} +2 -5
- package/dist/chunk-Q5RYLX6Z.js.map +1 -0
- package/dist/{chunk-OT2H5EHA.js → chunk-RGGHVAAK.js} +2 -5
- package/dist/chunk-RGGHVAAK.js.map +1 -0
- package/dist/{chunk-22WSB5V2.js → chunk-RI647FTV.js} +9 -10
- package/dist/chunk-RI647FTV.js.map +1 -0
- package/dist/{chunk-QRYUUXNJ.js → chunk-SCSRGIEL.js} +4 -7
- package/dist/chunk-SCSRGIEL.js.map +1 -0
- package/dist/chunk-SEBNQYIE.js +30 -0
- package/dist/chunk-SEBNQYIE.js.map +1 -0
- package/dist/{chunk-GWRMQSME.js → chunk-VQYNQ5X7.js} +7 -11
- package/dist/chunk-VQYNQ5X7.js.map +1 -0
- package/dist/{chunk-NPA7L57G.js → chunk-WG5H5PTL.js} +4 -4
- package/dist/chunk-WG5H5PTL.js.map +1 -0
- package/dist/{chunk-5FNUPWPO.js → chunk-WLKT5YFP.js} +12 -5
- package/dist/chunk-WLKT5YFP.js.map +1 -0
- package/dist/{chunk-2BF2JIDK.js → chunk-WM2ETQIY.js} +9 -10
- package/dist/chunk-WM2ETQIY.js.map +1 -0
- package/dist/{chunk-MPZAPUVR.js → chunk-WTWZOQTD.js} +4 -6
- package/dist/chunk-WTWZOQTD.js.map +1 -0
- package/dist/{chunk-VWGOCWRF.js → chunk-X6FJMSL3.js} +9 -10
- package/dist/chunk-X6FJMSL3.js.map +1 -0
- package/dist/{chunk-R7VVXH5U.js → chunk-YPFDH2E7.js} +3 -6
- package/dist/chunk-YPFDH2E7.js.map +1 -0
- package/dist/{chunk-LYCBL2W3.js → chunk-Z47HLNM4.js} +4 -7
- package/dist/chunk-Z47HLNM4.js.map +1 -0
- package/dist/{chunk-ZIUAYN37.js → chunk-ZHRM4QOO.js} +2 -2
- package/dist/components/auth/countdown.js +3 -3
- package/dist/components/auth/forgot-password.js +4 -3
- package/dist/components/auth/reset-password-form.js +4 -3
- package/dist/components/auth/set-password.js +4 -3
- package/dist/components/auth/sign-in.js +4 -3
- package/dist/components/auth/sign-up.js +4 -3
- package/dist/components/auth/verification-form.js +4 -4
- package/dist/components/auth/verify-email.js +6 -5
- package/dist/components/auth/verify-phone.js +6 -5
- package/dist/components/authorization/deny.js +1 -2
- package/dist/components/authorization/grant.js +1 -2
- package/dist/components/iam/domains-page.js +7 -6
- package/dist/components/iam/iam-guard.js +2 -3
- package/dist/components/iam/permission-selector.js +1 -2
- package/dist/components/iam/permissions-page.js +3 -4
- package/dist/components/iam/permissions.js +1 -2
- package/dist/components/iam/role-detail-layout.js +2 -2
- package/dist/components/iam/role-detail-page.js +3 -4
- package/dist/components/iam/role-permissions-page.js +4 -5
- package/dist/components/iam/roles-page.js +7 -6
- package/dist/components/iam/roles.js +1 -2
- package/dist/components/iam/sessions-page.js +7 -6
- package/dist/components/iam/sessions.js +1 -2
- package/dist/components/iam/tenants-page.js +8 -7
- package/dist/components/iam/tenants.js +1 -2
- package/dist/components/iam/users-page.js +8 -7
- package/dist/components/iam/users.js +1 -2
- package/dist/components/profile/account.js +1 -2
- package/dist/components/profile/change-email-form.js +8 -8
- package/dist/components/profile/change-password-form.js +1 -2
- package/dist/components/profile/change-phone-form.js +8 -8
- package/dist/components/profile/otp-verification-modal.js +5 -5
- package/dist/components/profile/profile-layout.js +4 -3
- package/dist/components/profile/request-change-email-form.js +1 -2
- package/dist/components/profile/request-change-phone-form.js +1 -2
- package/dist/components/profile/security.js +13 -13
- package/dist/components/profile/verify-change-email-form.js +6 -6
- package/dist/components/profile/verify-change-phone-form.js +6 -6
- package/dist/index.js +66 -61
- package/dist/index.js.map +1 -1
- package/dist/pages/auth/layout.js +11 -9
- package/dist/pages/auth/layout.js.map +1 -1
- package/dist/pages/auth/route.d.ts +4 -0
- package/dist/pages/auth/route.js +44 -0
- package/dist/pages/auth/route.js.map +1 -0
- package/dist/pages/iam/shared/navigation.d.ts +1 -1
- package/dist/pages/iam/tenants/tenant-selector.d.ts +1 -0
- package/dist/pages/iam/tenants/tenant-selector.js +5 -5
- package/dist/pages/iam/tenants/tenants-data.d.ts +1 -0
- package/dist/pages/iam/tenants/tenants-data.js +1 -1
- package/dist/pages/iam/users/user-selector.d.ts +1 -0
- package/dist/pages/iam/users/user-selector.js +5 -5
- package/dist/pages/iam/users/users-data.d.ts +1 -0
- package/dist/pages/profile/account.js +1 -3
- package/dist/pages/profile/account.js.map +1 -1
- package/dist/pages/profile/security.js +1 -3
- package/dist/pages/profile/security.js.map +1 -1
- package/dist/providers/nuqs-adapter.d.ts +6 -0
- package/dist/utils/navigation.d.ts +5 -0
- package/package.json +43 -8
- package/dist/chunk-22WSB5V2.js.map +0 -1
- package/dist/chunk-25DJGLNU.js.map +0 -1
- package/dist/chunk-2BF2JIDK.js.map +0 -1
- package/dist/chunk-4YPLJ2P6.js.map +0 -1
- package/dist/chunk-5FNUPWPO.js.map +0 -1
- package/dist/chunk-75K2SCNC.js.map +0 -1
- package/dist/chunk-7QJBDRTL.js.map +0 -1
- package/dist/chunk-B7BS57X7.js.map +0 -1
- package/dist/chunk-CBR5NTFM.js.map +0 -1
- package/dist/chunk-DQB4WY5T.js.map +0 -1
- package/dist/chunk-DRAUYDZ5.js.map +0 -1
- package/dist/chunk-G2RWFKGF.js.map +0 -1
- package/dist/chunk-GWRMQSME.js.map +0 -1
- package/dist/chunk-HOROLWBY.js.map +0 -1
- package/dist/chunk-I46PN4JU.js.map +0 -1
- package/dist/chunk-IUCTHMVY.js.map +0 -1
- package/dist/chunk-KL2XZKDU.js.map +0 -1
- package/dist/chunk-LI7WPOVY.js.map +0 -1
- package/dist/chunk-LYCBL2W3.js.map +0 -1
- package/dist/chunk-M2K6O5CN.js.map +0 -1
- package/dist/chunk-M677DPBR.js.map +0 -1
- package/dist/chunk-MPZAPUVR.js.map +0 -1
- package/dist/chunk-NJGVOQIU.js.map +0 -1
- package/dist/chunk-NPA7L57G.js.map +0 -1
- package/dist/chunk-ORQZZUVL.js.map +0 -1
- package/dist/chunk-OT2H5EHA.js.map +0 -1
- package/dist/chunk-QRYUUXNJ.js.map +0 -1
- package/dist/chunk-R7VVXH5U.js.map +0 -1
- package/dist/chunk-SLIIENXJ.js +0 -34
- package/dist/chunk-SLIIENXJ.js.map +0 -1
- package/dist/chunk-TT3V6PC7.js.map +0 -1
- package/dist/chunk-UV7JR3YU.js.map +0 -1
- package/dist/chunk-VTANFZKG.js.map +0 -1
- package/dist/chunk-VWGOCWRF.js.map +0 -1
- package/dist/chunk-WAMZL5CS.js.map +0 -1
- package/dist/chunk-WY2F7475.js.map +0 -1
- package/dist/chunk-XTLFZ77E.js.map +0 -1
- package/dist/chunk-Y6KURGWG.js.map +0 -1
- /package/dist/{chunk-J7NROVB4.js.map → chunk-CY3MODZU.js.map} +0 -0
- /package/dist/{chunk-UIXR5GF3.js.map → chunk-ITKSPHYO.js.map} +0 -0
- /package/dist/{chunk-GP4XI5KB.js.map → chunk-L5UKZAA4.js.map} +0 -0
- /package/dist/{chunk-EWXK56WQ.js.map → chunk-LNG736CV.js.map} +0 -0
- /package/dist/{chunk-ZIUAYN37.js.map → chunk-ZHRM4QOO.js.map} +0 -0
|
@@ -1,11 +1,9 @@
|
|
|
1
1
|
import {
|
|
2
2
|
DataTable
|
|
3
3
|
} from "./chunk-TFVBER3Y.js";
|
|
4
|
-
import {
|
|
5
|
-
useApi
|
|
6
|
-
} from "./chunk-M2K6O5CN.js";
|
|
7
4
|
|
|
8
5
|
// src/components/iam/users.tsx
|
|
6
|
+
import { useApi } from "@mesob/auth-react";
|
|
9
7
|
import { Badge, Button } from "@mesob/ui/components";
|
|
10
8
|
import { useState } from "react";
|
|
11
9
|
import { jsx, jsxs } from "react/jsx-runtime";
|
|
@@ -104,4 +102,4 @@ function Users() {
|
|
|
104
102
|
export {
|
|
105
103
|
Users
|
|
106
104
|
};
|
|
107
|
-
//# sourceMappingURL=chunk-
|
|
105
|
+
//# sourceMappingURL=chunk-2SYBZ6TR.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/iam/users.tsx"],"sourcesContent":["'use client';\n\nimport { useApi } from '@mesob/auth-react';\nimport { Badge, Button } from '@mesob/ui/components';\nimport { useState } from 'react';\nimport { DataTable, type DataTableColumn } from '../shared/data-table';\n\n// User type from OpenAPI schema\ntype User = {\n id: string;\n fullName: string;\n email: string | null;\n phone: string | null;\n handle: string;\n emailVerified: boolean;\n phoneVerified: boolean;\n lastSignInAt: string | null;\n};\n\nexport function Users() {\n const { hooks } = useApi();\n const [page, setPage] = useState(1);\n const limit = 20;\n\n // Use openapi-react-query hooks\n const { data, isLoading, error } = hooks.useQuery('get', '/users', {\n params: {\n query: {\n page: String(page),\n limit: String(limit),\n },\n },\n });\n\n const columns: DataTableColumn<User>[] = [\n {\n key: 'fullName',\n header: 'Name',\n cell: (user) => (\n <div>\n <p className=\"font-medium\">{user.fullName}</p>\n </div>\n ),\n },\n {\n key: 'contact',\n header: 'Contact',\n cell: (user) => (\n <div className=\"space-y-1\">\n {user.email && (\n <div className=\"flex items-center gap-2\">\n <p className=\"text-sm\">{user.email}</p>\n {user.emailVerified && (\n <Badge variant=\"outline\" className=\"text-xs\">\n Verified\n </Badge>\n )}\n </div>\n )}\n {user.phone && (\n <div className=\"flex items-center gap-2\">\n <p className=\"text-sm\">{user.phone}</p>\n {user.phoneVerified && (\n <Badge variant=\"outline\" className=\"text-xs\">\n Verified\n </Badge>\n )}\n </div>\n )}\n </div>\n ),\n },\n {\n key: 'lastSignIn',\n header: 'Last Sign In',\n cell: (user) => (\n <p className=\"text-sm\">\n {user.lastSignInAt\n ? new Date(user.lastSignInAt).toLocaleDateString()\n : 'Never'}\n </p>\n ),\n },\n {\n key: 'actions',\n header: 'Actions',\n cell: (_user) => (\n <div className=\"flex gap-2\">\n <Button variant=\"outline\" size=\"sm\">\n View\n </Button>\n <Button variant=\"outline\" size=\"sm\">\n Edit\n </Button>\n </div>\n ),\n },\n ];\n\n if (error) {\n return (\n <div className=\"p-6 text-center\">\n <p className=\"text-destructive\">Error loading users</p>\n </div>\n );\n }\n\n return (\n <div className=\"w-full p-6 space-y-4\">\n <div className=\"flex justify-between items-center\">\n <div>\n <h1 className=\"text-3xl font-bold\">Users</h1>\n <p className=\"text-muted-foreground\">Manage user accounts</p>\n </div>\n <Button>Create User</Button>\n </div>\n\n <DataTable\n data={(data as { users: User[] })?.users || []}\n columns={columns}\n isLoading={isLoading}\n emptyMessage=\"No users found\"\n />\n\n {data &&\n 'users' in data &&\n data.users &&\n (data.users as User[]).length >= limit && (\n <div className=\"flex justify-between items-center\">\n <Button\n variant=\"outline\"\n disabled={page === 1}\n onClick={() => setPage((prev) => prev - 1)}\n >\n Previous\n </Button>\n <span className=\"text-sm text-muted-foreground\">Page {page}</span>\n <Button\n variant=\"outline\"\n onClick={() => setPage((prev) => prev + 1)}\n >\n Next\n </Button>\n </div>\n )}\n </div>\n );\n}\n"],"mappings":"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","names":[]}
|
|
@@ -3,13 +3,10 @@ import {
|
|
|
3
3
|
} from "./chunk-NPW7D2HZ.js";
|
|
4
4
|
import {
|
|
5
5
|
IAMGuard
|
|
6
|
-
} from "./chunk-
|
|
7
|
-
import {
|
|
8
|
-
useApi,
|
|
9
|
-
useConfig
|
|
10
|
-
} from "./chunk-M2K6O5CN.js";
|
|
6
|
+
} from "./chunk-ZHRM4QOO.js";
|
|
11
7
|
|
|
12
8
|
// src/components/iam/permissions-page.tsx
|
|
9
|
+
import { useApi, useConfig } from "@mesob/auth-react";
|
|
13
10
|
import {
|
|
14
11
|
Button,
|
|
15
12
|
EntityFilter,
|
|
@@ -278,4 +275,4 @@ function PermissionsPageContent() {
|
|
|
278
275
|
export {
|
|
279
276
|
PermissionsPage
|
|
280
277
|
};
|
|
281
|
-
//# sourceMappingURL=chunk-
|
|
278
|
+
//# sourceMappingURL=chunk-323NYGKW.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/iam/permissions-page.tsx","../src/pages/iam/permissions/_components/permissions-list.tsx","../src/pages/iam/permissions/_components/permission-card.tsx"],"sourcesContent":["'use client';\n\nimport { useApi, useConfig } from '@mesob/auth-react';\nimport {\n Button,\n EntityFilter,\n EntityHeader,\n EntitySearch,\n EntitySort,\n EntityViewToggle,\n PageBody,\n PageContainer,\n useBreadcrumbs,\n useEntityPagination,\n useEntityParams,\n} from '@mesob/ui/components';\nimport { IconKey } from '@tabler/icons-react';\nimport { useQueryClient } from '@tanstack/react-query';\nimport { toast } from 'sonner';\nimport type { paths } from '../../data/openapi';\nimport { defaultEntityQueryOptions } from '../../lib/query-options';\nimport { PermissionsList } from '../../pages/iam/permissions/_components/permissions-list';\nimport { IAMGuard } from './iam-guard';\n\nexport function PermissionsPage() {\n return (\n <IAMGuard>\n <PermissionsPageContent />\n </IAMGuard>\n );\n}\n\nfunction PermissionsPageContent() {\n const { hooks } = useApi();\n const { config } = useConfig();\n const homeHref = config.navigation?.defaultRedirectUrl || '/';\n const qc = useQueryClient();\n useBreadcrumbs({\n items: [\n { label: 'Home', href: homeHref },\n { label: 'IAM', href: '/iam/permissions' },\n { label: 'Permissions' },\n ],\n });\n const { queryConfig, params, setParams } = useEntityParams({\n searchKey: 'search',\n defaultSort: 'id',\n defaultOrder: 'asc',\n });\n const permissionsQuery = queryConfig as {\n params: {\n query: NonNullable<paths['/permissions']['get']['parameters']['query']>;\n };\n };\n\n const { data, isPending, isFetching } = hooks.useQuery(\n 'get',\n '/permissions',\n permissionsQuery,\n defaultEntityQueryOptions,\n );\n\n const seedPermissions = hooks.useMutation('post', '/permissions/seed', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/permissions'] });\n toast.success('Permissions seeded');\n },\n onError: () => {\n toast.error('Failed to seed permissions');\n },\n });\n\n const isLoading = isPending || isFetching;\n const permissions = data?.permissions ?? [];\n const { total, pageCount } = useEntityPagination({\n items: permissions,\n total: data?.total,\n pageSize: params.pageSize,\n });\n\n return (\n <PageContainer className=\"flex flex-1 flex-col gap-4 p-4 pt-0\">\n <PageBody className=\"px-0\">\n <EntityHeader\n icon={<IconKey className=\"h-5 w-5\" />}\n title=\"Permissions\"\n actions={\n <Button\n variant=\"outline\"\n onClick={() => seedPermissions.mutate({})}\n loading={seedPermissions.isPending}\n >\n Seed Permissions\n </Button>\n }\n search={\n <EntitySearch\n paramKey=\"search\"\n placeholder=\"Search permissions...\"\n />\n }\n filter={\n <EntityFilter\n options={[\n { label: 'All', value: '' },\n { label: 'Application', value: 'application' },\n { label: 'Feature', value: 'feature' },\n { label: 'Activity', value: 'activity' },\n ]}\n placeholder=\"Search field\"\n />\n }\n sort={\n <EntitySort\n defaultSort=\"id\"\n defaultOrder=\"asc\"\n options={[\n { label: 'ID', value: 'id' },\n { label: 'Application', value: 'application' },\n { label: 'Feature', value: 'feature' },\n { label: 'Activity', value: 'activity' },\n ]}\n />\n }\n view={<EntityViewToggle views={['table', 'card']} />}\n />\n <PermissionsList\n data={permissions}\n isLoading={isLoading}\n view={(params.view || 'table') as 'table' | 'card'}\n pageIndex={params.page - 1}\n pageSize={params.pageSize}\n pageCount={pageCount}\n totalRows={total}\n onPageChange={(p) => setParams({ page: p + 1 })}\n onPageSizeChange={(size) => setParams({ pageSize: size, page: 1 })}\n />\n </PageBody>\n </PageContainer>\n );\n}\n","'use client';\n\nimport {\n Badge,\n DataTablePagination,\n DisplayTable,\n EntityEmptyState,\n EntityLoadingState,\n Tbody,\n Td,\n Th,\n Thead,\n Tr,\n} from '@mesob/ui/components';\nimport { IconKey } from '@tabler/icons-react';\nimport { PermissionCard } from './permission-card';\nimport type { Permission } from './permissions-data';\n\nconst TABLE_COLUMN_COUNT = 4;\n\nfunction descStr(d: unknown): string {\n if (d == null) {\n return '';\n }\n if (typeof d === 'string') {\n return d;\n }\n if (typeof d === 'object' && d !== null && 'en' in d) {\n return String((d as { en?: string }).en ?? '');\n }\n return String(d);\n}\n\ntype PermissionsListProps = {\n data: Permission[];\n isLoading?: boolean;\n view: 'table' | 'card';\n pageIndex: number;\n pageSize: number;\n pageCount: number;\n totalRows: number;\n onPageChange: (page: number) => void;\n onPageSizeChange: (size: number) => void;\n};\n\nexport function PermissionsList({\n data,\n isLoading,\n view,\n pageIndex,\n pageSize,\n pageCount,\n totalRows,\n onPageChange,\n onPageSizeChange,\n}: PermissionsListProps) {\n if (isLoading) {\n return (\n <EntityLoadingState\n view={view}\n rowCount={pageSize}\n columnCount={TABLE_COLUMN_COUNT}\n cardCount={pageSize}\n />\n );\n }\n if (totalRows === 0) {\n return (\n <EntityEmptyState\n icon={IconKey}\n entityName=\"permission\"\n title=\"No permissions\"\n description=\"Permissions are managed by the system.\"\n />\n );\n }\n if (view === 'table') {\n return (\n <div className=\"space-y-4\">\n <DisplayTable withTableBorder>\n <Thead>\n <Tr>\n <Th>Application</Th>\n <Th>Feature</Th>\n <Th>Activity</Th>\n <Th>Description</Th>\n </Tr>\n </Thead>\n <Tbody>\n {data.map((p) => (\n <Tr key={p.id}>\n <Td>\n <Badge variant=\"secondary\">{p.application}</Badge>\n </Td>\n <Td>\n <Badge variant=\"outline\">{p.feature}</Badge>\n </Td>\n <Td>\n <span className=\"font-medium\">{p.activity}</span>\n </Td>\n <Td>\n <span className=\"text-muted-foreground line-clamp-1 max-w-[200px]\">\n {descStr(p.description) || '—'}\n </span>\n </Td>\n </Tr>\n ))}\n </Tbody>\n </DisplayTable>\n <DataTablePagination\n pageIndex={pageIndex}\n pageSize={pageSize}\n pageCount={pageCount}\n totalRows={totalRows}\n onPageChange={onPageChange}\n onPageSizeChange={onPageSizeChange}\n />\n </div>\n );\n }\n return (\n <div className=\"space-y-4\">\n <div className=\"grid grid-cols-1 gap-4 md:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4\">\n {data.map((p) => (\n <PermissionCard key={p.id} permission={p} />\n ))}\n </div>\n <DataTablePagination\n pageIndex={pageIndex}\n pageSize={pageSize}\n pageCount={pageCount}\n totalRows={totalRows}\n onPageChange={onPageChange}\n onPageSizeChange={onPageSizeChange}\n />\n </div>\n );\n}\n","'use client';\n\nimport { Badge, Card, CardContent, CardHeader } from '@mesob/ui/components';\nimport type { Permission } from './permissions-data';\n\ntype PermissionCardProps = { permission: Permission };\n\nfunction descStr(d: unknown): string {\n if (d == null) {\n return '';\n }\n if (typeof d === 'string') {\n return d;\n }\n if (typeof d === 'object' && d !== null && 'en' in d) {\n return String((d as { en?: string }).en ?? '');\n }\n return String(d);\n}\n\nexport function PermissionCard({ permission }: PermissionCardProps) {\n return (\n <Card className=\"hover:shadow-md transition-shadow\">\n <CardHeader className=\"pb-2\">\n <p className=\"font-semibold\">{permission.activity}</p>\n <div className=\"flex flex-wrap gap-1\">\n <Badge variant=\"secondary\">{permission.application}</Badge>\n <Badge variant=\"outline\">{permission.feature}</Badge>\n </div>\n </CardHeader>\n <CardContent>\n {descStr(permission.description) && (\n <p className=\"text-sm text-muted-foreground line-clamp-2\">\n {descStr(permission.description)}\n </p>\n )}\n </CardContent>\n </Card>\n );\n}\n"],"mappings":"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","names":["IconKey","Badge","jsx","jsxs","descStr","Badge","jsx","jsxs","IconKey"]}
|
|
@@ -1,11 +1,9 @@
|
|
|
1
1
|
import {
|
|
2
2
|
DataTable
|
|
3
3
|
} from "./chunk-TFVBER3Y.js";
|
|
4
|
-
import {
|
|
5
|
-
useApi
|
|
6
|
-
} from "./chunk-M2K6O5CN.js";
|
|
7
4
|
|
|
8
5
|
// src/components/iam/tenants.tsx
|
|
6
|
+
import { useApi } from "@mesob/auth-react";
|
|
9
7
|
import { Badge, Button } from "@mesob/ui/components";
|
|
10
8
|
import { useState } from "react";
|
|
11
9
|
import { jsx, jsxs } from "react/jsx-runtime";
|
|
@@ -101,4 +99,4 @@ function Tenants() {
|
|
|
101
99
|
export {
|
|
102
100
|
Tenants
|
|
103
101
|
};
|
|
104
|
-
//# sourceMappingURL=chunk-
|
|
102
|
+
//# sourceMappingURL=chunk-34AJJ2CI.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/iam/tenants.tsx"],"sourcesContent":["'use client';\n\nimport { useApi } from '@mesob/auth-react';\nimport { Badge, Button } from '@mesob/ui/components';\nimport { useState } from 'react';\nimport { DataTable, type DataTableColumn } from '../shared/data-table';\n\n// Tenant type from OpenAPI schema\ntype Tenant = {\n id: string;\n name: string;\n slug: string;\n status: string;\n createdAt: string;\n};\n\nexport function Tenants() {\n const { hooks } = useApi();\n const [page, setPage] = useState(1);\n const limit = 20;\n\n // Use openapi-react-query hooks\n const { data, isLoading, error } = hooks.useQuery('get', '/tenants', {\n params: {\n query: {\n page: String(page),\n limit: String(limit),\n },\n },\n });\n\n const columns: DataTableColumn<Tenant>[] = [\n {\n key: 'name',\n header: 'Tenant',\n cell: (tenant) => (\n <div>\n <p className=\"font-medium\">{tenant.name}</p>\n <p className=\"text-sm text-muted-foreground\">/{tenant.slug}</p>\n </div>\n ),\n },\n {\n key: 'status',\n header: 'Status',\n cell: (tenant) => (\n <Badge variant={tenant.status === 'active' ? 'default' : 'secondary'}>\n {tenant.status}\n </Badge>\n ),\n },\n {\n key: 'createdAt',\n header: 'Created',\n cell: (tenant) => (\n <p className=\"text-sm\">\n {new Date(tenant.createdAt).toLocaleDateString()}\n </p>\n ),\n },\n {\n key: 'actions',\n header: 'Actions',\n cell: (_tenant) => (\n <div className=\"flex gap-2\">\n <Button variant=\"outline\" size=\"sm\">\n Domains\n </Button>\n <Button variant=\"outline\" size=\"sm\">\n Edit\n </Button>\n </div>\n ),\n },\n ];\n\n if (error) {\n return (\n <div className=\"p-6 text-center\">\n <p className=\"text-destructive\">Error loading tenants</p>\n </div>\n );\n }\n\n return (\n <div className=\"w-full p-6 space-y-4\">\n <div className=\"flex justify-between items-center\">\n <div>\n <h1 className=\"text-3xl font-bold\">Tenants</h1>\n <p className=\"text-muted-foreground\">Manage tenant organizations</p>\n </div>\n <Button>Create Tenant</Button>\n </div>\n\n <DataTable\n data={(data as { tenants: Tenant[] })?.tenants || []}\n columns={columns}\n isLoading={isLoading}\n emptyMessage=\"No tenants found\"\n />\n\n {data &&\n 'tenants' in data &&\n data.tenants &&\n (data.tenants as Tenant[]).length >= limit && (\n <div className=\"flex justify-between items-center\">\n <Button\n variant=\"outline\"\n disabled={page === 1}\n onClick={() => setPage((prev) => prev - 1)}\n >\n Previous\n </Button>\n <span className=\"text-sm text-muted-foreground\">Page {page}</span>\n <Button\n variant=\"outline\"\n onClick={() => setPage((prev) => prev + 1)}\n >\n Next\n </Button>\n </div>\n )}\n </div>\n );\n}\n"],"mappings":";;;;;AAEA,SAAS,cAAc;AACvB,SAAS,OAAO,cAAc;AAC9B,SAAS,gBAAgB;AAiCf,cACA,YADA;AArBH,SAAS,UAAU;AACxB,QAAM,EAAE,MAAM,IAAI,OAAO;AACzB,QAAM,CAAC,MAAM,OAAO,IAAI,SAAS,CAAC;AAClC,QAAM,QAAQ;AAGd,QAAM,EAAE,MAAM,WAAW,MAAM,IAAI,MAAM,SAAS,OAAO,YAAY;AAAA,IACnE,QAAQ;AAAA,MACN,OAAO;AAAA,QACL,MAAM,OAAO,IAAI;AAAA,QACjB,OAAO,OAAO,KAAK;AAAA,MACrB;AAAA,IACF;AAAA,EACF,CAAC;AAED,QAAM,UAAqC;AAAA,IACzC;AAAA,MACE,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,MAAM,CAAC,WACL,qBAAC,SACC;AAAA,4BAAC,OAAE,WAAU,eAAe,iBAAO,MAAK;AAAA,QACxC,qBAAC,OAAE,WAAU,iCAAgC;AAAA;AAAA,UAAE,OAAO;AAAA,WAAK;AAAA,SAC7D;AAAA,IAEJ;AAAA,IACA;AAAA,MACE,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,MAAM,CAAC,WACL,oBAAC,SAAM,SAAS,OAAO,WAAW,WAAW,YAAY,aACtD,iBAAO,QACV;AAAA,IAEJ;AAAA,IACA;AAAA,MACE,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,MAAM,CAAC,WACL,oBAAC,OAAE,WAAU,WACV,cAAI,KAAK,OAAO,SAAS,EAAE,mBAAmB,GACjD;AAAA,IAEJ;AAAA,IACA;AAAA,MACE,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,MAAM,CAAC,YACL,qBAAC,SAAI,WAAU,cACb;AAAA,4BAAC,UAAO,SAAQ,WAAU,MAAK,MAAK,qBAEpC;AAAA,QACA,oBAAC,UAAO,SAAQ,WAAU,MAAK,MAAK,kBAEpC;AAAA,SACF;AAAA,IAEJ;AAAA,EACF;AAEA,MAAI,OAAO;AACT,WACE,oBAAC,SAAI,WAAU,mBACb,8BAAC,OAAE,WAAU,oBAAmB,mCAAqB,GACvD;AAAA,EAEJ;AAEA,SACE,qBAAC,SAAI,WAAU,wBACb;AAAA,yBAAC,SAAI,WAAU,qCACb;AAAA,2BAAC,SACC;AAAA,4BAAC,QAAG,WAAU,sBAAqB,qBAAO;AAAA,QAC1C,oBAAC,OAAE,WAAU,yBAAwB,yCAA2B;AAAA,SAClE;AAAA,MACA,oBAAC,UAAO,2BAAa;AAAA,OACvB;AAAA,IAEA;AAAA,MAAC;AAAA;AAAA,QACC,MAAO,MAAgC,WAAW,CAAC;AAAA,QACnD;AAAA,QACA;AAAA,QACA,cAAa;AAAA;AAAA,IACf;AAAA,IAEC,QACC,aAAa,QACb,KAAK,WACJ,KAAK,QAAqB,UAAU,SACnC,qBAAC,SAAI,WAAU,qCACb;AAAA;AAAA,QAAC;AAAA;AAAA,UACC,SAAQ;AAAA,UACR,UAAU,SAAS;AAAA,UACnB,SAAS,MAAM,QAAQ,CAAC,SAAS,OAAO,CAAC;AAAA,UAC1C;AAAA;AAAA,MAED;AAAA,MACA,qBAAC,UAAK,WAAU,iCAAgC;AAAA;AAAA,QAAM;AAAA,SAAK;AAAA,MAC3D;AAAA,QAAC;AAAA;AAAA,UACC,SAAQ;AAAA,UACR,SAAS,MAAM,QAAQ,CAAC,SAAS,OAAO,CAAC;AAAA,UAC1C;AAAA;AAAA,MAED;AAAA,OACF;AAAA,KAEN;AAEJ;","names":[]}
|
|
@@ -1,18 +1,15 @@
|
|
|
1
1
|
import {
|
|
2
2
|
authApi$
|
|
3
|
-
} from "./chunk-
|
|
3
|
+
} from "./chunk-F5SAYP67.js";
|
|
4
4
|
import {
|
|
5
5
|
defaultEntityQueryOptions
|
|
6
6
|
} from "./chunk-NPW7D2HZ.js";
|
|
7
7
|
import {
|
|
8
8
|
IAMGuard
|
|
9
|
-
} from "./chunk-
|
|
10
|
-
import {
|
|
11
|
-
useApi,
|
|
12
|
-
useConfig
|
|
13
|
-
} from "./chunk-M2K6O5CN.js";
|
|
9
|
+
} from "./chunk-ZHRM4QOO.js";
|
|
14
10
|
|
|
15
11
|
// src/components/iam/domains-page.tsx
|
|
12
|
+
import { useApi, useConfig } from "@mesob/auth-react";
|
|
16
13
|
import {
|
|
17
14
|
EntityDrawerTrigger,
|
|
18
15
|
EntityHeader,
|
|
@@ -598,4 +595,4 @@ function DomainsPageContent() {
|
|
|
598
595
|
export {
|
|
599
596
|
DomainsPage
|
|
600
597
|
};
|
|
601
|
-
//# sourceMappingURL=chunk-
|
|
598
|
+
//# sourceMappingURL=chunk-4O3LAHTY.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/iam/domains-page.tsx","../src/pages/iam/domains/_components/domain-form.tsx","../src/pages/iam/domains/_components/domains-list.tsx","../src/pages/iam/domains/_components/domain-card.tsx"],"sourcesContent":["'use client';\n\nimport { useApi, useConfig } from '@mesob/auth-react';\nimport {\n EntityDrawerTrigger,\n EntityHeader,\n EntitySearch,\n EntitySort,\n EntityViewToggle,\n PageBody,\n PageContainer,\n Select,\n SelectContent,\n SelectItem,\n SelectTrigger,\n useBreadcrumbs,\n useEntityPagination,\n useEntityParams,\n} from '@mesob/ui/components';\nimport { IconWorld } from '@tabler/icons-react';\nimport { useMemo, useState } from 'react';\nimport type { paths } from '../../data/openapi';\nimport { defaultEntityQueryOptions } from '../../lib/query-options';\nimport { DomainForm } from '../../pages/iam/domains/_components/domain-form';\nimport { DomainsList } from '../../pages/iam/domains/_components/domains-list';\nimport { IAMGuard } from './iam-guard';\n\nexport function DomainsPage() {\n return (\n <IAMGuard>\n <DomainsPageContent />\n </IAMGuard>\n );\n}\n\nfunction DomainsPageContent() {\n const { hooks } = useApi();\n const { config } = useConfig();\n const homeHref = config.navigation?.defaultRedirectUrl || '/';\n useBreadcrumbs({\n items: [\n { label: 'Home', href: homeHref },\n { label: 'IAM', href: '/iam/users' },\n { label: 'Domains' },\n ],\n });\n const [createOpen, setCreateOpen] = useState(false);\n\n const { params, setParams } = useEntityParams({\n searchKey: 'search',\n searchParamName: 'search',\n });\n const [statusFilter, setStatusFilter] = useState<string>('');\n\n const domainsQuery = useMemo(\n () =>\n ({\n params: {\n query: {\n page: params.page,\n limit: params.pageSize,\n ...(statusFilter\n ? {\n status: statusFilter as\n | 'pending'\n | 'active'\n | 'disabled'\n | 'deleted',\n }\n : {}),\n },\n },\n }) as {\n params: {\n query: NonNullable<paths['/domains']['get']['parameters']['query']>;\n };\n },\n [params.page, params.pageSize, statusFilter],\n );\n\n const { data, isPending, isFetching } = hooks.useQuery(\n 'get',\n '/domains',\n domainsQuery,\n defaultEntityQueryOptions,\n );\n\n const isLoading = isPending || isFetching;\n const domains = data?.domains ?? [];\n const { total, pageCount } = useEntityPagination({\n items: domains,\n total: data?.total,\n pageSize: params.pageSize,\n });\n\n return (\n <PageContainer className=\"flex flex-1 flex-col gap-4 p-4 pt-0\">\n <PageBody className=\"px-0\">\n <EntityHeader\n icon={<IconWorld className=\"h-5 w-5\" />}\n title=\"Domains\"\n actions={\n <EntityDrawerTrigger\n mode=\"new\"\n entity=\"Domain\"\n label=\"Add domain\"\n open={createOpen}\n onOpenChange={setCreateOpen}\n >\n {(open, onClose) => (\n <DomainForm\n open={open}\n onClose={onClose}\n mode=\"new\"\n onSuccess={() => setCreateOpen(false)}\n />\n )}\n </EntityDrawerTrigger>\n }\n search={\n <EntitySearch paramKey=\"search\" placeholder=\"Search domains...\" />\n }\n filter={\n <Select\n value={statusFilter || 'all'}\n onValueChange={(v) => {\n setStatusFilter(v === 'all' || v == null ? '' : v);\n setParams({ page: 1 });\n }}\n >\n <SelectTrigger className=\"h-9 min-w-[120px] w-[120px]\">\n <span className={statusFilter ? '' : 'text-muted-foreground'}>\n {statusFilter ? statusFilter : 'Status'}\n </span>\n </SelectTrigger>\n <SelectContent>\n <SelectItem value=\"all\">All</SelectItem>\n <SelectItem value=\"pending\">Pending</SelectItem>\n <SelectItem value=\"active\">Active</SelectItem>\n <SelectItem value=\"disabled\">Disabled</SelectItem>\n <SelectItem value=\"deleted\">Deleted</SelectItem>\n </SelectContent>\n </Select>\n }\n sort={\n <EntitySort\n defaultSort=\"updatedAt\"\n defaultOrder=\"desc\"\n options={[\n { label: 'Domain', value: 'domain' },\n { label: 'Status', value: 'status' },\n { label: 'Updated', value: 'updatedAt' },\n { label: 'Created', value: 'createdAt' },\n ]}\n />\n }\n view={<EntityViewToggle views={['table', 'card']} />}\n />\n <DomainsList\n data={domains}\n isLoading={isLoading}\n view={(params.view || 'table') as 'table' | 'card'}\n pageIndex={params.page - 1}\n pageSize={params.pageSize}\n pageCount={pageCount}\n totalRows={total}\n onCreateNew={() => setCreateOpen(true)}\n onPageChange={(p) => setParams({ page: p + 1 })}\n onPageSizeChange={(size) => setParams({ pageSize: size, page: 1 })}\n />\n </PageBody>\n </PageContainer>\n );\n}\n","'use client';\n\nimport { zodResolver } from '@hookform/resolvers/zod';\nimport {\n Checkbox,\n EntityDrawer,\n EntityFormActions,\n Form,\n FormControl,\n FormField,\n FormItem,\n FormLabel,\n FormMessage,\n Input,\n Select,\n SelectContent,\n SelectItem,\n SelectTrigger,\n Skeleton,\n Stack,\n} from '@mesob/ui/components';\nimport { useQueryClient } from '@tanstack/react-query';\nimport { useEffect } from 'react';\nimport { useForm } from 'react-hook-form';\nimport { z } from 'zod';\nimport { authApi$ } from '../../shared/page-helpers';\n\nconst statusOptions = [\n { value: 'pending', label: 'Pending' },\n { value: 'active', label: 'Active' },\n { value: 'disabled', label: 'Disabled' },\n { value: 'deleted', label: 'Deleted' },\n] as const;\n\nconst schema = z.object({\n domain: z\n .string()\n .min(1, 'Domain is required')\n .regex(\n /^[a-zA-Z0-9]([a-zA-Z0-9-]*[a-zA-Z0-9])?(\\.[a-zA-Z0-9]([a-zA-Z0-9-]*[a-zA-Z0-9])?)+$/,\n 'Invalid domain format (e.g. example.com)',\n ),\n status: z.enum(['pending', 'active', 'disabled', 'deleted']),\n isPrimary: z.boolean(),\n});\n\ntype FormData = z.infer<typeof schema>;\n\nconst defaults: FormData = {\n domain: '',\n status: 'pending',\n isPrimary: false,\n};\n\ntype DomainFormProps = {\n mode: 'new' | 'edit';\n domainId?: string;\n open: boolean;\n onClose: () => void;\n onSuccess?: () => void;\n};\n\nexport function DomainForm({\n mode,\n domainId,\n open,\n onClose,\n onSuccess,\n}: DomainFormProps) {\n const qc = useQueryClient();\n const { data, isLoading } = authApi$.useQuery(\n 'get',\n '/domains/{id}',\n { params: { path: { id: domainId ?? '' } } },\n { enabled: mode === 'edit' && !!domainId && open },\n );\n const domain = data?.domain;\n\n const form = useForm<FormData>({\n resolver: zodResolver(schema),\n defaultValues: defaults,\n });\n const { control, formState, reset } = form;\n\n useEffect(() => {\n if (!open) {\n return;\n }\n if (mode === 'edit' && domain && !isLoading) {\n reset({\n domain: domain.domain,\n status: (domain.status?.toLowerCase() ??\n 'pending') as FormData['status'],\n isPrimary: domain.isPrimary ?? false,\n });\n } else {\n reset(defaults);\n }\n }, [mode, domain, open, isLoading, reset]);\n\n const create = authApi$.useMutation('post', '/domains', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/domains'] });\n },\n });\n const update = authApi$.useMutation('put', '/domains/{id}', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/domains'] });\n if (domainId) {\n qc.invalidateQueries({ queryKey: ['get', '/domains/{id}'] });\n }\n },\n });\n const del = authApi$.useMutation('delete', '/domains/{id}', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/domains'] });\n },\n });\n\n const onSubmit = form.handleSubmit(async (d) => {\n if (mode === 'new') {\n await create.mutateAsync({\n body: {\n domain: d.domain,\n status: d.status,\n isPrimary: d.isPrimary,\n },\n });\n } else if (domainId) {\n await update.mutateAsync({\n params: { path: { id: domainId } },\n body: {\n domain: d.domain,\n status: d.status,\n isPrimary: d.isPrimary,\n },\n });\n }\n onSuccess?.();\n onClose();\n });\n\n const onDelete = async () => {\n if (!domainId) {\n return;\n }\n await del.mutateAsync({ params: { path: { id: domainId } } });\n onSuccess?.();\n onClose();\n };\n\n const isSubmitting = create.isPending || update.isPending;\n\n return (\n <EntityDrawer\n title={mode === 'new' ? 'New domain' : 'Edit domain'}\n open={open}\n onClose={onClose}\n isDirty={formState.isDirty}\n form={\n isLoading && mode === 'edit' ? (\n <FormSkeleton />\n ) : (\n <Form {...form}>\n <form onSubmit={onSubmit} className=\"space-y-6\">\n <Stack>\n <FormField\n control={control}\n name=\"domain\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>\n Domain <span className=\"text-destructive\">*</span>\n </FormLabel>\n <FormControl>\n <Input\n placeholder=\"example.com\"\n {...field}\n disabled={mode === 'edit'}\n />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n <FormField\n control={control}\n name=\"status\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>Status</FormLabel>\n <Select\n value={field.value}\n onValueChange={(v) =>\n field.onChange(v as FormData['status'])\n }\n >\n <FormControl>\n <SelectTrigger>\n {statusOptions.find((o) => o.value === field.value)\n ?.label ?? field.value}\n </SelectTrigger>\n </FormControl>\n <SelectContent>\n {statusOptions.map((o) => (\n <SelectItem key={o.value} value={o.value}>\n {o.label}\n </SelectItem>\n ))}\n </SelectContent>\n </Select>\n <FormMessage />\n </FormItem>\n )}\n />\n <FormField\n control={control}\n name=\"isPrimary\"\n render={({ field }) => (\n <FormItem className=\"flex flex-row items-center gap-2 space-y-0\">\n <FormControl>\n <Checkbox\n size=\"lg\"\n checked={field.value}\n onCheckedChange={(c) => field.onChange(c === true)}\n />\n </FormControl>\n <FormLabel className=\"cursor-pointer font-normal\">\n Primary domain\n </FormLabel>\n <FormMessage />\n </FormItem>\n )}\n />\n </Stack>\n </form>\n </Form>\n )\n }\n actions={\n <EntityFormActions\n mode={mode}\n onSubmit={onSubmit}\n onReset={mode === 'new' ? () => reset(defaults) : undefined}\n onDelete={mode === 'edit' ? onDelete : undefined}\n isSubmitting={isSubmitting}\n isDeleting={del.isPending}\n disabled={isLoading && mode === 'edit'}\n itemName=\"domain\"\n />\n }\n />\n );\n}\n\nfunction FormSkeleton() {\n return (\n <div className=\"space-y-4\">\n {[1, 2, 3].map((i) => (\n <div key={i} className=\"space-y-2\">\n <Skeleton className=\"h-4 w-20\" />\n <Skeleton className=\"h-10 w-full\" />\n </div>\n ))}\n </div>\n );\n}\n","'use client';\n\nimport {\n Badge,\n DataTableAction,\n DataTablePagination,\n DisplayTable,\n EntityEmptyState,\n EntityLoadingState,\n Tbody,\n Td,\n Th,\n Thead,\n Tr,\n} from '@mesob/ui/components';\nimport { IconWorld } from '@tabler/icons-react';\nimport { useState } from 'react';\nimport { DomainCard } from './domain-card';\nimport { DomainForm } from './domain-form';\nimport type { Domain } from './domains-data';\n\nconst TABLE_COLUMN_COUNT = 5;\n\nfunction statusVariant(\n s: string,\n): 'default' | 'secondary' | 'destructive' | 'outline' {\n const v = s?.toLowerCase();\n if (v === 'active') {\n return 'default';\n }\n if (v === 'pending') {\n return 'secondary';\n }\n if (v === 'disabled' || v === 'deleted') {\n return 'destructive';\n }\n return 'outline';\n}\n\nfunction DomainTableRow({\n domain,\n onEdit,\n}: {\n domain: Domain;\n onEdit: (id: string) => void;\n}) {\n return (\n <Tr className=\"group\">\n <Td className=\"font-medium\">{domain.domain}</Td>\n <Td>\n <Badge variant={statusVariant(domain.status)} className=\"text-xs\">\n {domain.status}\n </Badge>\n </Td>\n <Td>{domain.isPrimary ? 'Yes' : 'No'}</Td>\n <Td className=\"text-muted-foreground text-sm\">\n {domain.updatedAt\n ? new Date(domain.updatedAt).toLocaleDateString()\n : '—'}\n </Td>\n <Td>\n <DataTableAction onClick={() => onEdit(domain.id)} />\n </Td>\n </Tr>\n );\n}\n\ntype DomainsListProps = {\n data: Domain[];\n isLoading?: boolean;\n view: 'table' | 'card';\n pageIndex: number;\n pageSize: number;\n pageCount: number;\n totalRows: number;\n onPageChange: (page: number) => void;\n onPageSizeChange: (size: number) => void;\n onCreateNew?: () => void;\n};\n\nexport function DomainsList({\n data,\n isLoading,\n view,\n pageIndex,\n pageSize,\n pageCount,\n totalRows,\n onPageChange,\n onPageSizeChange,\n onCreateNew,\n}: DomainsListProps) {\n const [editingId, setEditingId] = useState<string | null>(null);\n\n if (isLoading) {\n return (\n <EntityLoadingState\n view={view}\n rowCount={pageSize}\n columnCount={TABLE_COLUMN_COUNT}\n cardCount={pageSize}\n />\n );\n }\n if (totalRows === 0) {\n return (\n <EntityEmptyState\n icon={IconWorld}\n entityName=\"domain\"\n title=\"No domains yet\"\n description=\"Add your first domain to get started.\"\n actionLabel=\"Add domain\"\n onAction={onCreateNew}\n />\n );\n }\n if (view === 'table') {\n return (\n <div className=\"space-y-4\">\n {editingId && (\n <DomainForm\n mode=\"edit\"\n domainId={editingId}\n open\n onClose={() => setEditingId(null)}\n />\n )}\n <DisplayTable withTableBorder>\n <Thead>\n <Tr>\n <Th>Domain</Th>\n <Th>Status</Th>\n <Th>Primary</Th>\n <Th>Updated</Th>\n <Th className=\"w-[50px]\" />\n </Tr>\n </Thead>\n <Tbody>\n {data.map((d) => (\n <DomainTableRow key={d.id} domain={d} onEdit={setEditingId} />\n ))}\n </Tbody>\n </DisplayTable>\n <DataTablePagination\n pageIndex={pageIndex}\n pageSize={pageSize}\n pageCount={pageCount}\n totalRows={totalRows}\n onPageChange={onPageChange}\n onPageSizeChange={onPageSizeChange}\n />\n </div>\n );\n }\n return (\n <div className=\"space-y-4\">\n <div className=\"grid grid-cols-1 gap-4 md:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4\">\n {data.map((d) => (\n <DomainCard key={d.id} domain={d} />\n ))}\n </div>\n <DataTablePagination\n pageIndex={pageIndex}\n pageSize={pageSize}\n pageCount={pageCount}\n totalRows={totalRows}\n onPageChange={onPageChange}\n onPageSizeChange={onPageSizeChange}\n />\n </div>\n );\n}\n","'use client';\n\nimport {\n Badge,\n Button,\n Card,\n CardContent,\n CardHeader,\n DropdownMenu,\n DropdownMenuContent,\n DropdownMenuItem,\n DropdownMenuPortal,\n DropdownMenuTrigger,\n} from '@mesob/ui/components';\nimport { IconDots, IconPencil } from '@tabler/icons-react';\nimport { useState } from 'react';\nimport { DomainForm } from './domain-form';\nimport type { Domain } from './domains-data';\n\ntype DomainCardProps = { domain: Domain };\n\nfunction statusVariant(\n s: string,\n): 'default' | 'secondary' | 'destructive' | 'outline' {\n const v = s?.toLowerCase();\n if (v === 'active') {\n return 'default';\n }\n if (v === 'pending') {\n return 'secondary';\n }\n if (v === 'disabled' || v === 'deleted') {\n return 'destructive';\n }\n return 'outline';\n}\n\nexport function DomainCard({ domain }: DomainCardProps) {\n const [editOpen, setEditOpen] = useState(false);\n return (\n <>\n <Card className=\"group hover:shadow-md transition-shadow\">\n <CardHeader className=\"pb-2\">\n <div className=\"flex items-start justify-between gap-2\">\n <span className=\"font-semibold truncate\">{domain.domain}</span>\n <DropdownMenu>\n <DropdownMenuTrigger\n render={\n <Button\n variant=\"ghost\"\n size=\"icon\"\n className=\"h-8 w-8 opacity-0 group-hover:opacity-100 transition-opacity\"\n />\n }\n >\n <IconDots className=\"h-4 w-4\" />\n </DropdownMenuTrigger>\n <DropdownMenuPortal>\n <DropdownMenuContent>\n <DropdownMenuItem onClick={() => setEditOpen(true)}>\n <IconPencil className=\"mr-2 h-4 w-4\" />\n Edit\n </DropdownMenuItem>\n </DropdownMenuContent>\n </DropdownMenuPortal>\n </DropdownMenu>\n </div>\n </CardHeader>\n <CardContent className=\"space-y-2\">\n <div className=\"flex flex-wrap gap-1\">\n <Badge variant={statusVariant(domain.status)} className=\"text-xs\">\n {domain.status}\n </Badge>\n {domain.isPrimary && (\n <Badge variant=\"outline\" className=\"text-xs\">\n Primary\n </Badge>\n )}\n </div>\n <p className=\"text-xs text-muted-foreground\">\n Updated{' '}\n {domain.updatedAt\n ? new Date(domain.updatedAt).toLocaleDateString()\n : '—'}\n </p>\n </CardContent>\n </Card>\n {editOpen && (\n <DomainForm\n mode=\"edit\"\n domainId={domain.id}\n open={editOpen}\n onClose={() => setEditOpen(false)}\n />\n )}\n </>\n );\n}\n"],"mappings":"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","names":["Select","SelectContent","SelectItem","SelectTrigger","IconWorld","useState","Badge","useState","jsx","jsxs","jsx","jsxs","statusVariant","Badge","useState","jsx","jsxs","useState","IconWorld","Select","SelectTrigger","SelectContent","SelectItem"]}
|
|
@@ -1,14 +1,10 @@
|
|
|
1
1
|
import {
|
|
2
2
|
normalizePhone
|
|
3
3
|
} from "./chunk-DBTKXQV7.js";
|
|
4
|
-
import {
|
|
5
|
-
useApi,
|
|
6
|
-
useConfig,
|
|
7
|
-
useSession
|
|
8
|
-
} from "./chunk-M2K6O5CN.js";
|
|
9
4
|
|
|
10
5
|
// src/components/profile/request-change-phone-form.tsx
|
|
11
6
|
import { zodResolver } from "@hookform/resolvers/zod";
|
|
7
|
+
import { useApi, useConfig, useSession } from "@mesob/auth-react";
|
|
12
8
|
import { Button, Input, Label, Spinner } from "@mesob/ui/components";
|
|
13
9
|
import { IconEye, IconEyeOff } from "@tabler/icons-react";
|
|
14
10
|
import { useEffect, useState } from "react";
|
|
@@ -263,4 +259,4 @@ function RequestChangePhoneForm({
|
|
|
263
259
|
export {
|
|
264
260
|
RequestChangePhoneForm
|
|
265
261
|
};
|
|
266
|
-
//# sourceMappingURL=chunk-
|
|
262
|
+
//# sourceMappingURL=chunk-57G5Z44Y.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/profile/request-change-phone-form.tsx"],"sourcesContent":["'use client';\n\nimport { zodResolver } from '@hookform/resolvers/zod';\nimport { useApi, useConfig, useSession } from '@mesob/auth-react';\nimport { Button, Input, Label, Spinner } from '@mesob/ui/components';\nimport { IconEye, IconEyeOff } from '@tabler/icons-react';\nimport { useEffect, useState } from 'react';\nimport { useForm } from 'react-hook-form';\nimport { toast } from 'sonner';\nimport { z } from 'zod';\nimport { normalizePhone } from '../../utils/normalize-phone';\n\nconst phonePasswordSchema = (phoneRegex: RegExp) =>\n z.object({\n phone: z\n .string()\n .trim()\n .min(1, { message: 'Phone number is required' })\n .refine((val) => phoneRegex.test(normalizePhone(val)), {\n message: 'Invalid phone number',\n }),\n password: z\n .string()\n .min(8, 'Password must be at least 8 characters')\n .max(128, 'Password too long'),\n });\n\ntype PhonePasswordFormData = z.infer<ReturnType<typeof phonePasswordSchema>>;\n\ntype AuthErrorLike = {\n code?: string;\n message?: string;\n name?: string;\n};\n\nfunction isAuthError(error: unknown): error is AuthErrorLike {\n return (\n typeof error === 'object' &&\n error !== null &&\n ('code' in error || 'message' in error || 'name' in error)\n );\n}\n\nfunction getErrorCode(error: AuthErrorLike): string | undefined {\n if (error.code) {\n return error.code;\n }\n if (error.message) {\n const upperMessage = error.message.toUpperCase().trim();\n const validCodes = [\n 'USER_NOT_FOUND',\n 'USER_EXISTS',\n 'INVALID_PASSWORD',\n 'VERIFICATION_EXPIRED',\n 'VERIFICATION_MISMATCH',\n 'VERIFICATION_NOT_FOUND',\n 'TOO_MANY_ATTEMPTS',\n 'UNAUTHORIZED',\n ];\n if (validCodes.includes(upperMessage)) {\n return upperMessage;\n }\n }\n return undefined;\n}\n\nfunction getErrorMessage(error: unknown): string {\n if (isAuthError(error)) {\n const errorCode = getErrorCode(error);\n switch (errorCode) {\n case 'USER_EXISTS':\n return 'This phone number is already taken. Please use a different number.';\n case 'VERIFICATION_EXPIRED':\n return 'Verification code has expired. Please request a new one.';\n case 'VERIFICATION_MISMATCH':\n return 'Invalid verification code. Please try again.';\n case 'VERIFICATION_NOT_FOUND':\n return 'Verification not found. Please request a new code.';\n default:\n return error.message || 'An error occurred. Please try again.';\n }\n }\n if (error instanceof Error) {\n return error.message;\n }\n return 'An error occurred. Please try again.';\n}\n\ninterface RequestChangePhoneFormProps {\n onSuccess: (verificationId: string, phone: string) => void;\n onCancel: () => void;\n buttonText: string;\n}\n\nexport function RequestChangePhoneForm({\n onSuccess,\n onCancel,\n buttonText,\n}: RequestChangePhoneFormProps) {\n const { user } = useSession();\n const { hooks } = useApi();\n const { config } = useConfig();\n const [isSubmitting, setIsSubmitting] = useState(false);\n const [isChecking, setIsChecking] = useState(true);\n const [showPassword, setShowPassword] = useState(false);\n\n const phoneRegex =\n typeof config.phoneRegex === 'string'\n ? new RegExp(config.phoneRegex)\n : config.phoneRegex || /^\\+251[79]\\d{8}$/;\n\n const getPendingAccountChangeQuery = hooks.useQuery(\n 'get',\n '/account-change/pending',\n {},\n { enabled: false },\n );\n const verifyPasswordMutation = hooks.useMutation('post', '/password/verify');\n const checkUserMutation = hooks.useMutation('post', '/check-account');\n const requestPhoneOtpMutation = hooks.useMutation(\n 'post',\n '/phone/verification/request',\n );\n\n const phonePasswordForm = useForm<PhonePasswordFormData>({\n resolver: zodResolver(phonePasswordSchema(phoneRegex)),\n defaultValues: {\n phone: '',\n password: '',\n },\n });\n\n const {\n register,\n handleSubmit,\n getValues,\n setValue,\n formState: { errors },\n } = phonePasswordForm;\n\n useEffect(() => {\n let active = true;\n const run = async () => {\n try {\n const data = await getPendingAccountChangeQuery.refetch();\n if (!active) {\n return;\n }\n const accountChange = data.data?.accountChange;\n const verificationId = data.data?.verificationId;\n if (accountChange?.changeType !== 'phone') {\n setIsChecking(false);\n return;\n }\n if (!accountChange.newPhone) {\n setIsChecking(false);\n return;\n }\n if (getValues('phone')) {\n setIsChecking(false);\n return;\n }\n setValue('phone', accountChange.newPhone, { shouldValidate: true });\n\n if (verificationId) {\n toast.message('Resuming verification…');\n onSuccess(verificationId, accountChange.newPhone);\n return;\n }\n\n setIsChecking(false);\n } catch {\n setIsChecking(false);\n }\n };\n run().catch(() => undefined);\n return () => {\n active = false;\n };\n }, [getPendingAccountChangeQuery.refetch, getValues, onSuccess, setValue]);\n\n const onPhonePasswordSubmit = async (data: PhonePasswordFormData) => {\n if (!user) {\n toast.error('User not found');\n return;\n }\n\n try {\n setIsSubmitting(true);\n\n const normalizedPhone = normalizePhone(data.phone);\n\n // Verify password first\n await verifyPasswordMutation.mutateAsync({\n body: { password: data.password },\n });\n\n // Check if phone exists\n const checkResult = await checkUserMutation.mutateAsync({\n body: { identifier: normalizedPhone },\n });\n if (checkResult.data?.exists) {\n // Check if it belongs to current user\n if (\n user?.phone?.replace(/\\s/g, '') === normalizedPhone.replace(/\\s/g, '')\n ) {\n toast.error('This is already your current phone number.');\n return;\n }\n toast.error(\n 'This phone number is already taken. Please use a different number.',\n );\n return;\n }\n\n // Request OTP\n const verification = await requestPhoneOtpMutation.mutateAsync({\n body: {\n phone: normalizedPhone,\n context: 'change-phone',\n },\n });\n\n toast.success('Verification code sent to your phone');\n onSuccess(verification.data?.verificationId ?? '', normalizedPhone);\n } catch (error) {\n const errorMessage = getErrorMessage(error);\n if (isAuthError(error)) {\n const errorCode = getErrorCode(error);\n if (\n errorCode === 'INVALID_PASSWORD' ||\n errorCode === 'USER_NOT_FOUND'\n ) {\n toast.error('Incorrect password. Please try again.');\n return;\n }\n }\n toast.error(errorMessage);\n } finally {\n setIsSubmitting(false);\n }\n };\n\n const isLoading = isSubmitting || isChecking;\n\n return (\n <form\n onSubmit={handleSubmit(onPhonePasswordSubmit)}\n className=\"p-4 space-y-4 border-t\"\n >\n <div className=\"space-y-4 w-full md:w-1/2\">\n <div className=\"space-y-2\">\n <Label htmlFor=\"phone\">Phone Number</Label>\n <Input\n id=\"phone\"\n type=\"tel\"\n placeholder=\"Enter your new phone number\"\n {...register('phone')}\n disabled={isLoading}\n />\n {errors.phone && (\n <p className=\"text-sm text-destructive\">{errors.phone.message}</p>\n )}\n </div>\n <div className=\"space-y-2\">\n <Label htmlFor=\"password\">Password</Label>\n <div className=\"relative\">\n <Input\n id=\"password\"\n type={showPassword ? 'text' : 'password'}\n autoComplete=\"current-password\"\n placeholder=\"Enter your password\"\n {...register('password')}\n disabled={isLoading}\n />\n <button\n type=\"button\"\n onClick={() => setShowPassword(!showPassword)}\n className=\"absolute right-3 top-1/2 -translate-y-1/2 text-muted-foreground hover:text-foreground\"\n >\n {showPassword ? (\n <IconEyeOff className=\"h-4 w-4\" />\n ) : (\n <IconEye className=\"h-4 w-4\" />\n )}\n </button>\n </div>\n {errors.password && (\n <p className=\"text-sm text-destructive\">\n {errors.password.message}\n </p>\n )}\n </div>\n </div>\n\n <div className=\"flex justify-end gap-2\">\n <Button\n type=\"button\"\n variant=\"outline\"\n onClick={onCancel}\n disabled={isLoading}\n >\n Cancel\n </Button>\n <Button type=\"submit\" disabled={isLoading}>\n {isLoading && <Spinner className=\"mr-2 h-4 w-4\" />}\n {isChecking ? 'Checking…' : buttonText}\n </Button>\n </div>\n </form>\n );\n}\n"],"mappings":"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","names":[]}
|
|
@@ -1,22 +1,19 @@
|
|
|
1
1
|
import {
|
|
2
2
|
str
|
|
3
|
-
} from "./chunk-
|
|
3
|
+
} from "./chunk-LNG736CV.js";
|
|
4
4
|
import {
|
|
5
5
|
Link,
|
|
6
6
|
authApi$
|
|
7
|
-
} from "./chunk-
|
|
7
|
+
} from "./chunk-F5SAYP67.js";
|
|
8
8
|
import {
|
|
9
9
|
defaultEntityQueryOptions
|
|
10
10
|
} from "./chunk-NPW7D2HZ.js";
|
|
11
11
|
import {
|
|
12
12
|
IAMGuard
|
|
13
|
-
} from "./chunk-
|
|
14
|
-
import {
|
|
15
|
-
useApi,
|
|
16
|
-
useConfig
|
|
17
|
-
} from "./chunk-M2K6O5CN.js";
|
|
13
|
+
} from "./chunk-ZHRM4QOO.js";
|
|
18
14
|
|
|
19
15
|
// src/components/iam/tenants-page.tsx
|
|
16
|
+
import { useApi, useConfig } from "@mesob/auth-react";
|
|
20
17
|
import {
|
|
21
18
|
EntityDrawerTrigger,
|
|
22
19
|
EntityFilter,
|
|
@@ -541,4 +538,4 @@ function TenantsPageContent() {
|
|
|
541
538
|
export {
|
|
542
539
|
TenantsPage
|
|
543
540
|
};
|
|
544
|
-
//# sourceMappingURL=chunk-
|
|
541
|
+
//# sourceMappingURL=chunk-5O6VWABF.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/iam/tenants-page.tsx","../src/pages/iam/tenants/_components/tenant-form.tsx","../src/pages/iam/tenants/_components/tenants-list.tsx","../src/pages/iam/tenants/_components/tenant-card.tsx"],"sourcesContent":["'use client';\n\nimport { useApi, useConfig } from '@mesob/auth-react';\nimport {\n EntityDrawerTrigger,\n EntityFilter,\n EntityHeader,\n EntitySearch,\n EntitySort,\n EntityViewToggle,\n PageBody,\n PageContainer,\n useBreadcrumbs,\n useEntityPagination,\n useEntityParams,\n} from '@mesob/ui/components';\nimport { IconBuilding } from '@tabler/icons-react';\nimport { useState } from 'react';\nimport type { paths } from '../../data/openapi';\nimport { defaultEntityQueryOptions } from '../../lib/query-options';\nimport { TenantForm } from '../../pages/iam/tenants/_components/tenant-form';\nimport { TenantsList } from '../../pages/iam/tenants/_components/tenants-list';\nimport { IAMGuard } from './iam-guard';\n\nexport function TenantsPage() {\n return (\n <IAMGuard>\n <TenantsPageContent />\n </IAMGuard>\n );\n}\n\nfunction TenantsPageContent() {\n const { hooks } = useApi();\n const { config } = useConfig();\n const homeHref = config.navigation?.defaultRedirectUrl || '/';\n useBreadcrumbs({\n items: [\n { label: 'Home', href: homeHref },\n { label: 'IAM', href: '/iam/tenants' },\n { label: 'Tenants' },\n ],\n });\n const [createOpen, setCreateOpen] = useState(false);\n\n const { queryConfig, params, setParams } = useEntityParams({\n searchKey: 'search',\n });\n const tenantsQuery = queryConfig as {\n params: {\n query: NonNullable<paths['/tenants']['get']['parameters']['query']>;\n };\n };\n\n const { data, error, isPending, isFetching } = hooks.useQuery(\n 'get',\n '/tenants',\n tenantsQuery,\n defaultEntityQueryOptions,\n );\n\n const isLoading = isPending || isFetching;\n const tenants = data?.tenants ?? [];\n const { total, pageCount } = useEntityPagination({\n items: tenants,\n total: data?.total,\n pageSize: params.pageSize,\n });\n const errorStatus = (error as { status?: number } | null)?.status;\n const hasAccessError = errorStatus === 401 || errorStatus === 403;\n const hasError = Boolean(error) && !isLoading;\n\n return (\n <PageContainer className=\"flex flex-1 flex-col gap-4 p-4 pt-0\">\n <PageBody className=\"px-0\">\n <EntityHeader\n icon={<IconBuilding className=\"size-5\" />}\n title=\"Tenants\"\n actions={\n hasAccessError ? null : (\n <EntityDrawerTrigger\n mode=\"new\"\n entity=\"Tenant\"\n open={createOpen}\n onOpenChange={setCreateOpen}\n >\n {(open, onClose) => (\n <TenantForm mode=\"new\" open={open} onClose={onClose} />\n )}\n </EntityDrawerTrigger>\n )\n }\n search={\n hasAccessError ? null : (\n <EntitySearch paramKey=\"search\" placeholder=\"Search tenants...\" />\n )\n }\n filter={\n hasAccessError ? null : (\n <EntityFilter\n options={[\n { label: 'All', value: '' },\n { label: 'Active', value: 'isActive:true' },\n { label: 'Inactive', value: 'isActive:false' },\n ]}\n placeholder=\"Filter\"\n />\n )\n }\n sort={\n hasAccessError ? null : (\n <EntitySort\n options={[\n { label: 'Created', value: 'createdAt' },\n { label: 'Updated', value: 'updatedAt' },\n { label: 'Name', value: 'name' },\n ]}\n />\n )\n }\n view={\n hasAccessError ? null : (\n <EntityViewToggle views={['table', 'card']} />\n )\n }\n />\n {hasError ? (\n <div className=\"rounded-[1.75rem] border border-border/60 bg-muted/20 p-8\">\n <div className=\"text-lg font-semibold\">\n {hasAccessError\n ? 'Tenant access denied'\n : 'Unable to load tenants'}\n </div>\n <p className=\"mt-2 text-sm text-muted-foreground\">\n {hasAccessError\n ? 'This account does not have permission to view or manage tenants.'\n : 'The tenants page returned an unexpected error. Retry after the API is healthy.'}\n </p>\n </div>\n ) : (\n <TenantsList\n data={tenants}\n isLoading={isLoading}\n view={(params.view || 'table') as 'table' | 'card'}\n pageIndex={params.page - 1}\n pageSize={params.pageSize}\n pageCount={pageCount}\n totalRows={total}\n onCreateNew={() => setCreateOpen(true)}\n onPageChange={(p) => setParams({ page: p + 1 })}\n onPageSizeChange={(size) => setParams({ pageSize: size, page: 1 })}\n />\n )}\n </PageBody>\n </PageContainer>\n );\n}\n","'use client';\n\nimport { zodResolver } from '@hookform/resolvers/zod';\nimport {\n EntityDrawer,\n EntityFormActions,\n Input,\n Label,\n Skeleton,\n Textarea,\n} from '@mesob/ui/components';\nimport { useQueryClient } from '@tanstack/react-query';\nimport { useEffect } from 'react';\nimport { useForm } from 'react-hook-form';\nimport { z } from 'zod';\nimport { authApi$ } from '../../shared/page-helpers';\nimport { str } from './tenants-data';\n\nconst schema = z.object({\n id: z.string().min(1).max(30),\n name: z.string().optional(),\n description: z.string().optional(),\n isActive: z.boolean(),\n});\n\ntype FormData = z.infer<typeof schema>;\n\nconst defaults: FormData = {\n id: '',\n name: '',\n description: '',\n isActive: true,\n};\n\ntype TenantFormProps = {\n mode: 'new' | 'edit';\n tenantId?: string;\n open: boolean;\n onClose: () => void;\n onSuccess?: () => void;\n};\n\nexport function TenantForm({\n mode,\n tenantId,\n open,\n onClose,\n onSuccess,\n}: TenantFormProps) {\n const qc = useQueryClient();\n const { data, isLoading } = authApi$.useQuery(\n 'get',\n '/tenants/{id}',\n { params: { path: { id: tenantId ?? '' } } },\n { enabled: mode === 'edit' && !!tenantId },\n );\n const create = authApi$.useMutation('post', '/tenants', {\n onSuccess: () => qc.invalidateQueries({ queryKey: ['get', '/tenants'] }),\n });\n const update = authApi$.useMutation('put', '/tenants/{id}', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/tenants'] });\n if (tenantId) {\n qc.invalidateQueries({ queryKey: ['get', '/tenants/{id}'] });\n }\n },\n });\n const del = authApi$.useMutation('delete', '/tenants/{id}', {\n onSuccess: () => qc.invalidateQueries({ queryKey: ['get', '/tenants'] }),\n });\n\n const form = useForm<FormData>({\n resolver: zodResolver(schema),\n defaultValues: defaults,\n });\n\n const { reset, formState, register } = form;\n\n useEffect(() => {\n if (!open) {\n return;\n }\n if (mode === 'edit' && data?.tenant && !isLoading) {\n const t = data.tenant;\n reset({\n id: t.id,\n name: str(t.name) || '',\n description: str(t.description) || '',\n isActive: t.isActive,\n });\n } else {\n reset(defaults);\n }\n }, [mode, data, open, isLoading, reset]);\n\n const onSubmit = form.handleSubmit(async (d) => {\n const body = {\n id: d.id,\n name: d.name || undefined,\n description: d.description || undefined,\n isActive: d.isActive,\n };\n if (mode === 'new') {\n await create.mutateAsync({ body });\n } else if (tenantId) {\n await update.mutateAsync({\n params: { path: { id: tenantId } },\n body: {\n name: d.name || undefined,\n description: d.description || undefined,\n isActive: d.isActive,\n },\n });\n }\n onSuccess?.();\n onClose();\n });\n\n const onDelete = async () => {\n if (!tenantId) {\n return;\n }\n await del.mutateAsync({ params: { path: { id: tenantId } } });\n onSuccess?.();\n onClose();\n };\n\n return (\n <EntityDrawer\n title={mode === 'new' ? 'New tenant' : 'Edit tenant'}\n open={open}\n onClose={onClose}\n isDirty={formState.isDirty}\n size=\"md\"\n form={\n isLoading ? (\n <FormSkeleton />\n ) : (\n <form onSubmit={onSubmit} className=\"space-y-4\">\n <div className=\"space-y-2\">\n <Label htmlFor=\"id\">\n ID <span className=\"text-destructive\">*</span>\n </Label>\n <Input\n id=\"id\"\n placeholder=\"e.g. acme\"\n {...register('id')}\n disabled={mode === 'edit'}\n />\n {formState.errors.id && (\n <p className=\"text-sm text-destructive\">\n {formState.errors.id.message}\n </p>\n )}\n </div>\n <div className=\"space-y-2\">\n <Label htmlFor=\"name\">Name</Label>\n <Input\n id=\"name\"\n placeholder=\"Display name\"\n {...register('name')}\n />\n </div>\n <div className=\"space-y-2\">\n <Label htmlFor=\"description\">Description</Label>\n <Textarea\n id=\"description\"\n placeholder=\"Description\"\n rows={3}\n {...register('description')}\n />\n </div>\n {mode === 'edit' && (\n <div className=\"flex items-center gap-2\">\n <input\n type=\"checkbox\"\n id=\"isActive\"\n {...register('isActive', {\n setValueAs: (v) => v === true || v === 'on',\n })}\n className=\"h-4 w-4\"\n />\n <Label htmlFor=\"isActive\">Active</Label>\n </div>\n )}\n </form>\n )\n }\n actions={\n <EntityFormActions\n mode={mode}\n onSubmit={onSubmit}\n onReset={mode === 'new' ? () => reset(defaults) : undefined}\n onDelete={mode === 'edit' ? onDelete : undefined}\n isSubmitting={create.isPending || update.isPending}\n isDeleting={del.isPending}\n disabled={isLoading}\n itemName=\"tenant\"\n />\n }\n />\n );\n}\n\nfunction FormSkeleton() {\n return (\n <div className=\"space-y-4\">\n <div className=\"space-y-2\">\n <Skeleton className=\"h-4 w-12\" />\n <Skeleton className=\"h-10 w-full\" />\n </div>\n <div className=\"space-y-2\">\n <Skeleton className=\"h-4 w-16\" />\n <Skeleton className=\"h-10 w-full\" />\n </div>\n <div className=\"space-y-2\">\n <Skeleton className=\"h-4 w-24\" />\n <Skeleton className=\"h-20 w-full\" />\n </div>\n </div>\n );\n}\n","'use client';\n\nimport {\n Badge,\n DataTableAction,\n DataTablePagination,\n DisplayTable,\n EntityEmptyState,\n EntityLoadingState,\n Tbody,\n Td,\n Th,\n Thead,\n Tr,\n} from '@mesob/ui/components';\nimport { IconBuilding, IconCalendar } from '@tabler/icons-react';\nimport { useState } from 'react';\nimport { Link } from '../../shared/page-helpers';\nimport { TenantCard } from './tenant-card';\nimport { TenantForm } from './tenant-form';\nimport type { Tenant } from './tenants-data';\nimport { str } from './tenants-data';\n\nconst TABLE_COLUMN_COUNT = 5;\n\ntype TenantsListProps = {\n data: Tenant[];\n isLoading?: boolean;\n view: 'table' | 'card';\n pageIndex: number;\n pageSize: number;\n pageCount: number;\n totalRows: number;\n onPageChange: (page: number) => void;\n onPageSizeChange: (size: number) => void;\n onCreateNew?: () => void;\n};\n\nexport function TenantsList({\n data,\n isLoading,\n view,\n pageIndex,\n pageSize,\n pageCount,\n totalRows,\n onPageChange,\n onPageSizeChange,\n onCreateNew,\n}: TenantsListProps) {\n const [editingTenantId, setEditingTenantId] = useState<string | null>(null);\n\n if (isLoading) {\n return (\n <EntityLoadingState\n view={view}\n rowCount={pageSize}\n columnCount={TABLE_COLUMN_COUNT}\n cardCount={pageSize}\n />\n );\n }\n if (totalRows === 0) {\n return (\n <EntityEmptyState\n icon={IconBuilding}\n entityName=\"tenant\"\n title=\"No tenants yet\"\n description=\"Create your first tenant to get started.\"\n onAction={onCreateNew}\n />\n );\n }\n if (view === 'table') {\n return (\n <div className=\"space-y-4\">\n {editingTenantId && (\n <TenantForm\n mode=\"edit\"\n tenantId={editingTenantId}\n open\n onClose={() => setEditingTenantId(null)}\n />\n )}\n <DisplayTable withTableBorder>\n <Thead>\n <Tr>\n <Th>Tenant</Th>\n <Th>Description</Th>\n <Th>Status</Th>\n <Th>Created</Th>\n <Th className=\"w-[50px]\" />\n </Tr>\n </Thead>\n <Tbody>\n {data.map((tenant) => (\n <Tr key={tenant.id} className=\"group\">\n <Td>\n <Link\n href={`/iam/tenants/${tenant.id}`}\n className=\"block text-left font-medium hover:text-primary hover:underline cursor-pointer\"\n >\n <p>{str(tenant.name) || tenant.id}</p>\n <p className=\"text-sm text-muted-foreground\">{tenant.id}</p>\n </Link>\n </Td>\n <Td>\n <span className=\"text-muted-foreground line-clamp-1 max-w-[200px]\">\n {str(tenant.description) || '—'}\n </span>\n </Td>\n <Td>\n <Badge variant={tenant.isActive ? 'default' : 'secondary'}>\n {tenant.isActive ? 'Active' : 'Inactive'}\n </Badge>\n </Td>\n <Td>\n <div className=\"flex items-center gap-1 text-muted-foreground\">\n <IconCalendar className=\"h-4 w-4\" />\n {new Date(tenant.createdAt).toLocaleDateString()}\n </div>\n </Td>\n <Td>\n <DataTableAction\n onClick={() => setEditingTenantId(tenant.id)}\n />\n </Td>\n </Tr>\n ))}\n </Tbody>\n </DisplayTable>\n <DataTablePagination\n pageIndex={pageIndex}\n pageSize={pageSize}\n pageCount={pageCount}\n totalRows={totalRows}\n onPageChange={onPageChange}\n onPageSizeChange={onPageSizeChange}\n />\n </div>\n );\n }\n return (\n <div className=\"space-y-4\">\n <div className=\"grid grid-cols-1 gap-4 md:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4\">\n {data.map((t) => (\n <TenantCard key={t.id} tenant={t} />\n ))}\n </div>\n <DataTablePagination\n pageIndex={pageIndex}\n pageSize={pageSize}\n pageCount={pageCount}\n totalRows={totalRows}\n onPageChange={onPageChange}\n onPageSizeChange={onPageSizeChange}\n />\n </div>\n );\n}\n","'use client';\n\nimport {\n Badge,\n Button,\n Card,\n CardContent,\n CardHeader,\n DropdownMenu,\n DropdownMenuContent,\n DropdownMenuItem,\n DropdownMenuPortal,\n DropdownMenuTrigger,\n} from '@mesob/ui/components';\nimport { IconDots, IconPencil } from '@tabler/icons-react';\nimport { useState } from 'react';\nimport { Link } from '../../shared/page-helpers';\nimport { TenantForm } from './tenant-form';\nimport { str, type Tenant } from './tenants-data';\n\ntype TenantCardProps = { tenant: Tenant };\n\nexport function TenantCard({ tenant }: TenantCardProps) {\n const [editOpen, setEditOpen] = useState(false);\n return (\n <>\n <Card className=\"group hover:shadow-md transition-shadow\">\n <CardHeader className=\"pb-2\">\n <div className=\"flex items-start justify-between\">\n <Link\n href={`/iam/tenants/${tenant.id}`}\n className=\"text-left font-semibold hover:text-primary hover:underline\"\n >\n {str(tenant.name) || tenant.id}\n </Link>\n <DropdownMenu>\n <DropdownMenuTrigger\n render={\n <Button\n variant=\"ghost\"\n size=\"icon\"\n className=\"h-8 w-8 opacity-0 group-hover:opacity-100 transition-opacity\"\n />\n }\n >\n <IconDots className=\"h-4 w-4\" />\n </DropdownMenuTrigger>\n <DropdownMenuPortal>\n <DropdownMenuContent>\n <DropdownMenuItem onClick={() => setEditOpen(true)}>\n <IconPencil className=\"mr-2 h-4 w-4\" />\n Edit\n </DropdownMenuItem>\n </DropdownMenuContent>\n </DropdownMenuPortal>\n </DropdownMenu>\n </div>\n <p className=\"text-sm text-muted-foreground\">{tenant.id}</p>\n </CardHeader>\n <CardContent className=\"space-y-2\">\n <Badge variant={tenant.isActive ? 'default' : 'secondary'}>\n {tenant.isActive ? 'Active' : 'Inactive'}\n </Badge>\n {str(tenant.description) ? (\n <p className=\"text-sm text-muted-foreground line-clamp-2\">\n {str(tenant.description)}\n </p>\n ) : null}\n <p className=\"text-xs text-muted-foreground\">\n Created {new Date(tenant.createdAt).toLocaleDateString()}\n </p>\n </CardContent>\n </Card>\n {editOpen ? (\n <TenantForm\n mode=\"edit\"\n tenantId={tenant.id}\n open={editOpen}\n onClose={() => setEditOpen(false)}\n />\n ) : null}\n </>\n );\n}\n"],"mappings":"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","names":["IconBuilding","useState","Badge","useState","jsx","jsxs","jsx","jsxs","useState","Badge","jsx","jsxs","useState","IconBuilding"]}
|
|
@@ -1,13 +1,11 @@
|
|
|
1
1
|
import {
|
|
2
2
|
DataTable
|
|
3
3
|
} from "./chunk-TFVBER3Y.js";
|
|
4
|
-
import {
|
|
5
|
-
useApi
|
|
6
|
-
} from "./chunk-M2K6O5CN.js";
|
|
7
4
|
|
|
8
5
|
// src/components/iam/roles.tsx
|
|
6
|
+
import { useApi } from "@mesob/auth-react";
|
|
9
7
|
import { Badge, Button } from "@mesob/ui/components";
|
|
10
|
-
import {
|
|
8
|
+
import { useMesobLocale } from "@mesob/ui/providers";
|
|
11
9
|
import { useState } from "react";
|
|
12
10
|
import { jsx, jsxs } from "react/jsx-runtime";
|
|
13
11
|
function getTranslation(value, locale) {
|
|
@@ -24,7 +22,7 @@ function getTranslation(value, locale) {
|
|
|
24
22
|
}
|
|
25
23
|
function Roles() {
|
|
26
24
|
const { hooks } = useApi();
|
|
27
|
-
const locale =
|
|
25
|
+
const locale = useMesobLocale();
|
|
28
26
|
const [page, setPage] = useState(1);
|
|
29
27
|
const limit = 20;
|
|
30
28
|
const { data, isLoading, error } = hooks.useQuery("get", "/roles", {
|
|
@@ -115,4 +113,4 @@ function Roles() {
|
|
|
115
113
|
export {
|
|
116
114
|
Roles
|
|
117
115
|
};
|
|
118
|
-
//# sourceMappingURL=chunk-
|
|
116
|
+
//# sourceMappingURL=chunk-5PGLBU4A.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/iam/roles.tsx"],"sourcesContent":["'use client';\n\nimport { useApi } from '@mesob/auth-react';\nimport { Badge, Button } from '@mesob/ui/components';\nimport { useMesobLocale } from '@mesob/ui/providers';\nimport { useState } from 'react';\nimport { DataTable, type DataTableColumn } from '../shared/data-table';\n\n// Role type from OpenAPI schema\ntype Role = {\n id: string;\n code: string;\n name: string | { am?: string; en?: string };\n description?: string | { am?: string; en?: string };\n createdAt: string;\n isSystem?: boolean;\n permissionCount?: number;\n};\n\nfunction getTranslation(\n value: string | { am?: string; en?: string } | null | undefined,\n locale: string,\n): string {\n if (!value) {\n return '';\n }\n if (typeof value === 'string') {\n return value;\n }\n if (typeof value === 'object') {\n return value[locale as 'am' | 'en'] ?? value.en ?? value.am ?? '';\n }\n return '';\n}\n\nexport function Roles() {\n const { hooks } = useApi();\n const locale = useMesobLocale();\n const [page, setPage] = useState(1);\n const limit = 20;\n\n // Use openapi-react-query hooks\n const { data, isLoading, error } = hooks.useQuery('get', '/roles', {\n params: {\n query: {\n page: String(page),\n limit: String(limit),\n },\n },\n });\n\n const columns: DataTableColumn<Role>[] = [\n {\n key: 'name',\n header: 'Role',\n cell: (role) => (\n <div>\n <p className=\"font-medium\">{getTranslation(role.name, locale)}</p>\n <Badge variant=\"outline\" className=\"mt-1\">\n {role.code}\n </Badge>\n </div>\n ),\n },\n {\n key: 'access',\n header: 'Access',\n cell: (role) => (\n <div className=\"flex flex-wrap gap-2\">\n {role.isSystem ? <Badge>System</Badge> : null}\n <Badge variant=\"secondary\">\n {role.permissionCount ?? 0} permissions\n </Badge>\n </div>\n ),\n },\n {\n key: 'createdAt',\n header: 'Created',\n cell: (role) => (\n <p className=\"text-sm\">\n {new Date(role.createdAt).toLocaleDateString()}\n </p>\n ),\n },\n {\n key: 'actions',\n header: 'Actions',\n cell: (_role) => (\n <div className=\"flex gap-2\">\n <Button variant=\"outline\" size=\"sm\">\n Edit\n </Button>\n </div>\n ),\n },\n ];\n\n if (error) {\n return (\n <div className=\"p-6 text-center\">\n <p className=\"text-destructive\">Error loading roles</p>\n </div>\n );\n }\n\n return (\n <div className=\"w-full p-6 space-y-4\">\n <div className=\"flex justify-between items-center\">\n <div>\n <h1 className=\"text-3xl font-bold\">Roles</h1>\n <p className=\"text-muted-foreground\">\n Manage roles and assigned permissions\n </p>\n </div>\n <Button>Create Role</Button>\n </div>\n\n <DataTable\n data={(data as { roles: Role[] })?.roles || []}\n columns={columns}\n isLoading={isLoading}\n emptyMessage=\"No roles found\"\n />\n\n {data &&\n 'roles' in data &&\n data.roles &&\n (data.roles as Role[]).length >= limit && (\n <div className=\"flex justify-between items-center\">\n <Button\n variant=\"outline\"\n disabled={page === 1}\n onClick={() => setPage((prev) => prev - 1)}\n >\n Previous\n </Button>\n <span className=\"text-sm text-muted-foreground\">Page {page}</span>\n <Button\n variant=\"outline\"\n onClick={() => setPage((prev) => prev + 1)}\n >\n Next\n </Button>\n </div>\n )}\n </div>\n );\n}\n"],"mappings":"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","names":[]}
|
|
@@ -1,9 +1,6 @@
|
|
|
1
|
-
import {
|
|
2
|
-
useSession
|
|
3
|
-
} from "./chunk-M2K6O5CN.js";
|
|
4
|
-
|
|
5
1
|
// src/components/profile/change-password-form.tsx
|
|
6
2
|
import { zodResolver } from "@hookform/resolvers/zod";
|
|
3
|
+
import { useSession } from "@mesob/auth-react";
|
|
7
4
|
import {
|
|
8
5
|
Button,
|
|
9
6
|
Collapsible,
|
|
@@ -236,4 +233,4 @@ function ChangePasswordForm() {
|
|
|
236
233
|
export {
|
|
237
234
|
ChangePasswordForm
|
|
238
235
|
};
|
|
239
|
-
//# sourceMappingURL=chunk-
|
|
236
|
+
//# sourceMappingURL=chunk-5YAYZDKX.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/profile/change-password-form.tsx"],"sourcesContent":["'use client';\n\nimport { zodResolver } from '@hookform/resolvers/zod';\nimport { useSession } from '@mesob/auth-react';\nimport {\n Button,\n Collapsible,\n CollapsibleContent,\n CollapsibleTrigger,\n Input,\n Label,\n Spinner,\n} from '@mesob/ui/components';\nimport { IconChevronDown, IconEye, IconEyeOff } from '@tabler/icons-react';\nimport { useState } from 'react';\nimport { useForm } from 'react-hook-form';\nimport { toast } from 'sonner';\nimport { z } from 'zod';\n\nconst changePasswordSchema = z\n .object({\n currentPassword: z\n .string()\n .min(8, 'Password must be at least 8 characters'),\n newPassword: z.string().min(8, 'Password must be at least 8 characters'),\n confirmPassword: z\n .string()\n .min(8, 'Password must be at least 8 characters'),\n })\n .refine((data) => data.newPassword === data.confirmPassword, {\n message: \"Passwords don't match\",\n path: ['confirmPassword'],\n });\n\ntype ChangePasswordFormData = z.infer<typeof changePasswordSchema>;\n\ntype AuthErrorLike = {\n code?: string;\n message?: string;\n name?: string;\n};\n\nfunction isAuthError(error: unknown): error is AuthErrorLike {\n return (\n typeof error === 'object' &&\n error !== null &&\n ('code' in error || 'message' in error || 'name' in error)\n );\n}\n\nfunction getErrorCode(error: AuthErrorLike): string | undefined {\n if (error.code) {\n return error.code;\n }\n if (error.message) {\n const upperMessage = error.message.toUpperCase().trim();\n const validCodes = [\n 'INVALID_PASSWORD',\n 'UNAUTHORIZED',\n 'HAS_NO_PASSWORD',\n 'USER_NOT_FOUND',\n 'USER_EXISTS',\n 'VERIFICATION_EXPIRED',\n 'VERIFICATION_MISMATCH',\n 'VERIFICATION_NOT_FOUND',\n 'TOO_MANY_ATTEMPTS',\n 'REQUIRES_VERIFICATION',\n 'ACCESS_DENIED',\n ];\n if (validCodes.includes(upperMessage)) {\n return upperMessage;\n }\n }\n return undefined;\n}\n\nfunction getPasswordChangeErrorMessage(error: unknown): string {\n if (isAuthError(error)) {\n const errorCode = getErrorCode(error);\n switch (errorCode) {\n case 'INVALID_PASSWORD':\n return 'The current password you entered is incorrect. Please try again.';\n case 'UNAUTHORIZED':\n return 'You are not authorized to perform this action. Please sign in again.';\n case 'HAS_NO_PASSWORD':\n return 'Your account does not have a password set. Please use password reset instead.';\n default:\n return error.message || 'Failed to change password. Please try again.';\n }\n }\n if (error instanceof Error) {\n return error.message;\n }\n return 'Failed to change password. Please try again.';\n}\n\nexport function ChangePasswordForm() {\n const { user: _user } = useSession();\n const [isOpen, setIsOpen] = useState(false);\n const [isSubmitting, setIsSubmitting] = useState(false);\n const [showPassword, setShowPassword] = useState(false);\n const [showNewPassword, setShowNewPassword] = useState(false);\n const [showConfirmPassword, setShowConfirmPassword] = useState(false);\n\n const form = useForm<ChangePasswordFormData>({\n resolver: zodResolver(changePasswordSchema),\n defaultValues: {\n currentPassword: '',\n newPassword: '',\n confirmPassword: '',\n },\n });\n\n const { register, handleSubmit, formState, reset } = form;\n\n const onSubmit = (_data: ChangePasswordFormData) => {\n try {\n setIsSubmitting(true);\n toast.error('Password change unavailable');\n setIsOpen(false);\n } catch (error) {\n const errorMessage = getPasswordChangeErrorMessage(error);\n toast.error(errorMessage);\n } finally {\n setIsSubmitting(false);\n }\n };\n\n return (\n <Collapsible open={isOpen} onOpenChange={setIsOpen}>\n <div className=\"border rounded-lg\">\n <CollapsibleTrigger\n render={\n <Button\n variant=\"ghost\"\n className=\"w-full justify-between p-4 h-auto\"\n />\n }\n >\n <div className=\"flex flex-col items-start\">\n <span className=\"font-medium\">Change Password</span>\n <span className=\"text-sm text-muted-foreground\">\n Update your account password\n </span>\n </div>\n <IconChevronDown\n className={`h-4 w-4 transition-transform ${\n isOpen ? 'rotate-180' : ''\n }`}\n />\n </CollapsibleTrigger>\n\n <CollapsibleContent>\n <form\n onSubmit={handleSubmit(onSubmit)}\n className=\"p-4 space-y-4 border-t\"\n >\n <div className=\"space-y-2 w-full md:w-1/2\">\n <Label htmlFor=\"currentPassword\">Old Password</Label>\n <div className=\"relative\">\n <Input\n id=\"currentPassword\"\n type={showPassword ? 'text' : 'password'}\n autoComplete=\"current-password\"\n placeholder=\"Enter your current password\"\n {...register('currentPassword')}\n />\n <button\n type=\"button\"\n onClick={() => setShowPassword(!showPassword)}\n className=\"absolute right-3 top-1/2 -translate-y-1/2 text-muted-foreground hover:text-foreground\"\n >\n {showPassword ? (\n <IconEyeOff className=\"h-4 w-4\" />\n ) : (\n <IconEye className=\"h-4 w-4\" />\n )}\n </button>\n </div>\n {formState.errors.currentPassword && (\n <p className=\"text-sm text-destructive\">\n {formState.errors.currentPassword.message}\n </p>\n )}\n </div>\n\n <div className=\"space-y-2 w-full md:w-1/2\">\n <Label htmlFor=\"newPassword\">New Password</Label>\n <div className=\"relative\">\n <Input\n id=\"newPassword\"\n type={showNewPassword ? 'text' : 'password'}\n placeholder=\"Enter your new password\"\n autoComplete=\"new-password\"\n {...register('newPassword')}\n />\n\n <button\n type=\"button\"\n onClick={() => setShowNewPassword(!showNewPassword)}\n className=\"absolute right-3 top-1/2 -translate-y-1/2 text-muted-foreground hover:text-foreground\"\n >\n {showNewPassword ? (\n <IconEyeOff className=\"h-4 w-4\" />\n ) : (\n <IconEye className=\"h-4 w-4\" />\n )}\n </button>\n </div>\n {formState.errors.newPassword && (\n <p className=\"text-sm text-destructive\">\n {formState.errors.newPassword.message}\n </p>\n )}\n </div>\n\n <div className=\"space-y-2 w-full md:w-1/2\">\n <Label htmlFor=\"confirmPassword\">Confirm New Password</Label>\n <div className=\"relative\">\n <Input\n id=\"confirmPassword\"\n type={showConfirmPassword ? 'text' : 'password'}\n placeholder=\"Confirm your new password\"\n autoComplete=\"new-password\"\n {...register('confirmPassword')}\n />\n\n <button\n type=\"button\"\n onClick={() => setShowConfirmPassword(!showConfirmPassword)}\n className=\"absolute right-3 top-1/2 -translate-y-1/2 text-muted-foreground hover:text-foreground\"\n >\n {showConfirmPassword ? (\n <IconEyeOff className=\"h-4 w-4\" />\n ) : (\n <IconEye className=\"h-4 w-4\" />\n )}\n </button>\n </div>\n {formState.errors.confirmPassword && (\n <p className=\"text-sm text-destructive\">\n {formState.errors.confirmPassword.message}\n </p>\n )}\n </div>\n\n <div className=\"flex justify-end gap-2\">\n <Button\n type=\"button\"\n variant=\"outline\"\n onClick={() => {\n reset();\n setIsOpen(false);\n }}\n disabled={isSubmitting}\n >\n Cancel\n </Button>\n <Button type=\"submit\" disabled={isSubmitting}>\n {isSubmitting && <Spinner className=\"mr-2 h-4 w-4\" />}\n Change Password\n </Button>\n </div>\n </form>\n </CollapsibleContent>\n </div>\n </Collapsible>\n );\n}\n"],"mappings":"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","names":[]}
|