@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
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/pages/iam/users/_components/user-form.tsx","../src/pages/iam/users/_components/user-card.tsx"],"sourcesContent":["'use client';\n\nimport { zodResolver } from '@hookform/resolvers/zod';\nimport {\n AlertDialog,\n AlertDialogAction,\n AlertDialogCancel,\n AlertDialogContent,\n AlertDialogDescription,\n AlertDialogFooter,\n AlertDialogHeader,\n AlertDialogTitle,\n AlertDialogTrigger,\n Button,\n Checkbox,\n EntityDrawer,\n EntityFormActions,\n Form,\n FormControl,\n FormField,\n FormItem,\n FormLabel,\n FormMessage,\n Input,\n Skeleton,\n Stack,\n} from '@mesob/ui/components';\nimport { IconBan, IconUserCheck } from '@tabler/icons-react';\nimport { useQueryClient } from '@tanstack/react-query';\nimport { useEffect } from 'react';\nimport { useForm, useWatch } from 'react-hook-form';\nimport { z } from 'zod';\nimport type { Role } from '../../roles/_components/roles-data';\nimport { str } from '../../roles/_components/roles-data';\nimport { authApi$ } from '../../shared/page-helpers';\n\nconst schema = z.object({\n fullName: z.string().trim().min(1, 'Full name is required'),\n email: z.string().email('Invalid email').nullable().or(z.literal('')),\n phone: z.string().nullable(),\n emailVerified: z.boolean(),\n phoneVerified: z.boolean(),\n roleIds: z.array(z.string().uuid()),\n});\n\ntype FormData = z.infer<typeof schema>;\n\nconst defaults: FormData = {\n fullName: '',\n email: '',\n phone: null,\n emailVerified: false,\n phoneVerified: false,\n roleIds: [],\n};\n\ntype UserFormProps = {\n mode: 'new' | 'edit';\n userId?: string;\n open: boolean;\n onClose: () => void;\n onSuccess?: () => void;\n};\n\nexport function UserForm({\n mode,\n userId,\n open,\n onClose,\n onSuccess,\n}: UserFormProps) {\n const qc = useQueryClient();\n const { data: userData, isLoading: userLoading } = authApi$.useQuery(\n 'get',\n '/users/{id}',\n { params: { path: { id: userId ?? '' } } },\n {\n enabled: mode === 'edit' && !!userId && open,\n refetchOnMount: 'always',\n },\n );\n const { data: rolesData } = authApi$.useQuery(\n 'get',\n '/roles',\n { params: { query: { limit: 100 } } },\n { enabled: open },\n );\n const allRoles = rolesData?.roles ?? [];\n const roles = allRoles\n .filter((r: Role) => (r.code ?? '').toLowerCase() !== 'owner')\n .sort((a: Role, b: Role) =>\n (str(a.name) || a.code)\n .toLowerCase()\n .localeCompare((str(b.name) || b.code).toLowerCase()),\n );\n const user = userData?.user;\n\n const form = useForm<FormData>({\n resolver: zodResolver(schema),\n defaultValues: defaults,\n });\n const { control, formState, reset, setValue } = form;\n const email = useWatch({ control, name: 'email' }) ?? '';\n const phone = useWatch({ control, name: 'phone' }) ?? '';\n const roleIds = useWatch({ control, name: 'roleIds' }) ?? [];\n const hasEmail = (email ?? '').toString().trim().length > 0;\n const hasPhone = (phone ?? '').toString().trim().length > 0;\n\n useEffect(() => {\n if (!hasEmail) {\n setValue('emailVerified', false, { shouldDirty: false });\n }\n if (!hasPhone) {\n setValue('phoneVerified', false, { shouldDirty: false });\n }\n }, [hasEmail, hasPhone, setValue]);\n\n useEffect(() => {\n if (!open) {\n return;\n }\n if (mode === 'edit' && user && !userLoading) {\n reset({\n fullName: user.fullName,\n email: user.email ?? '',\n phone: user.phone ?? null,\n emailVerified: user.emailVerified ?? false,\n phoneVerified: user.phoneVerified ?? false,\n roleIds: user.roles ?? [],\n });\n } else {\n reset(defaults);\n }\n }, [mode, user, open, userLoading, reset]);\n\n const create = authApi$.useMutation('post', '/users', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/users'] });\n },\n });\n const update = authApi$.useMutation('put', '/users/{id}', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/users'] });\n qc.invalidateQueries({ queryKey: ['get', '/users/{id}'] });\n },\n });\n const del = authApi$.useMutation('delete', '/users/{id}', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/users'] });\n },\n });\n const ban = authApi$.useMutation('post', '/users/{id}/ban', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/users'] });\n qc.invalidateQueries({ queryKey: ['get', '/users/{id}'] });\n },\n });\n const bannedUntil = (user as { bannedUntil?: string | null } | undefined)\n ?.bannedUntil;\n const isBanned = !!bannedUntil && new Date(bannedUntil) > new Date();\n\n const onSubmit = form.handleSubmit(async (d) => {\n const base = {\n fullName: d.fullName,\n email: d.email?.trim() || undefined,\n phone: d.phone?.trim() || undefined,\n emailVerified: d.emailVerified,\n phoneVerified: d.phoneVerified,\n };\n if (mode === 'new') {\n await create.mutateAsync({\n body: {\n ...base,\n email: base.email || undefined,\n phone: base.phone || undefined,\n },\n });\n } else if (userId) {\n await update.mutateAsync({\n params: { path: { id: userId } },\n body: {\n ...base,\n email: base.email ?? null,\n phone: base.phone ?? null,\n roleIds: d.roleIds,\n },\n });\n }\n onSuccess?.();\n onClose();\n });\n\n const onDelete = async () => {\n if (!userId) {\n return;\n }\n await del.mutateAsync({ params: { path: { id: userId } } });\n onSuccess?.();\n onClose();\n };\n\n const onBanUnban = async () => {\n if (!userId) {\n return;\n }\n await ban.mutateAsync({\n params: { path: { id: userId } },\n body: {\n bannedUntil: isBanned ? null : '9999-12-31T23:59:59.999Z',\n },\n });\n onSuccess?.();\n };\n\n const toggleRole = (roleId: string, checked: boolean) => {\n const next = checked\n ? [...roleIds, roleId]\n : roleIds.filter((id) => id !== roleId);\n setValue('roleIds', next, { shouldDirty: true });\n };\n\n const isSubmitting = create.isPending || update.isPending;\n const isBanning = ban.isPending;\n\n return (\n <EntityDrawer\n title={mode === 'new' ? 'New user' : 'Edit user'}\n open={open}\n onClose={onClose}\n isDirty={formState.isDirty}\n size=\"lg\"\n form={\n userLoading && 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=\"fullName\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>\n Full name <span className=\"text-destructive\">*</span>\n </FormLabel>\n <FormControl>\n <Input placeholder=\"Full name\" {...field} />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n <FormField\n control={control}\n name=\"email\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>Email</FormLabel>\n <FormControl>\n <Input\n type=\"email\"\n placeholder=\"user@example.com\"\n {...field}\n value={field.value ?? ''}\n />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n <FormField\n control={control}\n name=\"phone\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>Phone</FormLabel>\n <FormControl>\n <Input\n placeholder=\"+251911223344\"\n {...field}\n value={field.value ?? ''}\n />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n <div className=\"space-y-4\">\n <FormLabel>Verification</FormLabel>\n <div className=\"flex flex-wrap gap-6 pt-1\">\n <FormField\n control={control}\n name=\"emailVerified\"\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 disabled={!hasEmail}\n onCheckedChange={(c) =>\n field.onChange(c === true)\n }\n />\n </FormControl>\n <FormLabel className=\"cursor-pointer font-normal\">\n Email verified\n </FormLabel>\n <FormMessage />\n </FormItem>\n )}\n />\n <FormField\n control={control}\n name=\"phoneVerified\"\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 disabled={!hasPhone}\n onCheckedChange={(c) =>\n field.onChange(c === true)\n }\n />\n </FormControl>\n <FormLabel className=\"cursor-pointer font-normal\">\n Phone verified\n </FormLabel>\n <FormMessage />\n </FormItem>\n )}\n />\n </div>\n </div>\n </Stack>\n\n {mode === 'edit' && roles.length > 0 && (\n <div className=\"space-y-4\">\n <FormLabel>Roles</FormLabel>\n <div className=\"flex flex-col gap-3 pt-1\">\n {roles.map((role: Role) => {\n const desc = str(role.description);\n return (\n <div key={role.id} className=\"flex items-start gap-3\">\n <Checkbox\n size=\"lg\"\n id={`role-${role.id}`}\n checked={roleIds.includes(role.id)}\n onCheckedChange={(c) =>\n toggleRole(role.id, c === true)\n }\n className=\"mt-0.5\"\n />\n <label\n htmlFor={`role-${role.id}`}\n className=\"flex min-w-0 flex-1 cursor-pointer flex-col gap-0.5\"\n >\n <span className=\"text-sm font-medium\">\n {str(role.name) || role.code}\n </span>\n {desc ? (\n <span className=\"text-xs text-muted-foreground\">\n {desc}\n </span>\n ) : null}\n </label>\n </div>\n );\n })}\n </div>\n </div>\n )}\n </form>\n </Form>\n )\n }\n actions={\n <div className=\"flex items-center gap-2\">\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={userLoading && mode === 'edit'}\n itemName=\"user\"\n />\n {mode === 'edit' && userId && (\n <AlertDialog>\n <AlertDialogTrigger\n render={\n <Button\n variant=\"outline\"\n disabled={userLoading}\n loading={isBanning}\n className=\"cursor-pointer\"\n leftIcon={\n isBanned ? (\n <IconUserCheck className=\"size-4\" />\n ) : (\n <IconBan className=\"size-4\" />\n )\n }\n />\n }\n >\n {isBanned ? 'Unban' : 'Ban'}\n </AlertDialogTrigger>\n <AlertDialogContent>\n <AlertDialogHeader>\n <AlertDialogTitle>\n {isBanned ? 'Unban user?' : 'Ban user?'}\n </AlertDialogTitle>\n <AlertDialogDescription>\n {isBanned\n ? 'This user will be able to sign in again.'\n : 'This user will not be able to sign in until unbanned.'}\n </AlertDialogDescription>\n </AlertDialogHeader>\n <AlertDialogFooter>\n <AlertDialogCancel>Cancel</AlertDialogCancel>\n <AlertDialogAction\n onClick={onBanUnban}\n variant={isBanned ? undefined : 'destructive'}\n >\n {isBanned ? 'Unban' : 'Ban'}\n </AlertDialogAction>\n </AlertDialogFooter>\n </AlertDialogContent>\n </AlertDialog>\n )}\n </div>\n }\n />\n );\n}\n\nfunction FormSkeleton() {\n return (\n <div className=\"space-y-4\">\n {[1, 2, 3, 4, 5].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 Button,\n Card,\n CardContent,\n CardHeader,\n DropdownMenu,\n DropdownMenuContent,\n DropdownMenuItem,\n DropdownMenuPortal,\n DropdownMenuTrigger,\n Tooltip,\n TooltipContent,\n TooltipTrigger,\n} from '@mesob/ui/components';\nimport { IconCircleCheck, IconDots, IconPencil } from '@tabler/icons-react';\nimport { useState } from 'react';\nimport { Link } from '../../shared/page-helpers';\nimport { UserForm } from './user-form';\nimport type { User } from './users-data';\n\ntype UserCardProps = { user: User };\n\nexport function UserCard({ user }: UserCardProps) {\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/users/${user.id}`}\n className=\"text-left font-semibold hover:text-primary hover:underline\"\n >\n {user.fullName}\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 </CardHeader>\n <CardContent className=\"space-y-2\">\n {user.email && (\n <div className=\"flex items-center gap-2\">\n <span className=\"text-sm\">{user.email}</span>\n {user.emailVerified && (\n <Tooltip>\n <TooltipTrigger>\n <IconCircleCheck className=\"size-4 text-muted-foreground\" />\n </TooltipTrigger>\n <TooltipContent>Email verified</TooltipContent>\n </Tooltip>\n )}\n </div>\n )}\n <p className=\"text-xs text-muted-foreground\">\n Last sign in{' '}\n {user.lastSignInAt\n ? new Date(user.lastSignInAt).toLocaleDateString()\n : 'never'}\n </p>\n </CardContent>\n </Card>\n {editOpen && (\n <UserForm\n mode=\"edit\"\n userId={user.id}\n open={editOpen}\n onClose={() => setEditOpen(false)}\n />\n )}\n </>\n );\n}\n"],"mappings":";;;;;;;;;AAEA,SAAS,mBAAmB;AAC5B;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,SAAS,qBAAqB;AACvC,SAAS,sBAAsB;AAC/B,SAAS,iBAAiB;AAC1B,SAAS,SAAS,gBAAgB;AAClC,SAAS,SAAS;AA0MR,cAUY,YAVZ;AArMV,IAAM,SAAS,EAAE,OAAO;AAAA,EACtB,UAAU,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,GAAG,uBAAuB;AAAA,EAC1D,OAAO,EAAE,OAAO,EAAE,MAAM,eAAe,EAAE,SAAS,EAAE,GAAG,EAAE,QAAQ,EAAE,CAAC;AAAA,EACpE,OAAO,EAAE,OAAO,EAAE,SAAS;AAAA,EAC3B,eAAe,EAAE,QAAQ;AAAA,EACzB,eAAe,EAAE,QAAQ;AAAA,EACzB,SAAS,EAAE,MAAM,EAAE,OAAO,EAAE,KAAK,CAAC;AACpC,CAAC;AAID,IAAM,WAAqB;AAAA,EACzB,UAAU;AAAA,EACV,OAAO;AAAA,EACP,OAAO;AAAA,EACP,eAAe;AAAA,EACf,eAAe;AAAA,EACf,SAAS,CAAC;AACZ;AAUO,SAAS,SAAS;AAAA,EACvB;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AACF,GAAkB;AAChB,QAAM,KAAK,eAAe;AAC1B,QAAM,EAAE,MAAM,UAAU,WAAW,YAAY,IAAI,SAAS;AAAA,IAC1D;AAAA,IACA;AAAA,IACA,EAAE,QAAQ,EAAE,MAAM,EAAE,IAAI,UAAU,GAAG,EAAE,EAAE;AAAA,IACzC;AAAA,MACE,SAAS,SAAS,UAAU,CAAC,CAAC,UAAU;AAAA,MACxC,gBAAgB;AAAA,IAClB;AAAA,EACF;AACA,QAAM,EAAE,MAAM,UAAU,IAAI,SAAS;AAAA,IACnC;AAAA,IACA;AAAA,IACA,EAAE,QAAQ,EAAE,OAAO,EAAE,OAAO,IAAI,EAAE,EAAE;AAAA,IACpC,EAAE,SAAS,KAAK;AAAA,EAClB;AACA,QAAM,WAAW,WAAW,SAAS,CAAC;AACtC,QAAM,QAAQ,SACX,OAAO,CAAC,OAAa,EAAE,QAAQ,IAAI,YAAY,MAAM,OAAO,EAC5D;AAAA,IAAK,CAAC,GAAS,OACb,IAAI,EAAE,IAAI,KAAK,EAAE,MACf,YAAY,EACZ,eAAe,IAAI,EAAE,IAAI,KAAK,EAAE,MAAM,YAAY,CAAC;AAAA,EACxD;AACF,QAAM,OAAO,UAAU;AAEvB,QAAM,OAAO,QAAkB;AAAA,IAC7B,UAAU,YAAY,MAAM;AAAA,IAC5B,eAAe;AAAA,EACjB,CAAC;AACD,QAAM,EAAE,SAAS,WAAW,OAAO,SAAS,IAAI;AAChD,QAAM,QAAQ,SAAS,EAAE,SAAS,MAAM,QAAQ,CAAC,KAAK;AACtD,QAAM,QAAQ,SAAS,EAAE,SAAS,MAAM,QAAQ,CAAC,KAAK;AACtD,QAAM,UAAU,SAAS,EAAE,SAAS,MAAM,UAAU,CAAC,KAAK,CAAC;AAC3D,QAAM,YAAY,SAAS,IAAI,SAAS,EAAE,KAAK,EAAE,SAAS;AAC1D,QAAM,YAAY,SAAS,IAAI,SAAS,EAAE,KAAK,EAAE,SAAS;AAE1D,YAAU,MAAM;AACd,QAAI,CAAC,UAAU;AACb,eAAS,iBAAiB,OAAO,EAAE,aAAa,MAAM,CAAC;AAAA,IACzD;AACA,QAAI,CAAC,UAAU;AACb,eAAS,iBAAiB,OAAO,EAAE,aAAa,MAAM,CAAC;AAAA,IACzD;AAAA,EACF,GAAG,CAAC,UAAU,UAAU,QAAQ,CAAC;AAEjC,YAAU,MAAM;AACd,QAAI,CAAC,MAAM;AACT;AAAA,IACF;AACA,QAAI,SAAS,UAAU,QAAQ,CAAC,aAAa;AAC3C,YAAM;AAAA,QACJ,UAAU,KAAK;AAAA,QACf,OAAO,KAAK,SAAS;AAAA,QACrB,OAAO,KAAK,SAAS;AAAA,QACrB,eAAe,KAAK,iBAAiB;AAAA,QACrC,eAAe,KAAK,iBAAiB;AAAA,QACrC,SAAS,KAAK,SAAS,CAAC;AAAA,MAC1B,CAAC;AAAA,IACH,OAAO;AACL,YAAM,QAAQ;AAAA,IAChB;AAAA,EACF,GAAG,CAAC,MAAM,MAAM,MAAM,aAAa,KAAK,CAAC;AAEzC,QAAM,SAAS,SAAS,YAAY,QAAQ,UAAU;AAAA,IACpD,WAAW,MAAM;AACf,SAAG,kBAAkB,EAAE,UAAU,CAAC,OAAO,QAAQ,EAAE,CAAC;AAAA,IACtD;AAAA,EACF,CAAC;AACD,QAAM,SAAS,SAAS,YAAY,OAAO,eAAe;AAAA,IACxD,WAAW,MAAM;AACf,SAAG,kBAAkB,EAAE,UAAU,CAAC,OAAO,QAAQ,EAAE,CAAC;AACpD,SAAG,kBAAkB,EAAE,UAAU,CAAC,OAAO,aAAa,EAAE,CAAC;AAAA,IAC3D;AAAA,EACF,CAAC;AACD,QAAM,MAAM,SAAS,YAAY,UAAU,eAAe;AAAA,IACxD,WAAW,MAAM;AACf,SAAG,kBAAkB,EAAE,UAAU,CAAC,OAAO,QAAQ,EAAE,CAAC;AAAA,IACtD;AAAA,EACF,CAAC;AACD,QAAM,MAAM,SAAS,YAAY,QAAQ,mBAAmB;AAAA,IAC1D,WAAW,MAAM;AACf,SAAG,kBAAkB,EAAE,UAAU,CAAC,OAAO,QAAQ,EAAE,CAAC;AACpD,SAAG,kBAAkB,EAAE,UAAU,CAAC,OAAO,aAAa,EAAE,CAAC;AAAA,IAC3D;AAAA,EACF,CAAC;AACD,QAAM,cAAe,MACjB;AACJ,QAAM,WAAW,CAAC,CAAC,eAAe,IAAI,KAAK,WAAW,IAAI,oBAAI,KAAK;AAEnE,QAAM,WAAW,KAAK,aAAa,OAAO,MAAM;AAC9C,UAAM,OAAO;AAAA,MACX,UAAU,EAAE;AAAA,MACZ,OAAO,EAAE,OAAO,KAAK,KAAK;AAAA,MAC1B,OAAO,EAAE,OAAO,KAAK,KAAK;AAAA,MAC1B,eAAe,EAAE;AAAA,MACjB,eAAe,EAAE;AAAA,IACnB;AACA,QAAI,SAAS,OAAO;AAClB,YAAM,OAAO,YAAY;AAAA,QACvB,MAAM;AAAA,UACJ,GAAG;AAAA,UACH,OAAO,KAAK,SAAS;AAAA,UACrB,OAAO,KAAK,SAAS;AAAA,QACvB;AAAA,MACF,CAAC;AAAA,IACH,WAAW,QAAQ;AACjB,YAAM,OAAO,YAAY;AAAA,QACvB,QAAQ,EAAE,MAAM,EAAE,IAAI,OAAO,EAAE;AAAA,QAC/B,MAAM;AAAA,UACJ,GAAG;AAAA,UACH,OAAO,KAAK,SAAS;AAAA,UACrB,OAAO,KAAK,SAAS;AAAA,UACrB,SAAS,EAAE;AAAA,QACb;AAAA,MACF,CAAC;AAAA,IACH;AACA,gBAAY;AACZ,YAAQ;AAAA,EACV,CAAC;AAED,QAAM,WAAW,YAAY;AAC3B,QAAI,CAAC,QAAQ;AACX;AAAA,IACF;AACA,UAAM,IAAI,YAAY,EAAE,QAAQ,EAAE,MAAM,EAAE,IAAI,OAAO,EAAE,EAAE,CAAC;AAC1D,gBAAY;AACZ,YAAQ;AAAA,EACV;AAEA,QAAM,aAAa,YAAY;AAC7B,QAAI,CAAC,QAAQ;AACX;AAAA,IACF;AACA,UAAM,IAAI,YAAY;AAAA,MACpB,QAAQ,EAAE,MAAM,EAAE,IAAI,OAAO,EAAE;AAAA,MAC/B,MAAM;AAAA,QACJ,aAAa,WAAW,OAAO;AAAA,MACjC;AAAA,IACF,CAAC;AACD,gBAAY;AAAA,EACd;AAEA,QAAM,aAAa,CAAC,QAAgB,YAAqB;AACvD,UAAM,OAAO,UACT,CAAC,GAAG,SAAS,MAAM,IACnB,QAAQ,OAAO,CAAC,OAAO,OAAO,MAAM;AACxC,aAAS,WAAW,MAAM,EAAE,aAAa,KAAK,CAAC;AAAA,EACjD;AAEA,QAAM,eAAe,OAAO,aAAa,OAAO;AAChD,QAAM,YAAY,IAAI;AAEtB,SACE;AAAA,IAAC;AAAA;AAAA,MACC,OAAO,SAAS,QAAQ,aAAa;AAAA,MACrC;AAAA,MACA;AAAA,MACA,SAAS,UAAU;AAAA,MACnB,MAAK;AAAA,MACL,MACE,eAAe,SAAS,SACtB,oBAAC,gBAAa,IAEd,oBAAC,QAAM,GAAG,MACR,+BAAC,UAAK,UAAoB,WAAU,aAClC;AAAA,6BAAC,SACC;AAAA;AAAA,YAAC;AAAA;AAAA,cACC;AAAA,cACA,MAAK;AAAA,cACL,QAAQ,CAAC,EAAE,MAAM,MACf,qBAAC,YACC;AAAA,qCAAC,aAAU;AAAA;AAAA,kBACC,oBAAC,UAAK,WAAU,oBAAmB,eAAC;AAAA,mBAChD;AAAA,gBACA,oBAAC,eACC,8BAAC,SAAM,aAAY,aAAa,GAAG,OAAO,GAC5C;AAAA,gBACA,oBAAC,eAAY;AAAA,iBACf;AAAA;AAAA,UAEJ;AAAA,UACA;AAAA,YAAC;AAAA;AAAA,cACC;AAAA,cACA,MAAK;AAAA,cACL,QAAQ,CAAC,EAAE,MAAM,MACf,qBAAC,YACC;AAAA,oCAAC,aAAU,mBAAK;AAAA,gBAChB,oBAAC,eACC;AAAA,kBAAC;AAAA;AAAA,oBACC,MAAK;AAAA,oBACL,aAAY;AAAA,oBACX,GAAG;AAAA,oBACJ,OAAO,MAAM,SAAS;AAAA;AAAA,gBACxB,GACF;AAAA,gBACA,oBAAC,eAAY;AAAA,iBACf;AAAA;AAAA,UAEJ;AAAA,UACA;AAAA,YAAC;AAAA;AAAA,cACC;AAAA,cACA,MAAK;AAAA,cACL,QAAQ,CAAC,EAAE,MAAM,MACf,qBAAC,YACC;AAAA,oCAAC,aAAU,mBAAK;AAAA,gBAChB,oBAAC,eACC;AAAA,kBAAC;AAAA;AAAA,oBACC,aAAY;AAAA,oBACX,GAAG;AAAA,oBACJ,OAAO,MAAM,SAAS;AAAA;AAAA,gBACxB,GACF;AAAA,gBACA,oBAAC,eAAY;AAAA,iBACf;AAAA;AAAA,UAEJ;AAAA,UACA,qBAAC,SAAI,WAAU,aACb;AAAA,gCAAC,aAAU,0BAAY;AAAA,YACvB,qBAAC,SAAI,WAAU,6BACb;AAAA;AAAA,gBAAC;AAAA;AAAA,kBACC;AAAA,kBACA,MAAK;AAAA,kBACL,QAAQ,CAAC,EAAE,MAAM,MACf,qBAAC,YAAS,WAAU,8CAClB;AAAA,wCAAC,eACC;AAAA,sBAAC;AAAA;AAAA,wBACC,MAAK;AAAA,wBACL,SAAS,MAAM;AAAA,wBACf,UAAU,CAAC;AAAA,wBACX,iBAAiB,CAAC,MAChB,MAAM,SAAS,MAAM,IAAI;AAAA;AAAA,oBAE7B,GACF;AAAA,oBACA,oBAAC,aAAU,WAAU,8BAA6B,4BAElD;AAAA,oBACA,oBAAC,eAAY;AAAA,qBACf;AAAA;AAAA,cAEJ;AAAA,cACA;AAAA,gBAAC;AAAA;AAAA,kBACC;AAAA,kBACA,MAAK;AAAA,kBACL,QAAQ,CAAC,EAAE,MAAM,MACf,qBAAC,YAAS,WAAU,8CAClB;AAAA,wCAAC,eACC;AAAA,sBAAC;AAAA;AAAA,wBACC,MAAK;AAAA,wBACL,SAAS,MAAM;AAAA,wBACf,UAAU,CAAC;AAAA,wBACX,iBAAiB,CAAC,MAChB,MAAM,SAAS,MAAM,IAAI;AAAA;AAAA,oBAE7B,GACF;AAAA,oBACA,oBAAC,aAAU,WAAU,8BAA6B,4BAElD;AAAA,oBACA,oBAAC,eAAY;AAAA,qBACf;AAAA;AAAA,cAEJ;AAAA,eACF;AAAA,aACF;AAAA,WACF;AAAA,QAEC,SAAS,UAAU,MAAM,SAAS,KACjC,qBAAC,SAAI,WAAU,aACb;AAAA,8BAAC,aAAU,mBAAK;AAAA,UAChB,oBAAC,SAAI,WAAU,4BACZ,gBAAM,IAAI,CAAC,SAAe;AACzB,kBAAM,OAAO,IAAI,KAAK,WAAW;AACjC,mBACE,qBAAC,SAAkB,WAAU,0BAC3B;AAAA;AAAA,gBAAC;AAAA;AAAA,kBACC,MAAK;AAAA,kBACL,IAAI,QAAQ,KAAK,EAAE;AAAA,kBACnB,SAAS,QAAQ,SAAS,KAAK,EAAE;AAAA,kBACjC,iBAAiB,CAAC,MAChB,WAAW,KAAK,IAAI,MAAM,IAAI;AAAA,kBAEhC,WAAU;AAAA;AAAA,cACZ;AAAA,cACA;AAAA,gBAAC;AAAA;AAAA,kBACC,SAAS,QAAQ,KAAK,EAAE;AAAA,kBACxB,WAAU;AAAA,kBAEV;AAAA,wCAAC,UAAK,WAAU,uBACb,cAAI,KAAK,IAAI,KAAK,KAAK,MAC1B;AAAA,oBACC,OACC,oBAAC,UAAK,WAAU,iCACb,gBACH,IACE;AAAA;AAAA;AAAA,cACN;AAAA,iBAtBQ,KAAK,EAuBf;AAAA,UAEJ,CAAC,GACH;AAAA,WACF;AAAA,SAEJ,GACF;AAAA,MAGJ,SACE,qBAAC,SAAI,WAAU,2BACb;AAAA;AAAA,UAAC;AAAA;AAAA,YACC;AAAA,YACA;AAAA,YACA,SAAS,SAAS,QAAQ,MAAM,MAAM,QAAQ,IAAI;AAAA,YAClD,UAAU,SAAS,SAAS,WAAW;AAAA,YACvC;AAAA,YACA,YAAY,IAAI;AAAA,YAChB,UAAU,eAAe,SAAS;AAAA,YAClC,UAAS;AAAA;AAAA,QACX;AAAA,QACC,SAAS,UAAU,UAClB,qBAAC,eACC;AAAA;AAAA,YAAC;AAAA;AAAA,cACC,QACE;AAAA,gBAAC;AAAA;AAAA,kBACC,SAAQ;AAAA,kBACR,UAAU;AAAA,kBACV,SAAS;AAAA,kBACT,WAAU;AAAA,kBACV,UACE,WACE,oBAAC,iBAAc,WAAU,UAAS,IAElC,oBAAC,WAAQ,WAAU,UAAS;AAAA;AAAA,cAGlC;AAAA,cAGD,qBAAW,UAAU;AAAA;AAAA,UACxB;AAAA,UACA,qBAAC,sBACC;AAAA,iCAAC,qBACC;AAAA,kCAAC,oBACE,qBAAW,gBAAgB,aAC9B;AAAA,cACA,oBAAC,0BACE,qBACG,6CACA,yDACN;AAAA,eACF;AAAA,YACA,qBAAC,qBACC;AAAA,kCAAC,qBAAkB,oBAAM;AAAA,cACzB;AAAA,gBAAC;AAAA;AAAA,kBACC,SAAS;AAAA,kBACT,SAAS,WAAW,SAAY;AAAA,kBAE/B,qBAAW,UAAU;AAAA;AAAA,cACxB;AAAA,eACF;AAAA,aACF;AAAA,WACF;AAAA,SAEJ;AAAA;AAAA,EAEJ;AAEJ;AAEA,SAAS,eAAe;AACtB,SACE,oBAAC,SAAI,WAAU,aACZ,WAAC,GAAG,GAAG,GAAG,GAAG,CAAC,EAAE,IAAI,CAAC,MACpB,qBAAC,SAAY,WAAU,aACrB;AAAA,wBAAC,YAAS,WAAU,YAAW;AAAA,IAC/B,oBAAC,YAAS,WAAU,eAAc;AAAA,OAF1B,CAGV,CACD,GACH;AAEJ;;;AClcA;AAAA,EACE,UAAAA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,iBAAiB,UAAU,kBAAkB;AACtD,SAAS,gBAAgB;AAUrB,mBAIQ,OAAAC,MAoBM,QAAAC,aAxBd;AAHG,SAAS,SAAS,EAAE,KAAK,GAAkB;AAChD,QAAM,CAAC,UAAU,WAAW,IAAI,SAAS,KAAK;AAC9C,SACE,gBAAAA,MAAA,YACE;AAAA,oBAAAA,MAAC,QAAK,WAAU,2CACd;AAAA,sBAAAD,KAAC,cAAW,WAAU,QACpB,0BAAAC,MAAC,SAAI,WAAU,oCACb;AAAA,wBAAAD;AAAA,UAAC;AAAA;AAAA,YACC,MAAM,cAAc,KAAK,EAAE;AAAA,YAC3B,WAAU;AAAA,YAET,eAAK;AAAA;AAAA,QACR;AAAA,QACA,gBAAAC,MAAC,gBACC;AAAA,0BAAAD;AAAA,YAAC;AAAA;AAAA,cACC,QACE,gBAAAA;AAAA,gBAACE;AAAA,gBAAA;AAAA,kBACC,SAAQ;AAAA,kBACR,MAAK;AAAA,kBACL,WAAU;AAAA;AAAA,cACZ;AAAA,cAGF,0BAAAF,KAAC,YAAS,WAAU,WAAU;AAAA;AAAA,UAChC;AAAA,UACA,gBAAAA,KAAC,sBACC,0BAAAA,KAAC,uBACC,0BAAAC,MAAC,oBAAiB,SAAS,MAAM,YAAY,IAAI,GAC/C;AAAA,4BAAAD,KAAC,cAAW,WAAU,gBAAe;AAAA,YAAE;AAAA,aAEzC,GACF,GACF;AAAA,WACF;AAAA,SACF,GACF;AAAA,MACA,gBAAAC,MAAC,eAAY,WAAU,aACpB;AAAA,aAAK,SACJ,gBAAAA,MAAC,SAAI,WAAU,2BACb;AAAA,0BAAAD,KAAC,UAAK,WAAU,WAAW,eAAK,OAAM;AAAA,UACrC,KAAK,iBACJ,gBAAAC,MAAC,WACC;AAAA,4BAAAD,KAAC,kBACC,0BAAAA,KAAC,mBAAgB,WAAU,gCAA+B,GAC5D;AAAA,YACA,gBAAAA,KAAC,kBAAe,4BAAc;AAAA,aAChC;AAAA,WAEJ;AAAA,QAEF,gBAAAC,MAAC,OAAE,WAAU,iCAAgC;AAAA;AAAA,UAC9B;AAAA,UACZ,KAAK,eACF,IAAI,KAAK,KAAK,YAAY,EAAE,mBAAmB,IAC/C;AAAA,WACN;AAAA,SACF;AAAA,OACF;AAAA,IACC,YACC,gBAAAD;AAAA,MAAC;AAAA;AAAA,QACC,MAAK;AAAA,QACL,QAAQ,KAAK;AAAA,QACb,MAAM;AAAA,QACN,SAAS,MAAM,YAAY,KAAK;AAAA;AAAA,IAClC;AAAA,KAEJ;AAEJ;","names":["Button","jsx","jsxs","Button"]}
|
|
@@ -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/permissions.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";
|
|
@@ -95,4 +93,4 @@ function Permissions() {
|
|
|
95
93
|
export {
|
|
96
94
|
Permissions
|
|
97
95
|
};
|
|
98
|
-
//# sourceMappingURL=chunk-
|
|
96
|
+
//# sourceMappingURL=chunk-OUBYTCTD.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/iam/permissions.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// Permission type from OpenAPI schema\ntype Permission = {\n id: string;\n description?: unknown;\n activity: string;\n application: string;\n feature: string;\n};\n\nexport function Permissions() {\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', '/permissions', {\n params: {\n query: {\n page: String(page),\n limit: String(limit),\n },\n },\n });\n\n const columns: DataTableColumn<Permission>[] = [\n {\n key: 'activity',\n header: 'Permission',\n cell: (permission) => (\n <div>\n <p className=\"font-medium\">{permission.activity}</p>\n <Badge variant=\"outline\" className=\"mt-1 font-mono text-xs\">\n {permission.id}\n </Badge>\n </div>\n ),\n },\n {\n key: 'application',\n header: 'Application',\n cell: (permission) => (\n <Badge variant=\"secondary\">{permission.application}</Badge>\n ),\n },\n {\n key: 'feature',\n header: 'Feature',\n cell: (permission) => (\n <Badge variant=\"outline\">{permission.feature}</Badge>\n ),\n },\n {\n key: 'description',\n header: 'Description',\n cell: (permission) => (\n <p className=\"text-sm text-muted-foreground max-w-xs truncate\">\n {String(permission.description ?? '—')}\n </p>\n ),\n },\n ];\n\n if (error) {\n return (\n <div className=\"p-6 text-center\">\n <p className=\"text-destructive\">Error loading permissions</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\">Permissions</h1>\n <p className=\"text-muted-foreground\">\n View the permission catalog for this tenant\n </p>\n </div>\n <Button variant=\"outline\">Seed Permissions</Button>\n </div>\n\n <DataTable\n data={(data as { permissions: Permission[] })?.permissions || []}\n columns={columns}\n isLoading={isLoading}\n emptyMessage=\"No permissions found\"\n />\n\n {data &&\n 'permissions' in data &&\n data.permissions &&\n (data.permissions as Permission[]).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;AAgCjB,SACE,KADF;AApBD,SAAS,cAAc;AAC5B,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,gBAAgB;AAAA,IACvE,QAAQ;AAAA,MACN,OAAO;AAAA,QACL,MAAM,OAAO,IAAI;AAAA,QACjB,OAAO,OAAO,KAAK;AAAA,MACrB;AAAA,IACF;AAAA,EACF,CAAC;AAED,QAAM,UAAyC;AAAA,IAC7C;AAAA,MACE,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,MAAM,CAAC,eACL,qBAAC,SACC;AAAA,4BAAC,OAAE,WAAU,eAAe,qBAAW,UAAS;AAAA,QAChD,oBAAC,SAAM,SAAQ,WAAU,WAAU,0BAChC,qBAAW,IACd;AAAA,SACF;AAAA,IAEJ;AAAA,IACA;AAAA,MACE,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,MAAM,CAAC,eACL,oBAAC,SAAM,SAAQ,aAAa,qBAAW,aAAY;AAAA,IAEvD;AAAA,IACA;AAAA,MACE,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,MAAM,CAAC,eACL,oBAAC,SAAM,SAAQ,WAAW,qBAAW,SAAQ;AAAA,IAEjD;AAAA,IACA;AAAA,MACE,KAAK;AAAA,MACL,QAAQ;AAAA,MACR,MAAM,CAAC,eACL,oBAAC,OAAE,WAAU,mDACV,iBAAO,WAAW,eAAe,QAAG,GACvC;AAAA,IAEJ;AAAA,EACF;AAEA,MAAI,OAAO;AACT,WACE,oBAAC,SAAI,WAAU,mBACb,8BAAC,OAAE,WAAU,oBAAmB,uCAAyB,GAC3D;AAAA,EAEJ;AAEA,SACE,qBAAC,SAAI,WAAU,wBACb;AAAA,yBAAC,SAAI,WAAU,qCACb;AAAA,2BAAC,SACC;AAAA,4BAAC,QAAG,WAAU,sBAAqB,yBAAW;AAAA,QAC9C,oBAAC,OAAE,WAAU,yBAAwB,yDAErC;AAAA,SACF;AAAA,MACA,oBAAC,UAAO,SAAQ,WAAU,8BAAgB;AAAA,OAC5C;AAAA,IAEA;AAAA,MAAC;AAAA;AAAA,QACC,MAAO,MAAwC,eAAe,CAAC;AAAA,QAC/D;AAAA,QACA;AAAA,QACA,cAAa;AAAA;AAAA,IACf;AAAA,IAEC,QACC,iBAAiB,QACjB,KAAK,eACJ,KAAK,YAA6B,UAAU,SAC3C,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,8 +1,5 @@
|
|
|
1
|
-
import {
|
|
2
|
-
useSession
|
|
3
|
-
} from "./chunk-M2K6O5CN.js";
|
|
4
|
-
|
|
5
1
|
// src/components/authorization/grant.tsx
|
|
2
|
+
import { useSession } from "@mesob/auth-react";
|
|
6
3
|
import { grant as canGrant } from "@mesob/common";
|
|
7
4
|
import { Fragment, jsx } from "react/jsx-runtime";
|
|
8
5
|
function Grant({
|
|
@@ -25,4 +22,4 @@ function Grant({
|
|
|
25
22
|
export {
|
|
26
23
|
Grant
|
|
27
24
|
};
|
|
28
|
-
//# sourceMappingURL=chunk-
|
|
25
|
+
//# sourceMappingURL=chunk-Q5RYLX6Z.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/authorization/grant.tsx"],"sourcesContent":["'use client';\n\nimport { useSession } from '@mesob/auth-react';\nimport { grant as canGrant } from '@mesob/common';\nimport type { ReactNode } from 'react';\n\ntype GrantProps = {\n permissions: readonly string[];\n userPermissions?: readonly string[] | null;\n fallback?: ReactNode;\n children: ReactNode;\n};\n\nexport function Grant({\n permissions,\n userPermissions,\n fallback = null,\n children,\n}: GrantProps) {\n const { isLoading, user } = useSession();\n\n if (userPermissions === undefined && isLoading) {\n return null;\n }\n\n const resolvedPermissions = userPermissions ?? user?.permissions ?? [];\n\n if (canGrant(permissions, resolvedPermissions)) {\n return <>{children}</>;\n }\n\n return <>{fallback}</>;\n}\n"],"mappings":";AAEA,SAAS,kBAAkB;AAC3B,SAAS,SAAS,gBAAgB;AAyBvB;AAfJ,SAAS,MAAM;AAAA,EACpB;AAAA,EACA;AAAA,EACA,WAAW;AAAA,EACX;AACF,GAAe;AACb,QAAM,EAAE,WAAW,KAAK,IAAI,WAAW;AAEvC,MAAI,oBAAoB,UAAa,WAAW;AAC9C,WAAO;AAAA,EACT;AAEA,QAAM,sBAAsB,mBAAmB,MAAM,eAAe,CAAC;AAErE,MAAI,SAAS,aAAa,mBAAmB,GAAG;AAC9C,WAAO,gCAAG,UAAS;AAAA,EACrB;AAEA,SAAO,gCAAG,oBAAS;AACrB;","names":[]}
|
|
@@ -1,8 +1,5 @@
|
|
|
1
|
-
import {
|
|
2
|
-
useSession
|
|
3
|
-
} from "./chunk-M2K6O5CN.js";
|
|
4
|
-
|
|
5
1
|
// src/components/authorization/deny.tsx
|
|
2
|
+
import { useSession } from "@mesob/auth-react";
|
|
6
3
|
import { deny as canDeny } from "@mesob/common";
|
|
7
4
|
import { Fragment, jsx } from "react/jsx-runtime";
|
|
8
5
|
function Deny({ permissions, userPermissions, children }) {
|
|
@@ -20,4 +17,4 @@ function Deny({ permissions, userPermissions, children }) {
|
|
|
20
17
|
export {
|
|
21
18
|
Deny
|
|
22
19
|
};
|
|
23
|
-
//# sourceMappingURL=chunk-
|
|
20
|
+
//# sourceMappingURL=chunk-RGGHVAAK.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/authorization/deny.tsx"],"sourcesContent":["'use client';\n\nimport { useSession } from '@mesob/auth-react';\nimport { deny as canDeny } from '@mesob/common';\nimport type { ReactNode } from 'react';\n\ntype DenyProps = {\n permissions: readonly string[];\n userPermissions?: readonly string[] | null;\n children: ReactNode;\n};\n\nexport function Deny({ permissions, userPermissions, children }: DenyProps) {\n const { isLoading, user } = useSession();\n\n if (userPermissions === undefined && isLoading) {\n return null;\n }\n\n const resolvedPermissions = userPermissions ?? user?.permissions ?? [];\n\n if (canDeny(permissions, resolvedPermissions)) {\n return <>{children}</>;\n }\n\n return null;\n}\n"],"mappings":";AAEA,SAAS,kBAAkB;AAC3B,SAAS,QAAQ,eAAe;AAmBrB;AAVJ,SAAS,KAAK,EAAE,aAAa,iBAAiB,SAAS,GAAc;AAC1E,QAAM,EAAE,WAAW,KAAK,IAAI,WAAW;AAEvC,MAAI,oBAAoB,UAAa,WAAW;AAC9C,WAAO;AAAA,EACT;AAEA,QAAM,sBAAsB,mBAAmB,MAAM,eAAe,CAAC;AAErE,MAAI,QAAQ,aAAa,mBAAmB,GAAG;AAC7C,WAAO,gCAAG,UAAS;AAAA,EACrB;AAEA,SAAO;AACT;","names":[]}
|
|
@@ -7,19 +7,19 @@ import {
|
|
|
7
7
|
import {
|
|
8
8
|
handleError
|
|
9
9
|
} from "./chunk-W7UHDTBI.js";
|
|
10
|
+
import {
|
|
11
|
+
createNavigate
|
|
12
|
+
} from "./chunk-6YHUCPJ4.js";
|
|
10
13
|
import {
|
|
11
14
|
AuthLayout
|
|
12
15
|
} from "./chunk-DPH2PHK3.js";
|
|
13
16
|
import {
|
|
14
17
|
useTranslator
|
|
15
|
-
} from "./chunk-
|
|
16
|
-
import {
|
|
17
|
-
useApi,
|
|
18
|
-
useConfig
|
|
19
|
-
} from "./chunk-M2K6O5CN.js";
|
|
18
|
+
} from "./chunk-WG5H5PTL.js";
|
|
20
19
|
|
|
21
20
|
// src/components/auth/forgot-password.tsx
|
|
22
21
|
import { zodResolver } from "@hookform/resolvers/zod";
|
|
22
|
+
import { useApi, useConfig } from "@mesob/auth-react";
|
|
23
23
|
import {
|
|
24
24
|
Alert,
|
|
25
25
|
AlertDescription,
|
|
@@ -53,10 +53,9 @@ var ForgotPassword = () => {
|
|
|
53
53
|
const [error, setError] = useState(null);
|
|
54
54
|
const forgotPasswordMutation = hooks.useMutation("post", "/password/forgot");
|
|
55
55
|
const signInLink = config.navigation?.links?.signIn || "/auth/sign-in";
|
|
56
|
-
const onNavigate =
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
}
|
|
56
|
+
const onNavigate = createNavigate({
|
|
57
|
+
onNavigate: config.navigation?.onNavigate,
|
|
58
|
+
mesobNavigate: mesob?.navigate
|
|
60
59
|
});
|
|
61
60
|
const logoImage = config.ui.logoImage;
|
|
62
61
|
const phoneRegex = typeof config.phoneRegex === "string" ? new RegExp(config.phoneRegex) : config.phoneRegex || /^\+251[79]\d{8}$/;
|
|
@@ -171,4 +170,4 @@ var ForgotPassword = () => {
|
|
|
171
170
|
export {
|
|
172
171
|
ForgotPassword
|
|
173
172
|
};
|
|
174
|
-
//# sourceMappingURL=chunk-
|
|
173
|
+
//# sourceMappingURL=chunk-RI647FTV.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/auth/forgot-password.tsx"],"sourcesContent":["'use client';\n\nimport { zodResolver } from '@hookform/resolvers/zod';\nimport { useApi, useConfig } from '@mesob/auth-react';\nimport {\n Alert,\n AlertDescription,\n AlertTitle,\n Button,\n Form,\n FormControl,\n FormField,\n FormItem,\n FormLabel,\n FormMessage,\n Input,\n} from '@mesob/ui/components';\nimport { useMesob } from '@mesob/ui/providers';\nimport { IconAlertCircle } from '@tabler/icons-react';\nimport { useEffect, useState } from 'react';\nimport { useForm } from 'react-hook-form';\nimport { toast } from 'sonner';\nimport { z } from 'zod';\nimport { useTranslator } from '../../hooks/use-translator';\nimport type { AuthErrorContent } from '../../utils/handle-error';\nimport { handleError } from '../../utils/handle-error';\nimport { createNavigate } from '../../utils/navigation';\nimport { normalizeEmail } from '../../utils/normalize-email';\nimport { normalizePhone } from '../../utils/normalize-phone';\nimport { AuthLayout } from './auth-layout';\n\ntype ForgotPasswordFormValues = {\n account: string;\n};\n\nconst forgotPasswordSchema = (t: (key: string) => string) =>\n z.object({\n account: z.string().min(1, t('errors.accountRequired')),\n });\n\nexport const ForgotPassword = () => {\n const { hooks } = useApi();\n const { config } = useConfig();\n const mesob = useMesob();\n const t = useTranslator('Auth.forgotPassword');\n const Link = mesob?.navigation?.Link;\n const [isLoading, setIsLoading] = useState(false);\n const [error, setError] = useState<AuthErrorContent | null>(null);\n\n const forgotPasswordMutation = hooks.useMutation('post', '/password/forgot');\n\n const signInLink = config.navigation?.links?.signIn || '/auth/sign-in';\n const onNavigate = createNavigate({\n onNavigate: config.navigation?.onNavigate,\n mesobNavigate: mesob?.navigate,\n });\n const logoImage = config.ui.logoImage;\n const phoneRegex =\n typeof config.phoneRegex === 'string'\n ? new RegExp(config.phoneRegex)\n : config.phoneRegex || /^\\+251[79]\\d{8}$/;\n\n const form = useForm<ForgotPasswordFormValues>({\n resolver: zodResolver(forgotPasswordSchema(t)),\n defaultValues: {\n account: '',\n },\n });\n\n useEffect(() => {\n if (error) {\n toast.error(error.title || 'Error', {\n description: error.description,\n });\n }\n }, [error]);\n\n const handleSubmit = form.handleSubmit(async (values) => {\n setIsLoading(true);\n setError(null);\n\n try {\n const raw = values.account.trim();\n const asPhone = normalizePhone(raw);\n const identifier = phoneRegex.test(asPhone)\n ? asPhone\n : normalizeEmail(raw);\n const res = await forgotPasswordMutation.mutateAsync({\n body: {\n identifier,\n },\n });\n\n if ('verificationId' in res && res.verificationId) {\n onNavigate(`/auth/reset-password?verificationId=${res.verificationId}`);\n } else {\n onNavigate(\n `/auth/reset-password?identifier=${encodeURIComponent(identifier)}`,\n );\n }\n } catch (err) {\n handleError(err, setError, t);\n } finally {\n setIsLoading(false);\n }\n });\n\n let errorContent: AuthErrorContent | null = null;\n if (error) {\n if (typeof error === 'string') {\n errorContent = { title: 'Error', description: error };\n } else {\n errorContent = error;\n }\n }\n\n return (\n <AuthLayout\n title={config.ui.name}\n description={t('description')}\n logoImage={logoImage}\n footer={\n Link ? (\n <Link href={signInLink} className=\"text-primary hover:underline\">\n {t('footer.backToSignIn')}\n </Link>\n ) : (\n <a\n href={signInLink}\n onClick={(e) => {\n e.preventDefault();\n onNavigate(signInLink);\n }}\n className=\"text-primary hover:underline\"\n >\n {t('footer.backToSignIn')}\n </a>\n )\n }\n >\n <Form {...form}>\n <form\n id=\"forgot-password-form\"\n onSubmit={handleSubmit}\n className=\"space-y-4\"\n >\n <FormField\n control={form.control}\n name=\"account\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>{t('form.accountLabel')}</FormLabel>\n <FormControl>\n <Input {...field} type=\"text\" />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n <Button\n type=\"submit\"\n form=\"forgot-password-form\"\n className=\"w-full\"\n disabled={isLoading || forgotPasswordMutation.isPending}\n loading={isLoading || forgotPasswordMutation.isPending}\n >\n {isLoading || forgotPasswordMutation.isPending\n ? t('form.submitting')\n : t('form.submit')}\n </Button>\n </form>\n </Form>\n {errorContent && (\n <Alert variant=\"destructive\" className=\"mt-4\">\n <IconAlertCircle className=\"h-4 w-4\" />\n <AlertTitle>{errorContent.title}</AlertTitle>\n <AlertDescription>{errorContent.description}</AlertDescription>\n </Alert>\n )}\n </AuthLayout>\n );\n};\n"],"mappings":";;;;;;;;;;;;;;;;;;;;AAEA,SAAS,mBAAmB;AAC5B,SAAS,QAAQ,iBAAiB;AAClC;AAAA,EACE;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,EACA;AAAA,OACK;AACP,SAAS,gBAAgB;AACzB,SAAS,uBAAuB;AAChC,SAAS,WAAW,gBAAgB;AACpC,SAAS,eAAe;AACxB,SAAS,aAAa;AACtB,SAAS,SAAS;AAqGR,cA2BI,YA3BJ;AAxFV,IAAM,uBAAuB,CAAC,MAC5B,EAAE,OAAO;AAAA,EACP,SAAS,EAAE,OAAO,EAAE,IAAI,GAAG,EAAE,wBAAwB,CAAC;AACxD,CAAC;AAEI,IAAM,iBAAiB,MAAM;AAClC,QAAM,EAAE,MAAM,IAAI,OAAO;AACzB,QAAM,EAAE,OAAO,IAAI,UAAU;AAC7B,QAAM,QAAQ,SAAS;AACvB,QAAM,IAAI,cAAc,qBAAqB;AAC7C,QAAM,OAAO,OAAO,YAAY;AAChC,QAAM,CAAC,WAAW,YAAY,IAAI,SAAS,KAAK;AAChD,QAAM,CAAC,OAAO,QAAQ,IAAI,SAAkC,IAAI;AAEhE,QAAM,yBAAyB,MAAM,YAAY,QAAQ,kBAAkB;AAE3E,QAAM,aAAa,OAAO,YAAY,OAAO,UAAU;AACvD,QAAM,aAAa,eAAe;AAAA,IAChC,YAAY,OAAO,YAAY;AAAA,IAC/B,eAAe,OAAO;AAAA,EACxB,CAAC;AACD,QAAM,YAAY,OAAO,GAAG;AAC5B,QAAM,aACJ,OAAO,OAAO,eAAe,WACzB,IAAI,OAAO,OAAO,UAAU,IAC5B,OAAO,cAAc;AAE3B,QAAM,OAAO,QAAkC;AAAA,IAC7C,UAAU,YAAY,qBAAqB,CAAC,CAAC;AAAA,IAC7C,eAAe;AAAA,MACb,SAAS;AAAA,IACX;AAAA,EACF,CAAC;AAED,YAAU,MAAM;AACd,QAAI,OAAO;AACT,YAAM,MAAM,MAAM,SAAS,SAAS;AAAA,QAClC,aAAa,MAAM;AAAA,MACrB,CAAC;AAAA,IACH;AAAA,EACF,GAAG,CAAC,KAAK,CAAC;AAEV,QAAM,eAAe,KAAK,aAAa,OAAO,WAAW;AACvD,iBAAa,IAAI;AACjB,aAAS,IAAI;AAEb,QAAI;AACF,YAAM,MAAM,OAAO,QAAQ,KAAK;AAChC,YAAM,UAAU,eAAe,GAAG;AAClC,YAAM,aAAa,WAAW,KAAK,OAAO,IACtC,UACA,eAAe,GAAG;AACtB,YAAM,MAAM,MAAM,uBAAuB,YAAY;AAAA,QACnD,MAAM;AAAA,UACJ;AAAA,QACF;AAAA,MACF,CAAC;AAED,UAAI,oBAAoB,OAAO,IAAI,gBAAgB;AACjD,mBAAW,uCAAuC,IAAI,cAAc,EAAE;AAAA,MACxE,OAAO;AACL;AAAA,UACE,mCAAmC,mBAAmB,UAAU,CAAC;AAAA,QACnE;AAAA,MACF;AAAA,IACF,SAAS,KAAK;AACZ,kBAAY,KAAK,UAAU,CAAC;AAAA,IAC9B,UAAE;AACA,mBAAa,KAAK;AAAA,IACpB;AAAA,EACF,CAAC;AAED,MAAI,eAAwC;AAC5C,MAAI,OAAO;AACT,QAAI,OAAO,UAAU,UAAU;AAC7B,qBAAe,EAAE,OAAO,SAAS,aAAa,MAAM;AAAA,IACtD,OAAO;AACL,qBAAe;AAAA,IACjB;AAAA,EACF;AAEA,SACE;AAAA,IAAC;AAAA;AAAA,MACC,OAAO,OAAO,GAAG;AAAA,MACjB,aAAa,EAAE,aAAa;AAAA,MAC5B;AAAA,MACA,QACE,OACE,oBAAC,QAAK,MAAM,YAAY,WAAU,gCAC/B,YAAE,qBAAqB,GAC1B,IAEA;AAAA,QAAC;AAAA;AAAA,UACC,MAAM;AAAA,UACN,SAAS,CAAC,MAAM;AACd,cAAE,eAAe;AACjB,uBAAW,UAAU;AAAA,UACvB;AAAA,UACA,WAAU;AAAA,UAET,YAAE,qBAAqB;AAAA;AAAA,MAC1B;AAAA,MAIJ;AAAA,4BAAC,QAAM,GAAG,MACR;AAAA,UAAC;AAAA;AAAA,YACC,IAAG;AAAA,YACH,UAAU;AAAA,YACV,WAAU;AAAA,YAEV;AAAA;AAAA,gBAAC;AAAA;AAAA,kBACC,SAAS,KAAK;AAAA,kBACd,MAAK;AAAA,kBACL,QAAQ,CAAC,EAAE,MAAM,MACf,qBAAC,YACC;AAAA,wCAAC,aAAW,YAAE,mBAAmB,GAAE;AAAA,oBACnC,oBAAC,eACC,8BAAC,SAAO,GAAG,OAAO,MAAK,QAAO,GAChC;AAAA,oBACA,oBAAC,eAAY;AAAA,qBACf;AAAA;AAAA,cAEJ;AAAA,cACA;AAAA,gBAAC;AAAA;AAAA,kBACC,MAAK;AAAA,kBACL,MAAK;AAAA,kBACL,WAAU;AAAA,kBACV,UAAU,aAAa,uBAAuB;AAAA,kBAC9C,SAAS,aAAa,uBAAuB;AAAA,kBAE5C,uBAAa,uBAAuB,YACjC,EAAE,iBAAiB,IACnB,EAAE,aAAa;AAAA;AAAA,cACrB;AAAA;AAAA;AAAA,QACF,GACF;AAAA,QACC,gBACC,qBAAC,SAAM,SAAQ,eAAc,WAAU,QACrC;AAAA,8BAAC,mBAAgB,WAAU,WAAU;AAAA,UACrC,oBAAC,cAAY,uBAAa,OAAM;AAAA,UAChC,oBAAC,oBAAkB,uBAAa,aAAY;AAAA,WAC9C;AAAA;AAAA;AAAA,EAEJ;AAEJ;","names":[]}
|
|
@@ -5,19 +5,16 @@ import {
|
|
|
5
5
|
Link,
|
|
6
6
|
authApi$,
|
|
7
7
|
useRouter
|
|
8
|
-
} from "./chunk-
|
|
8
|
+
} from "./chunk-F5SAYP67.js";
|
|
9
9
|
import {
|
|
10
10
|
defaultEntityQueryOptions
|
|
11
11
|
} from "./chunk-NPW7D2HZ.js";
|
|
12
12
|
import {
|
|
13
13
|
IAMGuard
|
|
14
|
-
} from "./chunk-
|
|
15
|
-
import {
|
|
16
|
-
useApi,
|
|
17
|
-
useConfig
|
|
18
|
-
} from "./chunk-M2K6O5CN.js";
|
|
14
|
+
} from "./chunk-ZHRM4QOO.js";
|
|
19
15
|
|
|
20
16
|
// src/components/iam/roles-page.tsx
|
|
17
|
+
import { useApi, useConfig } from "@mesob/auth-react";
|
|
21
18
|
import {
|
|
22
19
|
Button as Button2,
|
|
23
20
|
EntityDrawerTrigger,
|
|
@@ -589,4 +586,4 @@ export {
|
|
|
589
586
|
RolesList,
|
|
590
587
|
RolesPage
|
|
591
588
|
};
|
|
592
|
-
//# sourceMappingURL=chunk-
|
|
589
|
+
//# sourceMappingURL=chunk-SCSRGIEL.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/iam/roles-page.tsx","../src/pages/iam/roles/_components/role-form.tsx","../src/pages/iam/roles/_components/roles-list.tsx","../src/pages/iam/roles/_components/role-card.tsx"],"sourcesContent":["'use client';\n\nimport { useApi, useConfig } from '@mesob/auth-react';\nimport {\n Button,\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 { IconShield } from '@tabler/icons-react';\nimport { useQueryClient } from '@tanstack/react-query';\nimport { useState } from 'react';\nimport { toast } from 'sonner';\nimport type { paths } from '../../data/openapi';\nimport { defaultEntityQueryOptions } from '../../lib/query-options';\nimport { RoleForm } from '../../pages/iam/roles/_components/role-form';\nimport { RolesList } from '../../pages/iam/roles/_components/roles-list';\nimport { IAMGuard } from './iam-guard';\n\nexport function RolesPage() {\n return (\n <IAMGuard>\n <RolesPageContent />\n </IAMGuard>\n );\n}\n\nfunction RolesPageContent() {\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/roles' },\n { label: 'Roles' },\n ],\n });\n\n const [createOpen, setCreateOpen] = useState(false);\n const { queryConfig, params, setParams } = useEntityParams({\n searchKey: 'search',\n });\n const rolesQuery = queryConfig as {\n params: {\n query: NonNullable<paths['/roles']['get']['parameters']['query']>;\n };\n };\n\n const { data, isPending, isFetching } = hooks.useQuery(\n 'get',\n '/roles',\n rolesQuery,\n defaultEntityQueryOptions,\n );\n\n const seedRoles = hooks.useMutation('post', '/roles/seed', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/roles'] });\n toast.success('Roles seeded');\n },\n onError: () => {\n toast.error('Failed to seed roles');\n },\n });\n\n const isLoading = isPending || isFetching;\n const roles = data?.roles ?? [];\n const { total, pageCount } = useEntityPagination({\n items: roles,\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={<IconShield className=\"h-5 w-5\" />}\n title=\"Roles\"\n actions={\n <div className=\"flex items-center gap-2\">\n <Button\n variant=\"outline\"\n onClick={() => seedRoles.mutate({})}\n loading={seedRoles.isPending}\n >\n Seed Roles\n </Button>\n <EntityDrawerTrigger\n mode=\"new\"\n entity=\"Role\"\n open={createOpen}\n onOpenChange={setCreateOpen}\n >\n {(open, onClose) => (\n <RoleForm mode=\"new\" open={open} onClose={onClose} />\n )}\n </EntityDrawerTrigger>\n </div>\n }\n search={\n <EntitySearch paramKey=\"search\" placeholder=\"Search roles...\" />\n }\n filter={\n <EntityFilter\n options={[\n { label: 'All', value: '' },\n { label: 'By Code', value: 'code' },\n ]}\n placeholder=\"Filter\"\n />\n }\n sort={\n <EntitySort\n options={[\n { label: 'Created', value: 'createdAt' },\n { label: 'Updated', value: 'updatedAt' },\n { label: 'Code', value: 'code' },\n ]}\n />\n }\n view={<EntityViewToggle views={['table', 'card']} />}\n />\n <RolesList\n data={roles}\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 EntityDrawer,\n EntityFormActions,\n Form,\n FormControl,\n FormField,\n FormItem,\n FormLabel,\n FormMessage,\n Input,\n LocaleInputText,\n LocaleInputTextarea,\n Skeleton,\n} from '@mesob/ui/components';\nimport { useLocaleConfig, useLocaleSchemas } from '@mesob/ui/providers';\nimport { useQueryClient } from '@tanstack/react-query';\nimport { useEffect, useMemo } from 'react';\nimport type { FieldValues, Resolver, UseFormRegister } from 'react-hook-form';\nimport { useForm, useWatch } from 'react-hook-form';\nimport { z } from 'zod';\nimport { authApi$ } from '../../shared/page-helpers';\n\ntype RoleFormData = {\n name: Record<string, string>;\n code: string;\n description: Record<string, string>;\n permissionIds: string[];\n};\n\ntype RoleFormProps = {\n mode: 'new' | 'edit';\n roleId?: string;\n open: boolean;\n onClose: () => void;\n onSuccess?: () => void;\n};\n\nexport function RoleForm({\n mode,\n roleId,\n open,\n onClose,\n onSuccess,\n}: RoleFormProps) {\n const qc = useQueryClient();\n const { localeInputDefault, requiredSchema, optionalSchema } =\n useLocaleSchemas();\n const { defaultLanguage } = useLocaleConfig();\n const schema = useMemo(\n () =>\n z.object({\n name: requiredSchema,\n code: z.string().min(1, 'Code is required'),\n description: optionalSchema,\n permissionIds: z.array(z.string()),\n }),\n [requiredSchema, optionalSchema],\n );\n const defaults: RoleFormData = useMemo(\n () => ({\n name: { ...localeInputDefault },\n code: '',\n description: { ...localeInputDefault },\n permissionIds: [],\n }),\n [localeInputDefault],\n );\n\n const { data, isLoading } = authApi$.useQuery(\n 'get',\n '/roles/{id}',\n { params: { path: { id: roleId ?? '' } } },\n { enabled: mode === 'edit' && !!roleId },\n );\n const create = authApi$.useMutation('post', '/roles', {\n onSuccess: () => qc.invalidateQueries({ queryKey: ['get', '/roles'] }),\n });\n const update = authApi$.useMutation('put', '/roles/{id}', {\n onSuccess: () => {\n qc.invalidateQueries({ queryKey: ['get', '/roles'] });\n if (roleId) {\n qc.invalidateQueries({ queryKey: ['get', '/roles/{id}'] });\n }\n },\n });\n const del = authApi$.useMutation('delete', '/roles/{id}', {\n onSuccess: () => qc.invalidateQueries({ queryKey: ['get', '/roles'] }),\n });\n\n const form = useForm<RoleFormData>({\n resolver: zodResolver(schema) as Resolver<RoleFormData>,\n defaultValues: defaults,\n });\n\n const { reset, formState, control, register, setValue } = form;\n /** Locale inputs are typed with `FieldValues`; narrow cast avoids RHF `deps` mismatch. */\n const registerForLocaleFields =\n register as unknown as UseFormRegister<FieldValues>;\n const nameVal = useWatch({ control, name: 'name' });\n const codeVal = useWatch({ control, name: 'code' });\n\n useEffect(() => {\n const nameDefault =\n typeof nameVal === 'object' &&\n nameVal !== null &&\n defaultLanguage in nameVal\n ? ((nameVal as Record<string, string>)[defaultLanguage] ?? '')\n : '';\n if (\n mode === 'new' &&\n typeof nameDefault === 'string' &&\n nameDefault.trim().length > 0 &&\n !codeVal\n ) {\n const code = nameDefault\n .toLowerCase()\n .replace(/\\s+/g, '-')\n .replace(/[^a-z0-9-]/g, '');\n if (code) {\n setValue('code', code);\n }\n }\n }, [mode, nameVal, setValue, codeVal, defaultLanguage]);\n\n useEffect(() => {\n if (!open) {\n return;\n }\n if (mode === 'edit' && data?.role && !isLoading) {\n const r = data.role;\n reset({\n name: (r.name ?? {}) as RoleFormData['name'],\n code: r.code,\n description: (r.description ?? {}) as RoleFormData['description'],\n permissionIds: r.permissionIds ?? [],\n });\n } else {\n reset({ ...defaults, permissionIds: [] });\n }\n }, [mode, data, open, isLoading, reset, defaults]);\n\n const onSubmit = form.handleSubmit(async (d) => {\n const body = {\n name: d.name,\n code: d.code,\n description: d.description ?? undefined,\n permissionIds: d.permissionIds,\n };\n if (mode === 'new') {\n await create.mutateAsync({ body });\n } else if (roleId) {\n await update.mutateAsync({\n params: { path: { id: roleId } },\n body,\n });\n }\n onSuccess?.();\n onClose();\n });\n\n const onDelete = async () => {\n if (!roleId) {\n return;\n }\n await del.mutateAsync({ params: { path: { id: roleId } } });\n onSuccess?.();\n onClose();\n };\n\n const editable = mode === 'new' || data?.role?.isEditable !== false;\n\n return (\n <EntityDrawer\n title={mode === 'new' ? 'New role' : 'Edit role'}\n open={open}\n onClose={onClose}\n isDirty={formState.isDirty}\n size=\"md\"\n form={\n isLoading ? (\n <FormSkeleton />\n ) : (\n <Form {...form}>\n <form onSubmit={onSubmit} className=\"space-y-4\">\n <LocaleInputText\n label=\"Name\"\n field=\"name\"\n required\n register={registerForLocaleFields}\n errors={formState.errors}\n placeholder=\"e.g. Administrator\"\n disabled={!editable}\n />\n <FormField\n control={control}\n name=\"code\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>\n Code <span className=\"text-destructive\">*</span>\n </FormLabel>\n <FormControl>\n <Input\n placeholder=\"e.g. admin\"\n disabled={!editable}\n {...field}\n />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n <LocaleInputTextarea\n label=\"Description\"\n field=\"description\"\n register={registerForLocaleFields}\n errors={formState.errors}\n placeholder=\"Description\"\n rows={3}\n disabled={!editable}\n />\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={create.isPending || update.isPending}\n isDeleting={del.isPending}\n disabled={isLoading || !editable}\n itemName=\"role\"\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-16\" />\n <Skeleton className=\"h-10 w-full\" />\n </div>\n <div className=\"space-y-2\">\n <Skeleton className=\"h-4 w-14\" />\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 { IconCalendar, IconShield } from '@tabler/icons-react';\nimport { useState } from 'react';\nimport { Link } from '../../shared/page-helpers';\nimport { RoleCard } from './role-card';\nimport { RoleForm } from './role-form';\nimport type { Role } from './roles-data';\nimport { str } from './roles-data';\n\nconst TABLE_COLUMN_COUNT = 5;\n\ntype RolesListProps = {\n data: Role[];\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 RolesList({\n data,\n isLoading,\n view,\n pageIndex,\n pageSize,\n pageCount,\n totalRows,\n onPageChange,\n onPageSizeChange,\n onCreateNew,\n}: RolesListProps) {\n const [drawerRole, setDrawerRole] = useState<Role | 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={IconShield}\n entityName=\"role\"\n title=\"No roles yet\"\n description=\"Create your first role to get started.\"\n onAction={onCreateNew}\n />\n );\n }\n if (view === 'table') {\n return (\n <div className=\"space-y-4\">\n {drawerRole ? (\n <RoleForm\n mode=\"edit\"\n roleId={drawerRole.id}\n open\n onClose={() => setDrawerRole(null)}\n onSuccess={() => setDrawerRole(null)}\n />\n ) : null}\n <DisplayTable withTableBorder>\n <Thead>\n <Tr>\n <Th>Role</Th>\n <Th>Description</Th>\n <Th>Access</Th>\n <Th>Created</Th>\n <Th className=\"w-[50px]\" />\n </Tr>\n </Thead>\n <Tbody>\n {data.map((role) => (\n <Tr key={role.id} className=\"group\">\n <Td>\n <Link\n href={`/iam/roles/${role.id}`}\n className=\"block text-left font-medium hover:text-primary hover:underline cursor-pointer\"\n >\n <p>{str(role.name) || role.code}</p>\n </Link>\n </Td>\n <Td>\n <span className=\"text-muted-foreground line-clamp-1 max-w-[200px]\">\n {str(role.description) || '—'}\n </span>\n </Td>\n <Td>\n <div className=\"flex flex-wrap gap-2\">\n {role.isSystem ? <Badge>System</Badge> : null}\n <Badge variant=\"secondary\">\n {role.permissionCount ?? role.permissionIds?.length ?? 0}{' '}\n permissions\n </Badge>\n </div>\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(role.createdAt).toLocaleDateString()}\n </div>\n </Td>\n <Td>\n <DataTableAction onClick={() => setDrawerRole(role)} />\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((r) => (\n <RoleCard key={r.id} role={r} />\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, IconExternalLink } from '@tabler/icons-react';\nimport { useRouter } from '../../shared/page-helpers';\nimport type { Role } from './roles-data';\nimport { str } from './roles-data';\n\ntype RoleCardProps = {\n role: Role;\n mode?: 'navigate' | 'static';\n};\n\nexport function RoleCard({ role, mode = 'navigate' }: RoleCardProps) {\n const router = useRouter();\n const canNavigate = mode === 'navigate';\n\n return (\n <Card className=\"group transition-shadow hover:shadow-md\">\n <CardHeader className=\"pb-2\">\n <div className=\"flex items-start justify-between\">\n <button\n type=\"button\"\n onClick={() => {\n if (canNavigate) {\n router.push(`/iam/roles/${role.id}`);\n }\n }}\n className=\"text-left font-semibold hover:text-primary hover:underline\"\n >\n {str(role.name) || role.code}\n </button>\n {canNavigate ? (\n <DropdownMenu>\n <DropdownMenuTrigger\n render={\n <Button\n variant=\"ghost\"\n size=\"icon\"\n className=\"h-8 w-8 opacity-0 transition-opacity group-hover:opacity-100\"\n />\n }\n >\n <IconDots className=\"h-4 w-4\" />\n </DropdownMenuTrigger>\n <DropdownMenuPortal>\n <DropdownMenuContent>\n <DropdownMenuItem\n onClick={() => router.push(`/iam/roles/${role.id}`)}\n >\n <IconExternalLink className=\"mr-2 h-4 w-4\" />\n Open\n </DropdownMenuItem>\n </DropdownMenuContent>\n </DropdownMenuPortal>\n </DropdownMenu>\n ) : null}\n </div>\n <Badge variant=\"outline\" className=\"text-xs\">\n {role.code}\n </Badge>\n </CardHeader>\n <CardContent>\n <div className=\"mb-3 flex flex-wrap gap-2\">\n {role.isSystem ? <Badge>System</Badge> : null}\n <Badge variant=\"secondary\">\n {role.permissionCount ?? role.permissionIds?.length ?? 0}{' '}\n permissions\n </Badge>\n <Badge variant=\"outline\">{role.userCount ?? 0} users</Badge>\n </div>\n {str(role.description) && (\n <p className=\"line-clamp-2 text-sm text-muted-foreground\">\n {str(role.description)}\n </p>\n )}\n <p className=\"mt-2 text-xs text-muted-foreground\">\n Created {new Date(role.createdAt).toLocaleDateString()}\n </p>\n </CardContent>\n </Card>\n );\n}\n"],"mappings":"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","names":["Button","IconShield","useQueryClient","useState","Badge","jsx","jsxs","jsx","jsxs","Badge","jsx","jsxs","useQueryClient","useState","IconShield","Button"]}
|
|
@@ -0,0 +1,30 @@
|
|
|
1
|
+
// src/lib/translations.ts
|
|
2
|
+
function createTranslator(messages, namespace) {
|
|
3
|
+
return (key, params) => {
|
|
4
|
+
const fullKey = namespace ? `${namespace}.${key}` : key;
|
|
5
|
+
const keys = fullKey.split(".");
|
|
6
|
+
let value = messages;
|
|
7
|
+
for (const k of keys) {
|
|
8
|
+
if (value && typeof value === "object" && value !== null) {
|
|
9
|
+
value = value[k];
|
|
10
|
+
} else {
|
|
11
|
+
return fullKey;
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
if (typeof value !== "string") {
|
|
15
|
+
return fullKey;
|
|
16
|
+
}
|
|
17
|
+
if (params) {
|
|
18
|
+
return value.replace(
|
|
19
|
+
/\{(\w+)\}/g,
|
|
20
|
+
(_, param) => String(params[param] ?? `{${param}}`)
|
|
21
|
+
);
|
|
22
|
+
}
|
|
23
|
+
return value;
|
|
24
|
+
};
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
export {
|
|
28
|
+
createTranslator
|
|
29
|
+
};
|
|
30
|
+
//# sourceMappingURL=chunk-SEBNQYIE.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/lib/translations.ts"],"sourcesContent":["type Messages = Record<string, unknown>;\n\nexport function createTranslator(messages: Messages, namespace?: string) {\n return (key: string, params?: Record<string, string | number>): string => {\n const fullKey = namespace ? `${namespace}.${key}` : key;\n const keys = fullKey.split('.');\n\n let value: unknown = messages;\n for (const k of keys) {\n if (value && typeof value === 'object' && value !== null) {\n value = (value as Record<string, unknown>)[k];\n } else {\n return fullKey;\n }\n }\n\n if (typeof value !== 'string') {\n return fullKey;\n }\n\n // Simple parameter replacement\n if (params) {\n return value.replace(/\\{(\\w+)\\}/g, (_, param) =>\n String(params[param] ?? `{${param}}`),\n );\n }\n\n return value;\n };\n}\n"],"mappings":";AAEO,SAAS,iBAAiB,UAAoB,WAAoB;AACvE,SAAO,CAAC,KAAa,WAAqD;AACxE,UAAM,UAAU,YAAY,GAAG,SAAS,IAAI,GAAG,KAAK;AACpD,UAAM,OAAO,QAAQ,MAAM,GAAG;AAE9B,QAAI,QAAiB;AACrB,eAAW,KAAK,MAAM;AACpB,UAAI,SAAS,OAAO,UAAU,YAAY,UAAU,MAAM;AACxD,gBAAS,MAAkC,CAAC;AAAA,MAC9C,OAAO;AACL,eAAO;AAAA,MACT;AAAA,IACF;AAEA,QAAI,OAAO,UAAU,UAAU;AAC7B,aAAO;AAAA,IACT;AAGA,QAAI,QAAQ;AACV,aAAO,MAAM;AAAA,QAAQ;AAAA,QAAc,CAAC,GAAG,UACrC,OAAO,OAAO,KAAK,KAAK,IAAI,KAAK,GAAG;AAAA,MACtC;AAAA,IACF;AAEA,WAAO;AAAA,EACT;AACF;","names":[]}
|
|
@@ -1,23 +1,20 @@
|
|
|
1
1
|
import {
|
|
2
2
|
UserCard,
|
|
3
3
|
UserForm
|
|
4
|
-
} from "./chunk-
|
|
4
|
+
} from "./chunk-OMAJKLVW.js";
|
|
5
5
|
import {
|
|
6
6
|
Link,
|
|
7
7
|
authApi$
|
|
8
|
-
} from "./chunk-
|
|
8
|
+
} from "./chunk-F5SAYP67.js";
|
|
9
9
|
import {
|
|
10
10
|
defaultEntityQueryOptions
|
|
11
11
|
} from "./chunk-NPW7D2HZ.js";
|
|
12
12
|
import {
|
|
13
13
|
IAMGuard
|
|
14
|
-
} from "./chunk-
|
|
15
|
-
import {
|
|
16
|
-
useApi,
|
|
17
|
-
useConfig
|
|
18
|
-
} from "./chunk-M2K6O5CN.js";
|
|
14
|
+
} from "./chunk-ZHRM4QOO.js";
|
|
19
15
|
|
|
20
16
|
// src/components/iam/users-page.tsx
|
|
17
|
+
import { useApi, useConfig } from "@mesob/auth-react";
|
|
21
18
|
import {
|
|
22
19
|
EntityDrawerTrigger,
|
|
23
20
|
EntityFilter,
|
|
@@ -739,7 +736,7 @@ import {
|
|
|
739
736
|
TooltipTrigger,
|
|
740
737
|
Tr
|
|
741
738
|
} from "@mesob/ui/components";
|
|
742
|
-
import { useLocaleConfig } from "@mesob/ui/providers";
|
|
739
|
+
import { useLocaleConfig, useMesobLocale } from "@mesob/ui/providers";
|
|
743
740
|
import {
|
|
744
741
|
IconCalendar,
|
|
745
742
|
IconCircleCheck,
|
|
@@ -747,7 +744,6 @@ import {
|
|
|
747
744
|
IconUserPlus,
|
|
748
745
|
IconUsers as IconUsers2
|
|
749
746
|
} from "@tabler/icons-react";
|
|
750
|
-
import { useLocale } from "next-intl";
|
|
751
747
|
import { useState as useState3 } from "react";
|
|
752
748
|
import { jsx as jsx4, jsxs as jsxs4 } from "react/jsx-runtime";
|
|
753
749
|
function resolveRoleName(name, locale, defaultLanguage) {
|
|
@@ -912,7 +908,7 @@ function UsersList({
|
|
|
912
908
|
onCreateNew
|
|
913
909
|
}) {
|
|
914
910
|
const [editingUserId, setEditingUserId] = useState3(null);
|
|
915
|
-
const locale =
|
|
911
|
+
const locale = useMesobLocale();
|
|
916
912
|
const { defaultLanguage } = useLocaleConfig();
|
|
917
913
|
if (isLoading) {
|
|
918
914
|
return /* @__PURE__ */ jsx4(
|
|
@@ -1167,4 +1163,4 @@ function UsersPageContent() {
|
|
|
1167
1163
|
export {
|
|
1168
1164
|
UsersPage
|
|
1169
1165
|
};
|
|
1170
|
-
//# sourceMappingURL=chunk-
|
|
1166
|
+
//# sourceMappingURL=chunk-VQYNQ5X7.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../src/components/iam/users-page.tsx","../src/pages/iam/users/_components/bulk-invite-user-form.tsx","../src/pages/iam/users/_components/invite-user-shared.tsx","../src/pages/iam/users/_components/invite-user-form.tsx","../src/pages/iam/users/_components/users-list.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 Select,\n SelectContent,\n SelectItem,\n SelectTrigger,\n useBreadcrumbs,\n useEntityPagination,\n useEntityParams,\n} from '@mesob/ui/components';\nimport { cn } from '@mesob/ui/lib/utils';\nimport { IconFilter, IconUsers } from '@tabler/icons-react';\nimport { parseAsString, useQueryState } from 'nuqs';\nimport { useMemo, useState } from 'react';\nimport type { paths } from '../../data/openapi';\nimport { defaultEntityQueryOptions } from '../../lib/query-options';\nimport { BulkInviteUserForm } from '../../pages/iam/users/_components/bulk-invite-user-form';\nimport { InviteUserForm } from '../../pages/iam/users/_components/invite-user-form';\nimport { UsersList } from '../../pages/iam/users/_components/users-list';\nimport { IAMGuard } from './iam-guard';\n\nfunction capitalize(s: string) {\n return s.charAt(0).toUpperCase() + s.slice(1).toLowerCase();\n}\n\nexport function UsersPage() {\n return (\n <IAMGuard>\n <UsersPageContent />\n </IAMGuard>\n );\n}\n\nfunction UsersPageContent() {\n const { hooks } = useApi();\n const { config } = useConfig();\n const homeHref = config.navigation?.defaultRedirectUrl || '/';\n const defaultUserType = config.defaultUserType ?? 'employee';\n useBreadcrumbs({\n items: [\n { label: 'Home', href: homeHref },\n { label: 'IAM', href: '/iam/users' },\n { label: 'Users' },\n ],\n });\n const [inviteOpen, setInviteOpen] = useState(false);\n const [bulkInviteOpen, setBulkInviteOpen] = useState(false);\n\n const { queryConfig, params, setParams } = useEntityParams({\n searchKey: 'search',\n searchParamName: 'search',\n });\n const [userType, setUserType] = useQueryState(\n 'userType',\n parseAsString.withDefault(defaultUserType),\n );\n\n const usersQuery = useMemo(\n () =>\n ({\n params: {\n query: {\n ...queryConfig.params.query,\n userType: userType || defaultUserType,\n },\n },\n }) as {\n params: {\n query: NonNullable<paths['/users']['get']['parameters']['query']>;\n };\n },\n [queryConfig.params.query, userType, defaultUserType],\n );\n\n const { data, isPending, isFetching } = hooks.useQuery(\n 'get',\n '/users',\n usersQuery,\n defaultEntityQueryOptions,\n );\n\n const isLoading = isPending || isFetching;\n const users = data?.users ?? [];\n const { total, pageCount } = useEntityPagination({\n items: users,\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={<IconUsers className=\"h-5 w-5\" />}\n title=\"Users\"\n actions={\n <div className=\"flex items-center gap-2\">\n <EntityDrawerTrigger\n mode=\"new\"\n entity=\"User\"\n label=\"Invite user\"\n open={inviteOpen}\n onOpenChange={setInviteOpen}\n >\n {(open, onClose) => (\n <InviteUserForm open={open} onClose={onClose} />\n )}\n </EntityDrawerTrigger>\n <EntityDrawerTrigger\n mode=\"new\"\n entity=\"User\"\n label=\"Bulk invite\"\n variant=\"outline\"\n open={bulkInviteOpen}\n onOpenChange={setBulkInviteOpen}\n >\n {(open, onClose) => (\n <BulkInviteUserForm open={open} onClose={onClose} />\n )}\n </EntityDrawerTrigger>\n </div>\n }\n search={\n <EntitySearch paramKey=\"search\" placeholder=\"Search users...\" />\n }\n filter={\n <div className=\"flex flex-none flex-nowrap items-center gap-2\">\n <Select\n value={userType}\n onValueChange={(v) => {\n setUserType(v);\n setParams({ page: 1 });\n }}\n >\n <SelectTrigger className=\"h-9 min-w-[150px] w-[150px]\">\n <IconFilter className=\"size-4 shrink-0\" />\n <span\n className={cn(\n 'line-clamp-1 flex items-center gap-2',\n !userType && 'text-muted-foreground',\n )}\n >\n {userType === 'all'\n ? 'All'\n : capitalize(userType ?? defaultUserType)}\n </span>\n </SelectTrigger>\n <SelectContent>\n <SelectItem value=\"all\">All</SelectItem>\n <SelectItem value={defaultUserType}>\n {capitalize(defaultUserType)}\n </SelectItem>\n </SelectContent>\n </Select>\n <EntityFilter\n options={[\n { label: 'All', value: '' },\n { label: 'Verified', value: 'verified' },\n { label: 'Unverified', value: 'unverified' },\n ]}\n placeholder=\"Verification\"\n />\n </div>\n }\n sort={\n <EntitySort\n defaultSort=\"createdAt\"\n defaultOrder=\"desc\"\n options={[\n { label: 'Name', value: 'fullName' },\n { label: 'Phone', value: 'phone' },\n { label: 'Email', value: 'email' },\n { label: 'Type', value: 'userType' },\n { label: 'Roles', value: 'roleCount' },\n { label: 'Last login', value: 'lastSignInAt' },\n { label: 'Session', value: 'activeSessionCount' },\n { label: 'Created', value: 'createdAt' },\n ]}\n />\n }\n view={<EntityViewToggle views={['table', 'card']} />}\n />\n <UsersList\n data={users}\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={() => setInviteOpen(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 Badge,\n Button,\n Card,\n CardContent,\n CardHeader,\n CardTitle,\n EntityDrawer,\n EntityFormActions,\n Form,\n FormControl,\n FormField,\n FormItem,\n FormLabel,\n FormMessage,\n Input,\n Textarea,\n} from '@mesob/ui/components';\nimport { IconDownload, IconUsers } from '@tabler/icons-react';\nimport { useQueryClient } from '@tanstack/react-query';\nimport type { ChangeEvent } from 'react';\nimport { useEffect, useState } from 'react';\nimport { useForm } from 'react-hook-form';\nimport { toast } from 'sonner';\nimport { z } from 'zod';\nimport { authApi$ } from '../../shared/page-helpers';\nimport type {\n BulkInviteFailure,\n BulkInviteResponse,\n} from './invite-user-shared';\nimport {\n downloadInviteTemplate,\n InviteResultCard,\n inviteTemplateCsv,\n parseBulkInviteCsv,\n} from './invite-user-shared';\n\nconst schema = z.object({\n sourceFile: z.string().optional(),\n csvText: z.string().trim().min(1, 'CSV text is required'),\n});\n\ntype FormData = z.infer<typeof schema>;\n\ntype BulkInviteUserFormProps = {\n open: boolean;\n onClose: () => void;\n};\n\nconst defaults: FormData = {\n sourceFile: '',\n csvText: '',\n};\n\nexport function BulkInviteUserForm({ open, onClose }: BulkInviteUserFormProps) {\n const qc = useQueryClient();\n const [invited, setInvited] = useState<BulkInviteResponse['invited']>([]);\n const [failed, setFailed] = useState<BulkInviteFailure[]>([]);\n\n const form = useForm<FormData>({\n resolver: zodResolver(schema),\n defaultValues: defaults,\n });\n const { control, formState, getValues, reset, setValue } = form;\n\n const inviteBulk = authApi$.useMutation('post', '/users/invite/bulk', {\n onSuccess: async () => {\n await qc.invalidateQueries({ queryKey: ['get', '/users'] });\n },\n });\n\n useEffect(() => {\n if (open) {\n return;\n }\n\n reset(defaults);\n setInvited([]);\n setFailed([]);\n }, [open, reset]);\n\n const handleFileChange = async (\n event: ChangeEvent<HTMLInputElement>,\n onChange: (value: string) => void,\n ) => {\n const file = event.target.files?.[0];\n\n if (!file) {\n return;\n }\n\n try {\n const text = await file.text();\n onChange(file.name);\n setValue('csvText', text, {\n shouldDirty: true,\n shouldValidate: true,\n });\n toast.success(`Loaded ${file.name}`);\n } catch (error) {\n toast.error(\n error instanceof Error ? error.message : 'Failed to read CSV file',\n );\n } finally {\n event.target.value = '';\n }\n };\n\n const onSubmit = form.handleSubmit(async (values) => {\n try {\n const result = await inviteBulk.mutateAsync({\n body: {\n users: parseBulkInviteCsv(values.csvText),\n },\n });\n\n setInvited(result.invited);\n setFailed(result.failed);\n toast.success(\n `Bulk invite finished: ${result.invited.length} invited, ${result.failed.length} failed`,\n );\n } catch (error) {\n toast.error(\n error instanceof Error ? error.message : 'Failed to bulk invite users',\n );\n }\n });\n\n return (\n <EntityDrawer\n title=\"Bulk invite users\"\n open={open}\n onClose={onClose}\n isDirty={formState.isDirty}\n size=\"xl\"\n form={\n <Form {...form}>\n <form onSubmit={onSubmit} className=\"space-y-4\">\n <div className=\"space-y-2\">\n <p className=\"text-sm text-muted-foreground\">\n Download the CSV template, fill it with your users, then upload\n or paste the CSV below.\n </p>\n <Button\n variant=\"outline\"\n onClick={downloadInviteTemplate}\n leftIcon={<IconDownload className=\"size-4\" />}\n >\n Download template\n </Button>\n </div>\n\n <FormField\n control={control}\n name=\"sourceFile\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>Upload CSV</FormLabel>\n <FormControl>\n <Input\n type=\"file\"\n accept=\".csv,text/csv\"\n onChange={(event) =>\n handleFileChange(event, field.onChange)\n }\n />\n </FormControl>\n {field.value ? (\n <p className=\"text-sm text-muted-foreground\">\n Loaded file: {field.value}\n </p>\n ) : null}\n <FormMessage />\n </FormItem>\n )}\n />\n\n <FormField\n control={control}\n name=\"csvText\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>CSV textarea</FormLabel>\n <FormControl>\n <Textarea\n className=\"min-h-[240px] font-mono text-xs\"\n placeholder={inviteTemplateCsv}\n {...field}\n />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n\n {(invited.length > 0 || failed.length > 0) && (\n <>\n <Card className=\"border-border/60\">\n <CardHeader>\n <CardTitle className=\"flex items-center gap-2 text-base\">\n <IconUsers className=\"size-4\" />\n Invited\n <Badge>{invited.length}</Badge>\n </CardTitle>\n </CardHeader>\n <CardContent className=\"space-y-3\">\n {invited.length ? (\n invited.map((result, index) => (\n <InviteResultCard\n key={`${result.user.id}-${index}`}\n result={result}\n index={index}\n />\n ))\n ) : (\n <p className=\"text-sm text-muted-foreground\">\n Successful invites will show here.\n </p>\n )}\n </CardContent>\n </Card>\n\n <Card className=\"border-border/60\">\n <CardHeader>\n <CardTitle className=\"text-base\">\n Failed rows\n <Badge className=\"ml-2\" variant=\"destructive\">\n {failed.length}\n </Badge>\n </CardTitle>\n </CardHeader>\n <CardContent className=\"space-y-3\">\n {failed.length ? (\n failed.map((failure) => (\n <div\n key={`${failure.index}-${failure.identifier}`}\n className=\"rounded-lg border border-destructive/30 bg-destructive/5 p-4\"\n >\n <div className=\"flex items-center justify-between gap-3\">\n <span className=\"font-medium\">\n row {failure.index + 1}\n </span>\n {failure.identifier ? (\n <Badge variant=\"outline\">\n {failure.identifier}\n </Badge>\n ) : null}\n </div>\n <p className=\"mt-2 text-sm text-muted-foreground\">\n {failure.error}\n </p>\n </div>\n ))\n ) : (\n <p className=\"text-sm text-muted-foreground\">\n Failed invites will show here.\n </p>\n )}\n </CardContent>\n </Card>\n </>\n )}\n </form>\n </Form>\n }\n actions={\n <EntityFormActions\n mode=\"new\"\n onSubmit={onSubmit}\n onReset={() => {\n reset(defaults);\n setInvited([]);\n setFailed([]);\n }}\n isSubmitting={inviteBulk.isPending}\n submitLabel=\"Invite\"\n disabled={!getValues('csvText').trim()}\n />\n }\n />\n );\n}\n","'use client';\n\nimport {\n Badge,\n Card,\n CardContent,\n CardDescription,\n CardHeader,\n CardTitle,\n} from '@mesob/ui/components';\nimport type { paths } from '../../../../data/openapi';\n\ntype InviteUserPayload = NonNullable<\n NonNullable<\n paths['/users/invite']['post']['requestBody']\n >['content']['application/json']\n>;\n\ntype InviteResult =\n paths['/users/invite']['post']['responses'][201]['content']['application/json'];\n\ntype BulkInviteResponse =\n paths['/users/invite/bulk']['post']['responses'][200]['content']['application/json'];\n\ntype BulkInviteFailure = BulkInviteResponse['failed'][number];\n\nconst inviteTemplateCsv = `fullName,email,phone,emailVerified,phoneVerified,sendVia,password\nLulit Demo,lulit@example.com,,true,false,email,\nAbel Demo,,+251911223344,false,true,sms|email,TempPass123!\nMarta Demo,marta@example.com,+251922334455,true,true,email|sms,`;\n\nconst defaultInviteUserValues = {\n fullName: '',\n email: '',\n phone: '',\n password: '',\n emailVerified: false,\n phoneVerified: false,\n viaEmail: false,\n viaSms: false,\n};\n\nfunction buildInviteUserPayload(\n values: typeof defaultInviteUserValues,\n): InviteUserPayload {\n return {\n fullName: values.fullName.trim(),\n email: values.email.trim() || undefined,\n phone: values.phone.trim() || undefined,\n password: values.password.trim() || undefined,\n emailVerified: values.emailVerified,\n phoneVerified: values.phoneVerified,\n sendVia: [\n values.viaEmail ? 'email' : null,\n values.viaSms ? 'sms' : null,\n ].filter(Boolean) as InviteUserPayload['sendVia'],\n };\n}\n\nfunction parseCsvLine(line: string) {\n const cells: string[] = [];\n let current = '';\n let quoted = false;\n\n for (let i = 0; i < line.length; i += 1) {\n const char = line[i];\n const next = line[i + 1];\n\n if (char === '\"' && quoted && next === '\"') {\n current += '\"';\n i += 1;\n continue;\n }\n\n if (char === '\"') {\n quoted = !quoted;\n continue;\n }\n\n if (char === ',' && !quoted) {\n cells.push(current.trim());\n current = '';\n continue;\n }\n\n current += char;\n }\n\n cells.push(current.trim());\n return cells;\n}\n\nfunction parseBoolean(value: string | undefined) {\n if (!value) {\n return false;\n }\n\n return ['true', '1', 'yes', 'y'].includes(value.trim().toLowerCase());\n}\n\nfunction parseSendVia(value: string | undefined) {\n if (!value) {\n return [];\n }\n\n return value\n .split('|')\n .map((item) => item.trim().toLowerCase())\n .filter(\n (item): item is NonNullable<InviteUserPayload['sendVia']>[number] =>\n item === 'email' || item === 'sms',\n );\n}\n\nfunction parseBulkInviteCsv(text: string): InviteUserPayload[] {\n const lines = text\n .split(/\\r?\\n/)\n .map((line) => line.trim())\n .filter(Boolean);\n\n if (!lines.length) {\n throw new Error('Paste at least one CSV row.');\n }\n\n const header = parseCsvLine(lines[0]).map((cell) => cell.toLowerCase());\n const hasHeader = header.includes('fullname');\n const rows = hasHeader ? lines.slice(1) : lines;\n\n if (!rows.length) {\n throw new Error('CSV only has a header row.');\n }\n\n const keys = hasHeader\n ? header\n : [\n 'fullName',\n 'email',\n 'phone',\n 'emailVerified',\n 'phoneVerified',\n 'sendVia',\n 'password',\n ];\n\n return rows.map((line, index) => {\n const values = parseCsvLine(line);\n const record = Object.fromEntries(\n keys.map((key, valueIndex) => [key, values[valueIndex] ?? '']),\n );\n const fullName = String(\n record.fullname || record.fullName || record.name || '',\n ).trim();\n const email = String(record.email || '').trim();\n const phone = String(record.phone || '').trim();\n\n if (!fullName) {\n throw new Error(`Row ${index + 1}: fullName is required.`);\n }\n\n if (!(email || phone)) {\n throw new Error(`Row ${index + 1}: email or phone is required.`);\n }\n\n return {\n fullName,\n email: email || undefined,\n phone: phone || undefined,\n password: String(record.password || '').trim() || undefined,\n emailVerified: parseBoolean(\n String(record.emailverified || record.emailVerified || ''),\n ),\n phoneVerified: parseBoolean(\n String(record.phoneverified || record.phoneVerified || ''),\n ),\n sendVia: parseSendVia(String(record.sendvia || record.sendVia || '')),\n };\n });\n}\n\nfunction downloadInviteTemplate() {\n const blob = new Blob([inviteTemplateCsv], {\n type: 'text/csv;charset=utf-8',\n });\n const url = URL.createObjectURL(blob);\n const link = document.createElement('a');\n\n link.href = url;\n link.download = 'users-bulk-invite-template.csv';\n link.click();\n URL.revokeObjectURL(url);\n}\n\nfunction DeliveryBadges({ delivery }: { delivery: InviteResult['delivery'] }) {\n return (\n <div className=\"flex flex-wrap gap-2\">\n {delivery.email ? (\n <Badge\n variant={delivery.email === 'failed' ? 'destructive' : 'outline'}\n >\n email: {delivery.email}\n </Badge>\n ) : null}\n {delivery.sms ? (\n <Badge variant={delivery.sms === 'failed' ? 'destructive' : 'outline'}>\n sms: {delivery.sms}\n </Badge>\n ) : null}\n {delivery.email || delivery.sms ? null : (\n <Badge variant=\"secondary\">no delivery</Badge>\n )}\n </div>\n );\n}\n\nfunction InviteResultCard({\n result,\n index,\n}: {\n result: InviteResult;\n index?: number;\n}) {\n return (\n <Card className=\"border-border/60 bg-background/70\">\n <CardHeader className=\"pb-3\">\n <div className=\"flex items-start justify-between gap-3\">\n <div>\n <CardTitle className=\"text-base\">{result.user.fullName}</CardTitle>\n <CardDescription>\n {result.user.email || result.user.phone || 'No identifier'}\n </CardDescription>\n </div>\n <div className=\"flex gap-2\">\n {typeof index === 'number' ? (\n <Badge variant=\"outline\">row {index + 1}</Badge>\n ) : null}\n <Badge variant={result.hasPassword ? 'secondary' : 'default'}>\n {result.hasPassword ? 'password set' : 'set-password flow'}\n </Badge>\n </div>\n </div>\n </CardHeader>\n <CardContent className=\"space-y-3 text-sm\">\n <div className=\"flex flex-wrap gap-2\">\n <Badge variant={result.user.emailVerified ? 'default' : 'secondary'}>\n email {result.user.emailVerified ? 'verified' : 'pending'}\n </Badge>\n <Badge variant={result.user.phoneVerified ? 'default' : 'secondary'}>\n phone {result.user.phoneVerified ? 'verified' : 'pending'}\n </Badge>\n </div>\n <DeliveryBadges delivery={result.delivery} />\n {result.inviteUrl ? (\n <p className=\"rounded-md border border-dashed border-border/70 bg-muted/40 px-3 py-2 font-mono text-xs break-all\">\n {result.inviteUrl}\n </p>\n ) : null}\n </CardContent>\n </Card>\n );\n}\n\nexport {\n buildInviteUserPayload,\n defaultInviteUserValues,\n downloadInviteTemplate,\n inviteTemplateCsv,\n InviteResultCard,\n parseBulkInviteCsv,\n};\nexport type { BulkInviteFailure, BulkInviteResponse, InviteResult };\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 Stack,\n} from '@mesob/ui/components';\nimport { useQueryClient } from '@tanstack/react-query';\nimport type { ComponentProps } from 'react';\nimport { useEffect, useState } from 'react';\nimport { useForm, useWatch } from 'react-hook-form';\nimport { toast } from 'sonner';\nimport { z } from 'zod';\nimport { authApi$ } from '../../shared/page-helpers';\nimport {\n buildInviteUserPayload,\n defaultInviteUserValues,\n InviteResultCard,\n} from './invite-user-shared';\n\nconst schema = z\n .object({\n fullName: z.string().trim().min(1, 'Full name is required'),\n email: z.string().email('Invalid email').or(z.literal('')),\n phone: z.string(),\n password: z.string(),\n emailVerified: z.boolean(),\n phoneVerified: z.boolean(),\n viaEmail: z.boolean(),\n viaSms: z.boolean(),\n })\n .superRefine((value, ctx) => {\n if (!(value.email.trim() || value.phone.trim())) {\n ctx.addIssue({\n code: z.ZodIssueCode.custom,\n path: ['email'],\n message: 'Email or phone is required',\n });\n }\n });\n\ntype FormData = z.infer<typeof schema>;\n\ntype InviteUserFormProps = {\n open: boolean;\n onClose: () => void;\n};\n\nexport function InviteUserForm({ open, onClose }: InviteUserFormProps) {\n const qc = useQueryClient();\n const [result, setResult] = useState<\n ComponentProps<typeof InviteResultCard>['result'] | null\n >(null);\n\n const form = useForm<FormData>({\n resolver: zodResolver(schema),\n defaultValues: defaultInviteUserValues,\n });\n const { control, formState, reset, setValue } = form;\n const email = useWatch({ control, name: 'email' }) ?? '';\n const phone = useWatch({ control, name: 'phone' }) ?? '';\n const hasEmail = email.trim().length > 0;\n const hasPhone = phone.trim().length > 0;\n\n const inviteUser = authApi$.useMutation('post', '/users/invite', {\n onSuccess: async () => {\n await qc.invalidateQueries({ queryKey: ['get', '/users'] });\n },\n });\n\n useEffect(() => {\n if (!hasEmail) {\n setValue('emailVerified', false, { shouldDirty: false });\n setValue('viaEmail', false, { shouldDirty: false });\n }\n }, [hasEmail, setValue]);\n\n useEffect(() => {\n if (!hasPhone) {\n setValue('phoneVerified', false, { shouldDirty: false });\n setValue('viaSms', false, { shouldDirty: false });\n }\n }, [hasPhone, setValue]);\n\n useEffect(() => {\n if (open) {\n return;\n }\n\n reset(defaultInviteUserValues);\n setResult(null);\n }, [open, reset]);\n\n const onSubmit = form.handleSubmit(async (values) => {\n try {\n const inviteResult = await inviteUser.mutateAsync({\n body: buildInviteUserPayload(values),\n });\n\n setResult(inviteResult);\n reset(defaultInviteUserValues);\n toast.success('User invited');\n } catch (error) {\n toast.error(\n error instanceof Error ? error.message : 'Failed to invite user',\n );\n }\n });\n\n return (\n <EntityDrawer\n title=\"Invite user\"\n open={open}\n onClose={onClose}\n isDirty={formState.isDirty}\n size=\"lg\"\n form={\n <Form {...form}>\n <form onSubmit={onSubmit} className=\"space-y-4\">\n <Stack>\n <FormField\n control={control}\n name=\"fullName\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>\n Full name <span className=\"text-destructive\">*</span>\n </FormLabel>\n <FormControl>\n <Input placeholder=\"Aster Bekele\" {...field} />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n\n <FormField\n control={control}\n name=\"email\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>Email</FormLabel>\n <FormControl>\n <Input\n type=\"email\"\n placeholder=\"aster@example.com\"\n {...field}\n />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n\n <FormField\n control={control}\n name=\"phone\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>Phone</FormLabel>\n <FormControl>\n <Input placeholder=\"+251911223344\" {...field} />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n\n <FormField\n control={control}\n name=\"password\"\n render={({ field }) => (\n <FormItem>\n <FormLabel>Password</FormLabel>\n <FormControl>\n <Input\n type=\"password\"\n placeholder=\"Optional\"\n {...field}\n />\n </FormControl>\n <FormMessage />\n </FormItem>\n )}\n />\n\n <FormField\n control={control}\n name=\"emailVerified\"\n render={({ field }) => (\n <FormItem>\n <div className=\"flex items-start gap-3\">\n <FormControl>\n <Checkbox\n checked={field.value}\n disabled={!hasEmail}\n onCheckedChange={(checked) =>\n field.onChange(checked === true)\n }\n />\n </FormControl>\n <div className=\"grid gap-1\">\n <FormLabel>Email verified</FormLabel>\n <FormMessage />\n </div>\n </div>\n </FormItem>\n )}\n />\n\n <FormField\n control={control}\n name=\"phoneVerified\"\n render={({ field }) => (\n <FormItem>\n <div className=\"flex items-start gap-3\">\n <FormControl>\n <Checkbox\n checked={field.value}\n disabled={!hasPhone}\n onCheckedChange={(checked) =>\n field.onChange(checked === true)\n }\n />\n </FormControl>\n <div className=\"grid gap-1\">\n <FormLabel>Phone verified</FormLabel>\n <FormMessage />\n </div>\n </div>\n </FormItem>\n )}\n />\n\n <FormField\n control={control}\n name=\"viaEmail\"\n render={({ field }) => (\n <FormItem>\n <div className=\"flex items-start gap-3\">\n <FormControl>\n <Checkbox\n checked={field.value}\n disabled={!hasEmail}\n onCheckedChange={(checked) =>\n field.onChange(checked === true)\n }\n />\n </FormControl>\n <div className=\"grid gap-1\">\n <FormLabel>Send email invite</FormLabel>\n <FormMessage />\n </div>\n </div>\n </FormItem>\n )}\n />\n\n <FormField\n control={control}\n name=\"viaSms\"\n render={({ field }) => (\n <FormItem>\n <div className=\"flex items-start gap-3\">\n <FormControl>\n <Checkbox\n checked={field.value}\n disabled={!hasPhone}\n onCheckedChange={(checked) =>\n field.onChange(checked === true)\n }\n />\n </FormControl>\n <div className=\"grid gap-1\">\n <FormLabel>Send SMS invite</FormLabel>\n <FormMessage />\n </div>\n </div>\n </FormItem>\n )}\n />\n\n {result ? <InviteResultCard result={result} /> : null}\n </Stack>\n </form>\n </Form>\n }\n actions={\n <EntityFormActions\n mode=\"new\"\n onSubmit={onSubmit}\n onReset={() => {\n reset(defaultInviteUserValues);\n setResult(null);\n }}\n isSubmitting={inviteUser.isPending}\n submitLabel=\"Invite\"\n />\n }\n />\n );\n}\n","'use client';\n\nimport {\n Badge,\n DataTableAction,\n DataTablePagination,\n DisplayTable,\n EntityEmptyState,\n EntityLoadingState,\n LocaleText,\n Tbody,\n Td,\n Th,\n Thead,\n Tooltip,\n TooltipContent,\n TooltipTrigger,\n Tr,\n} from '@mesob/ui/components';\nimport { useLocaleConfig, useMesobLocale } from '@mesob/ui/providers';\nimport {\n IconCalendar,\n IconCircleCheck,\n IconDeviceDesktop,\n IconUserPlus,\n IconUsers,\n} from '@tabler/icons-react';\nimport { useState } from 'react';\nimport { Link } from '../../shared/page-helpers';\nimport { UserCard } from './user-card';\nimport { UserForm } from './user-form';\nimport type { User } from './users-data';\n\nfunction resolveRoleName(\n name: string | Record<string, string> | undefined,\n locale: string,\n defaultLanguage: string,\n): string {\n if (!name) {\n return '';\n }\n if (typeof name === 'string') {\n return name;\n }\n return name[locale] ?? name[defaultLanguage] ?? Object.values(name)[0] ?? '';\n}\n\nfunction roleInitials(name: string, maxWords = 2): string {\n const words = name.trim().split(/\\s+/).filter(Boolean);\n if (words.length === 0) {\n return '';\n }\n return words\n .slice(0, maxWords)\n .map((w) => w.charAt(0))\n .join('')\n .toUpperCase();\n}\n\nconst TABLE_COLUMN_COUNT = 6;\n\nfunction capitalize(s: string) {\n return s.charAt(0).toUpperCase() + s.slice(1).toLowerCase();\n}\n\nfunction UserTableRow({\n user,\n onEdit,\n locale,\n defaultLanguage,\n}: {\n user: User;\n onEdit: (id: string) => void;\n locale: string;\n defaultLanguage: string;\n}) {\n return (\n <Tr className=\"group\">\n <Td>\n <Link\n href={`/iam/users/${user.id}`}\n className=\"font-medium hover:text-primary hover:underline cursor-pointer\"\n >\n {user.fullName}\n </Link>\n </Td>\n <Td>\n <div className=\"space-y-1\">\n {user.email && (\n <div className=\"flex items-center gap-2\">\n <span className=\"text-sm\">{user.email}</span>\n {user.emailVerified && (\n <Tooltip>\n <TooltipTrigger>\n <IconCircleCheck className=\"size-4 text-muted-foreground\" />\n </TooltipTrigger>\n <TooltipContent>Email verified</TooltipContent>\n </Tooltip>\n )}\n </div>\n )}\n {user.phone && (\n <div className=\"flex items-center gap-2\">\n <span className=\"text-sm\">{user.phone}</span>\n {user.phoneVerified && (\n <Tooltip>\n <TooltipTrigger>\n <IconCircleCheck className=\"size-4 text-muted-foreground\" />\n </TooltipTrigger>\n <TooltipContent>Phone verified</TooltipContent>\n </Tooltip>\n )}\n </div>\n )}\n {!(user.email || user.phone) && (\n <span className=\"text-muted-foreground\">—</span>\n )}\n </div>\n </Td>\n <Td>\n {(user.userType?.length ?? 0) > 0 ? (\n <div className=\"flex flex-wrap items-center gap-1\">\n {(user.userType ?? []).slice(0, 3).map((t) => (\n <Tooltip key={t}>\n <TooltipTrigger>\n <Badge\n variant=\"secondary\"\n className=\"cursor-default px-1.5 py-0 text-xs font-medium\"\n >\n {t.charAt(0).toUpperCase()}\n </Badge>\n </TooltipTrigger>\n <TooltipContent>{capitalize(t)}</TooltipContent>\n </Tooltip>\n ))}\n {(user.userType?.length ?? 0) > 3 && (\n <Tooltip>\n <TooltipTrigger>\n <Badge\n variant=\"secondary\"\n className=\"cursor-default px-1.5 py-0 text-xs font-medium text-muted-foreground\"\n >\n +{(user.userType?.length ?? 0) - 3}\n </Badge>\n </TooltipTrigger>\n <TooltipContent>\n <div className=\"flex flex-col gap-0.5\">\n {(user.userType ?? []).slice(3).map((t) => (\n <span key={t}>{capitalize(t)}</span>\n ))}\n </div>\n </TooltipContent>\n </Tooltip>\n )}\n </div>\n ) : (\n <span className=\"text-muted-foreground\">—</span>\n )}\n </Td>\n <Td>\n {(user.userRoles?.length ?? 0) > 0 ? (\n <div className=\"flex flex-wrap items-center gap-1\">\n {(user.userRoles ?? []).slice(0, 2).map((r) => {\n const nameRecord =\n typeof r.name === 'object' && r.name !== null\n ? r.name\n : { en: typeof r.name === 'string' ? r.name : r.code };\n const label = resolveRoleName(r.name, locale, defaultLanguage);\n return (\n <Tooltip key={r.code}>\n <TooltipTrigger>\n <Badge\n variant=\"outline\"\n className=\"cursor-default px-1.5 py-0 text-xs font-medium\"\n >\n {roleInitials(label || r.code) ||\n r.code.slice(0, 2).toUpperCase()}\n </Badge>\n </TooltipTrigger>\n <TooltipContent>\n <LocaleText text={nameRecord} />\n </TooltipContent>\n </Tooltip>\n );\n })}\n {(user.userRoles?.length ?? 0) > 2 && (\n <Tooltip>\n <TooltipTrigger>\n <Badge\n variant=\"outline\"\n className=\"cursor-default px-1.5 py-0 text-xs font-medium text-muted-foreground\"\n >\n +{(user.userRoles?.length ?? 0) - 2}\n </Badge>\n </TooltipTrigger>\n <TooltipContent>\n <div className=\"flex flex-col gap-0.5\">\n {(user.userRoles ?? []).slice(2).map((r) => {\n const label = resolveRoleName(\n r.name,\n locale,\n defaultLanguage,\n );\n return <span key={r.code}>{label || r.code}</span>;\n })}\n </div>\n </TooltipContent>\n </Tooltip>\n )}\n </div>\n ) : (\n <span className=\"text-muted-foreground\">—</span>\n )}\n </Td>\n <Td>\n <div className=\"flex flex-wrap items-center gap-3 text-sm text-muted-foreground\">\n <Tooltip>\n <TooltipTrigger>\n <span className=\"flex min-w-[6.5rem] cursor-default items-center gap-1\">\n <IconCalendar className=\"h-4 w-4 shrink-0\" />\n {user.lastSignInAt\n ? new Date(user.lastSignInAt).toLocaleDateString()\n : 'Never'}\n </span>\n </TooltipTrigger>\n <TooltipContent>\n Last login\n {user.lastSignInAt\n ? `: ${new Date(user.lastSignInAt).toLocaleString()}`\n : ''}\n </TooltipContent>\n </Tooltip>\n {user.createdAt ? (\n <Tooltip>\n <TooltipTrigger>\n <span className=\"flex min-w-[6.5rem] cursor-default items-center gap-1\">\n <IconUserPlus className=\"h-4 w-4 shrink-0\" />\n {new Date(user.createdAt).toLocaleDateString()}\n </span>\n </TooltipTrigger>\n <TooltipContent>\n Registration date: {new Date(user.createdAt).toLocaleString()}\n </TooltipContent>\n </Tooltip>\n ) : (\n <span className=\"min-w-[6.5rem]\" />\n )}\n <Tooltip>\n <TooltipTrigger>\n <span className=\"flex min-w-[6.5rem] cursor-default items-center gap-1\">\n <IconDeviceDesktop className=\"h-4 w-4 shrink-0\" />\n {user.activeSessionCount ?? 0}\n </span>\n </TooltipTrigger>\n <TooltipContent>\n {user.activeSessionCount === 1\n ? '1 active session'\n : `${user.activeSessionCount ?? 0} active sessions`}\n </TooltipContent>\n </Tooltip>\n </div>\n </Td>\n <Td>\n <DataTableAction onClick={() => onEdit(user.id)} />\n </Td>\n </Tr>\n );\n}\n\ntype UsersListProps = {\n data: User[];\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 UsersList({\n data,\n isLoading,\n view,\n pageIndex,\n pageSize,\n pageCount,\n totalRows,\n onPageChange,\n onPageSizeChange,\n onCreateNew,\n}: UsersListProps) {\n const [editingUserId, setEditingUserId] = useState<string | null>(null);\n const locale = useMesobLocale();\n const { defaultLanguage } = useLocaleConfig();\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={IconUsers}\n entityName=\"user\"\n title=\"No users yet\"\n description=\"Invite your first user to get started.\"\n actionLabel=\"Invite user\"\n onAction={onCreateNew}\n />\n );\n }\n if (view === 'table') {\n return (\n <div className=\"space-y-4\">\n {editingUserId && (\n <UserForm\n mode=\"edit\"\n userId={editingUserId}\n open\n onClose={() => setEditingUserId(null)}\n />\n )}\n <DisplayTable withTableBorder>\n <Thead>\n <Tr>\n <Th>Name</Th>\n <Th>Contact</Th>\n <Th>Type</Th>\n <Th>Roles</Th>\n <Th>Activity</Th>\n <Th className=\"w-[50px]\" />\n </Tr>\n </Thead>\n <Tbody>\n {data.map((user) => (\n <UserTableRow\n key={user.id}\n user={user}\n onEdit={setEditingUserId}\n locale={locale}\n defaultLanguage={defaultLanguage}\n />\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((user) => (\n <UserCard key={user.id} user={user} />\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"],"mappings":"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","names":["IconUsers","useState","Badge","Card","CardContent","CardHeader","CardTitle","jsx","jsxs","Card","CardHeader","CardTitle","Badge","CardContent","zodResolver","EntityDrawer","EntityFormActions","Form","FormControl","FormField","FormItem","FormLabel","FormMessage","Input","useQueryClient","useEffect","useState","useForm","toast","z","jsx","jsxs","schema","z","useQueryClient","useState","useForm","zodResolver","useEffect","toast","EntityDrawer","Form","FormField","FormItem","FormLabel","FormControl","Input","FormMessage","EntityFormActions","Badge","IconUsers","useState","jsx","jsxs","Badge","useState","IconUsers","jsx","jsxs","capitalize","useState","IconUsers"]}
|