@firecms/user_management 3.0.0-canary.14 → 3.0.0-canary.140
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/LICENSE +114 -21
- package/README.md +2 -2
- package/dist/hooks/index.d.ts +1 -1
- package/dist/hooks/{useBuildFirestoreUserManagement.d.ts → useBuildUserManagement.d.ts} +14 -7
- package/dist/index.es.js +1167 -818
- package/dist/index.es.js.map +1 -1
- package/dist/index.umd.js +1635 -1
- package/dist/index.umd.js.map +1 -1
- package/dist/types/user_management.d.ts +14 -3
- package/dist/useUserManagementPlugin.d.ts +6 -1
- package/package.json +12 -28
- package/src/components/roles/RolesDetailsForm.tsx +69 -24
- package/src/components/roles/RolesTable.tsx +6 -4
- package/src/components/roles/RolesView.tsx +3 -1
- package/src/components/users/UserDetailsForm.tsx +11 -9
- package/src/components/users/UsersTable.tsx +6 -4
- package/src/hooks/index.ts +1 -1
- package/src/hooks/useBuildUserManagement.tsx +314 -0
- package/src/types/user_management.tsx +17 -3
- package/src/useUserManagementPlugin.tsx +88 -2
- package/src/utils/permissions.ts +7 -6
- package/src/hooks/useBuildFirestoreUserManagement.tsx +0 -240
package/dist/index.umd.js.map
CHANGED
@@ -1 +1 @@
|
|
1
|
-
{"version":3,"file":"index.umd.js","sources":["../src/utils/permissions.ts","../src/utils/local_storage.ts","../src/utils/colors.ts","../src/hooks/useBuildFirestoreUserManagement.tsx","../src/UserManagementProvider.tsx","../src/hooks/useUserManagement.tsx","../src/components/roles/RoleChip.tsx","../src/components/roles/RolesDetailsForm.tsx","../src/components/roles/default_roles.tsx","../src/components/roles/RolesTable.tsx","../src/components/roles/RolesView.tsx","../src/components/users/UserDetailsForm.tsx","../src/components/users/UsersTable.tsx","../src/components/users/UsersView.tsx","../src/useUserManagementPlugin.tsx","../src/admin_views.tsx"],"sourcesContent":["import { EntityCollection, Permissions, Role, User } from \"@firecms/core\";\n\nexport const RESERVED_GROUPS = [\"Admin\"];\n\nconst DEFAULT_PERMISSIONS = {\n read: false,\n edit: false,\n create: false,\n delete: false\n};\n\nexport function resolveUserRolePermissions<UserType extends User>\n({\n collection,\n user\n }: {\n collection: EntityCollection<any>,\n user: UserType | null\n}): Permissions {\n\n const roles = user?.roles;\n if (!roles) {\n return DEFAULT_PERMISSIONS;\n } else if (collection.ownerId === user?.uid) {\n return {\n read: true,\n create: true,\n edit: true,\n delete: true\n };\n } else {\n const basePermissions = {\n read: false,\n create: false,\n edit: false,\n delete: false\n };\n\n return roles\n .map(role => resolveCollectionRole(role, collection.id))\n .reduce(mergePermissions, basePermissions);\n }\n}\n\nfunction resolveCollectionRole(role: Role, id: string): Permissions {\n\n const basePermissions = {\n read: role.isAdmin || role.defaultPermissions?.read,\n create: role.isAdmin || role.defaultPermissions?.create,\n edit: role.isAdmin || role.defaultPermissions?.edit,\n delete: role.isAdmin || role.defaultPermissions?.delete\n };\n if (role.collectionPermissions && role.collectionPermissions[id]) {\n return mergePermissions(role.collectionPermissions[id], basePermissions);\n } else if (role.defaultPermissions) {\n return mergePermissions(role.defaultPermissions, basePermissions);\n } else {\n return basePermissions;\n }\n}\n\nconst mergePermissions = (permA: Permissions, permB: Permissions) => {\n return {\n read: permA.read || permB.read,\n create: permA.create || permB.create,\n edit: permA.edit || permB.edit,\n delete: permA.delete || permB.delete\n };\n}\n\nexport function getUserRoles(roles: Role[], fireCMSUser: User): Role[] | undefined {\n return !roles\n ? undefined\n : (fireCMSUser.roles\n ? fireCMSUser.roles\n .map(role => roles.find((r) => r.id === role.id))\n .filter(Boolean) as Role[]\n : []);\n}\n\nexport const areRolesEqual = (rolesA: Role[], rolesB: Role[]) => {\n const rolesAIds = rolesA.map(r => r.id);\n const rolesBIds = rolesB.map(r => r.id);\n return rolesAIds.length === rolesB.length && rolesAIds.every((role) => rolesBIds.includes(role));\n}\n","// const tokens = new Map<string, {\n// token: string,\n// expiry: Date\n// }>();\n\nexport function cacheDelegatedLoginToken(projectId: string, delegatedToken?: string) {\n if (!delegatedToken) {\n return;\n }\n\n const data = parseJwt(delegatedToken);\n // @ts-ignore\n const expiry = new Date(data.exp * 1000);\n localStorage.setItem(`auth_token::${projectId}`, JSON.stringify({\n token: delegatedToken,\n expiry\n }));\n\n}\n\nexport function getDelegatedLoginTokenFromCache(projectId: string) {\n const entry = localStorage.getItem(`auth_token::${projectId}`);\n if (entry) {\n const data = JSON.parse(entry);\n data.expiry = new Date(data.expiry);\n if (data.expiry > new Date()) {\n return data.token;\n }\n }\n return undefined;\n}\n\nexport function clearDelegatedLoginTokensCache() {\n for (let i = 0; i < localStorage.length; i++) {\n const key = localStorage.key(i);\n if (key?.startsWith(\"auth_token::\")) {\n localStorage.removeItem(key);\n }\n }\n}\n\nfunction parseJwt(token?: string): object {\n if (!token) {\n throw new Error(\"No JWT token\");\n }\n const base64Url = token.split(\".\")[1];\n const base64 = base64Url.replace(/-/g, \"+\").replace(/_/g, \"/\");\n const jsonPayload = decodeURIComponent(window.atob(base64).split(\"\").map(function (c) {\n return \"%\" + (\"00\" + c.charCodeAt(0).toString(16)).slice(-2);\n }).join(\"\"));\n\n return JSON.parse(jsonPayload);\n}\n","export function darkenColor(hexColor: string, darkenBy = 10): string {\n // Check input validity\n if (!/^#([0-9A-Fa-f]{3}){1,2}$/.test(hexColor)) {\n throw new Error(\"Invalid color format\");\n }\n\n // If shorthand form, convert to full form\n let color = hexColor.substring(1).split(\"\");\n if (color.length === 3) {\n color = [color[0], color[0], color[1], color[1], color[2], color[2]];\n }\n\n // Convert to RGB values\n let r = parseInt(color[0] + color[1], 16);\n let g = parseInt(color[2] + color[3], 16);\n let b = parseInt(color[4] + color[5], 16);\n\n // Reduce each color component by the specified percentage (darkenBy)\n r = Math.floor(r * (1 - darkenBy / 100));\n g = Math.floor(g * (1 - darkenBy / 100));\n b = Math.floor(b * (1 - darkenBy / 100));\n\n // Recombine into hex and return\n return \"#\" +\n (r < 16 ? \"0\" : \"\") + r.toString(16) +\n (g < 16 ? \"0\" : \"\") + g.toString(16) +\n (b < 16 ? \"0\" : \"\") + b.toString(16);\n}\n\nexport function hexToRgbaWithOpacity(hexColor: string, opacity = 10): string {\n // Check input validity\n if (!/^#([0-9A-Fa-f]{3}){1,2}$/.test(hexColor)) {\n throw new Error(\"Invalid color format\");\n }\n\n // If shorthand form, convert to full form\n let color = hexColor.substring(1).split(\"\");\n if (color.length === 3) {\n color = [color[0], color[0], color[1], color[1], color[2], color[2]];\n }\n\n // Convert to RGB values\n const r = parseInt(color[0] + color[1], 16);\n const g = parseInt(color[2] + color[3], 16);\n const b = parseInt(color[4] + color[5], 16);\n\n // Convert opacity to a decimal for CSS\n const alpha = opacity / 100;\n\n // Construct and return the RGBA color\n return `rgba(${r}, ${g}, ${b}, ${alpha})`;\n}\n","import React, { useCallback, useEffect, useRef } from \"react\";\nimport {\n collection,\n deleteDoc,\n doc,\n DocumentSnapshot,\n Firestore,\n getFirestore,\n onSnapshot,\n setDoc\n} from \"firebase/firestore\";\nimport { FirebaseApp } from \"firebase/app\";\nimport { UserManagement } from \"../types\";\nimport { PermissionsBuilder, Role, User } from \"@firecms/core\";\nimport { resolveUserRolePermissions } from \"../utils\";\n\ntype UserWithRoleIds = User & { roles: string[] };\n\nexport interface UserManagementParams {\n /**\n * The Firebase app to use for the user management. The config will be saved in the Firestore\n * collection indicated by `configPath`.\n */\n firebaseApp?: FirebaseApp;\n /**\n * Path where the plugin users configuration is stored.\n * Default: __FIRECMS/config/users\n * You can specify a different path if you want to store the user management configuration in a different place.\n * Please keep in mind that the FireCMS users are not necessarily the same as the Firebase users (but they can be).\n * The path should be relative to the root of the Firestore database, and should always have an odd number of segments.\n */\n usersPath?: string;\n\n /**\n * Path where the plugin roles configuration is stored.\n * Default: __FIRECMS/config/roles\n */\n rolesPath?: string;\n\n usersLimit?: number;\n\n canEditRoles?: boolean;\n\n /**\n * If there are no roles in the database, provide a button to create the default roles.\n */\n allowDefaultRolesCreation?: boolean;\n\n /**\n * Include the collection config permissions in the user management system.\n */\n includeCollectionConfigPermissions?: boolean;\n\n}\n\n/**\n * This hook is used to build a user management object that can be used to\n * manage users and roles in a Firestore backend.\n * @param backendFirebaseApp\n * @param usersPath\n * @param rolesPath\n * @param usersLimit\n * @param canEditRoles\n */\nexport function useBuildFirestoreUserManagement({\n firebaseApp,\n usersPath = \"__FIRECMS/config/users\",\n rolesPath = \"__FIRECMS/config/roles\",\n usersLimit,\n canEditRoles = true,\n allowDefaultRolesCreation,\n includeCollectionConfigPermissions\n }: UserManagementParams): UserManagement {\n\n const firestoreRef = useRef<Firestore>();\n\n const [rolesLoading, setRolesLoading] = React.useState<boolean>(true);\n const [usersLoading, setUsersLoading] = React.useState<boolean>(true);\n const [roles, setRoles] = React.useState<Role[]>([]);\n const [usersWithRoleIds, setUsersWithRoleIds] = React.useState<UserWithRoleIds[]>([]);\n\n const users = usersWithRoleIds.map(u => ({\n ...u,\n roles: roles.filter(r => u.roles?.includes(r.id))\n }) as User);\n\n const [rolesError, setRolesError] = React.useState<Error | undefined>();\n const [usersError, setUsersError] = React.useState<Error | undefined>();\n\n const loading = rolesLoading || usersLoading;\n\n useEffect(() => {\n if (!firebaseApp) return;\n firestoreRef.current = getFirestore(firebaseApp);\n }, [firebaseApp]);\n\n useEffect(() => {\n if (!firebaseApp || !rolesPath) return;\n const firestore = getFirestore(firebaseApp);\n\n return onSnapshot(collection(firestore, rolesPath),\n {\n next: (snapshot) => {\n setRolesError(undefined);\n try {\n const newRoles = docsToRoles(snapshot.docs);\n setRoles(newRoles);\n } catch (e) {\n // console.error(e);\n setRolesError(e as Error);\n }\n setRolesLoading(false);\n },\n error: (e) => {\n setRolesError(e);\n setRolesLoading(false);\n }\n }\n );\n }, [firebaseApp, rolesPath]);\n\n useEffect(() => {\n if (!firebaseApp || !usersPath) return;\n const firestore = getFirestore(firebaseApp);\n\n return onSnapshot(collection(firestore, usersPath),\n {\n next: (snapshot) => {\n setUsersError(undefined);\n try {\n const newUsers = docsToUsers(snapshot.docs);\n setUsersWithRoleIds(newUsers);\n } catch (e) {\n setUsersError(e as Error);\n }\n setUsersLoading(false);\n },\n error: (e) => {\n setUsersError(e);\n setUsersLoading(false);\n }\n }\n );\n }, [firebaseApp, usersPath]);\n\n const saveUser = useCallback(async (user: User): Promise<User> => {\n const firestore = firestoreRef.current;\n if (!firestore || !usersPath) throw Error(\"useFirestoreConfigurationPersistence Firestore not initialised\");\n console.debug(\"Persisting user\", user);\n const roleIds = user.roles?.map(r => r.id);\n const {\n uid,\n ...userData\n } = user;\n return setDoc(doc(firestore, usersPath, uid), {\n ...userData,\n roles: roleIds\n }, { merge: true }).then(() => user);\n }, [usersPath]);\n\n const saveRole = useCallback((role: Role): Promise<void> => {\n const firestore = firestoreRef.current;\n if (!firestore || !rolesPath) throw Error(\"useFirestoreConfigurationPersistence Firestore not initialised\");\n console.debug(\"Persisting role\", role);\n const {\n id,\n ...roleData\n } = role;\n const ref = doc(firestore, rolesPath, id);\n return setDoc(ref, roleData, { merge: true });\n }, [rolesPath]);\n\n const deleteUser = useCallback(async (user: User): Promise<void> => {\n const firestore = firestoreRef.current;\n if (!firestore || !usersPath) throw Error(\"useFirestoreConfigurationPersistence Firestore not initialised\");\n console.debug(\"Deleting\", user);\n const { uid } = user;\n return deleteDoc(doc(firestore, usersPath, uid));\n }, [usersPath]);\n\n const deleteRole = useCallback((role: Role): Promise<void> => {\n const firestore = firestoreRef.current;\n if (!firestore || !rolesPath) throw Error(\"useFirestoreConfigurationPersistence Firestore not initialised\");\n console.debug(\"Deleting\", role);\n const { id } = role;\n const ref = doc(firestore, rolesPath, id);\n return deleteDoc(ref);\n }, [rolesPath]);\n\n const collectionPermissions: PermissionsBuilder = useCallback(({\n collection,\n user\n }) => resolveUserRolePermissions({\n collection,\n user\n }), []);\n\n const userIds = users.map(u => u.uid);\n const defineRolesFor: ((user: User) => Role[] | undefined) = useCallback((user) => {\n if (!users) throw Error(\"Users not loaded\");\n const mgmtUser = users.find(u => u.email?.toLowerCase() === user?.email?.toLowerCase());\n return mgmtUser?.roles;\n }, [userIds]);\n\n return {\n loading,\n roles,\n users,\n saveUser,\n saveRole,\n deleteUser,\n deleteRole,\n usersLimit,\n canEditRoles: canEditRoles === undefined ? true : canEditRoles,\n allowDefaultRolesCreation: allowDefaultRolesCreation === undefined ? true : allowDefaultRolesCreation,\n includeCollectionConfigPermissions: Boolean(includeCollectionConfigPermissions),\n collectionPermissions,\n defineRolesFor\n }\n}\n\nconst docsToUsers = (docs: DocumentSnapshot[]): (UserWithRoleIds)[] => {\n return docs.map((doc) => {\n const data = doc.data() as any;\n const newVar = {\n uid: doc.id,\n ...data,\n created_on: data?.created_on?.toDate(),\n updated_on: data?.updated_on?.toDate()\n };\n return newVar as (UserWithRoleIds);\n });\n}\n\nconst docsToRoles = (docs: DocumentSnapshot[]): Role[] => {\n return docs.map((doc) => ({\n id: doc.id,\n ...doc.data()\n } as Role));\n}\n","import React, { PropsWithChildren } from \"react\";\nimport { UserManagement } from \"./types\";\nimport { User } from \"@firecms/core\";\n\nexport const UserManagementContext = React.createContext<UserManagement<any>>({} as any);\n\nexport interface UserManagementProviderProps<U extends User = User> {\n userManagement: UserManagement<U>\n}\n\nexport function UserManagementProvider<U extends User = User>({\n children,\n userManagement\n }: PropsWithChildren<UserManagementProviderProps<U>>) {\n return (\n <UserManagementContext.Provider value={userManagement}>\n {children}\n </UserManagementContext.Provider>\n );\n};\n","import { useContext } from \"react\";\nimport { UserManagement } from \"../types\";\nimport { UserManagementContext } from \"../UserManagementProvider\";\nimport { User } from \"@firecms/core\";\nexport const useUserManagement = <USER extends User>() => useContext<UserManagement<USER>>(UserManagementContext);\n","import { Chip, getColorSchemeForSeed } from \"@firecms/ui\";\nimport { Role } from \"@firecms/core\";\n\nexport type RoleChipProps = {\n role: Role;\n}\n\nexport function RoleChip({ role }: RoleChipProps) {\n let colorScheme;\n if (role.isAdmin) {\n colorScheme = \"blueDarker\";\n } else if (role.id === \"editor\") {\n colorScheme = \"yellowLight\";\n } else if (role.id === \"viewer\") {\n colorScheme = \"grayLight\";\n } else {\n colorScheme = getColorSchemeForSeed(role.id);\n }\n\n return (\n <Chip\n colorScheme={colorScheme}\n key={role.id}>\n {role.name}\n </Chip>\n );\n\n}\n","import React, { useCallback, useState } from \"react\";\nimport * as Yup from \"yup\";\n\nimport { EntityCollection, FieldCaption, Role, toSnakeCase, } from \"@firecms/core\";\nimport {\n Button,\n Checkbox,\n Dialog,\n DialogActions,\n DialogContent,\n DoneIcon,\n LoadingButton,\n Paper,\n Select,\n SelectItem,\n Table,\n TableBody,\n TableCell,\n TableHeader,\n TableRow,\n TextField,\n Tooltip,\n Typography\n} from \"@firecms/ui\";\nimport { useUserManagement } from \"../../hooks\";\nimport { Formex, getIn, useCreateFormex } from \"@firecms/formex\";\n\nexport const RoleYupSchema = Yup.object().shape({\n id: Yup.string().required(\"Required\"),\n name: Yup.string().required(\"Required\")\n});\n\nexport function RolesDetailsForm({\n open,\n role,\n editable,\n handleClose,\n collections\n }: {\n open: boolean,\n editable?: boolean,\n role?: Role,\n handleClose: () => void,\n collections?: EntityCollection[]\n}) {\n\n const { saveRole } = useUserManagement();\n const isNewRole = !role;\n\n const [savingError, setSavingError] = useState<Error | undefined>();\n\n const onRoleUpdated = useCallback((role: Role) => {\n setSavingError(undefined);\n return saveRole(role);\n }, [saveRole]);\n\n const formex = useCreateFormex({\n initialValues: role ?? {\n name: \"\"\n } as Role,\n onSubmit: (role: Role, formexController) => {\n return onRoleUpdated(role)\n .then(() => {\n formexController.resetForm({\n values: role\n });\n handleClose();\n })\n .catch(e => setSavingError(e));\n },\n validation: (values) => {\n return RoleYupSchema.validate(values, { abortEarly: false })\n .then(() => ({}))\n .catch((e) => {\n const errors: Record<string, string> = {};\n e.inner.forEach((error: any) => {\n errors[error.path] = error.message;\n });\n return errors;\n });\n }\n\n });\n\n const {\n isSubmitting,\n touched,\n values,\n errors,\n handleChange,\n setFieldValue,\n dirty,\n setFieldTouched\n } = formex;\n\n const isAdmin = values.isAdmin ?? false;\n const defaultCreate = values.defaultPermissions?.create ?? false;\n const defaultRead = values.defaultPermissions?.read ?? false;\n const defaultEdit = values.defaultPermissions?.edit ?? false;\n const defaultDelete = values.defaultPermissions?.delete ?? false;\n\n React.useEffect(() => {\n const idTouched = getIn(touched, \"id\");\n if (!idTouched && values.name) {\n setFieldValue(\"id\", toSnakeCase(values.name))\n }\n }, [touched, values.name]);\n\n return (\n <Dialog\n open={open}\n maxWidth={\"4xl\"}\n >\n <Formex value={formex}>\n <form noValidate\n autoComplete={\"off\"}\n onSubmit={formex.handleSubmit}\n style={{\n display: \"flex\",\n flexDirection: \"column\",\n position: \"relative\",\n height: \"100%\"\n }}>\n <DialogContent className=\"flex-grow\">\n <div\n className=\"flex flex-row pt-12 pb-8\">\n <Typography variant={\"h4\"}\n className=\"flex-grow\">\n Role\n </Typography>\n </div>\n\n <div className={\"grid grid-cols-12 gap-8\"}>\n\n <div className={\"col-span-12 md:col-span-8\"}>\n <TextField\n name=\"name\"\n required\n error={touched.name && Boolean(errors.name)}\n value={values.name}\n disabled={isAdmin || !editable}\n onChange={handleChange}\n aria-describedby=\"name-helper-text\"\n label=\"Name\"\n />\n <FieldCaption>\n {touched.name && Boolean(errors.name) ? errors.name : \"Name of this role\"}\n </FieldCaption>\n </div>\n\n <div className={\"col-span-12 md:col-span-4\"}>\n <TextField\n name=\"id\"\n required\n error={touched.id && Boolean(errors.id)}\n value={values.id}\n disabled={!isNewRole || !editable}\n onChange={(e) => {\n handleChange(e);\n setFieldTouched(\"id\", true)\n }}\n aria-describedby=\"id-helper-text\"\n label=\"ID\"\n />\n <FieldCaption>\n {touched.id && Boolean(errors.id) ? errors.id : \"ID of this role\"}\n </FieldCaption>\n </div>\n\n <div className={\"col-span-12\"}>\n <Paper\n\n className=\"bg-inherit\">\n <Table>\n <TableHeader>\n <TableCell></TableCell>\n <TableCell\n align=\"center\">Create\n entities\n </TableCell>\n <TableCell\n align=\"center\">Read\n entities\n </TableCell>\n <TableCell\n align=\"center\">Update\n entities\n </TableCell>\n <TableCell\n align=\"center\">Delete\n entities\n </TableCell>\n </TableHeader>\n\n <TableBody>\n <TableRow>\n <TableCell\n scope=\"row\">\n <strong>All\n collections</strong>\n </TableCell>\n <TableCell\n align=\"center\">\n <Tooltip\n title=\"Create entities in collections\">\n <Checkbox\n disabled={isAdmin || !editable}\n checked={(isAdmin || defaultCreate) ?? false}\n onCheckedChange={(checked) => setFieldValue(\"defaultPermissions.create\", checked)}\n />\n </Tooltip>\n </TableCell>\n\n <TableCell\n align=\"center\">\n <Tooltip\n title=\"Access all data in every collection\">\n <Checkbox\n disabled={isAdmin || !editable}\n checked={(isAdmin || defaultRead) ?? false}\n onCheckedChange={(checked) => setFieldValue(\"defaultPermissions.read\", checked)}\n />\n </Tooltip>\n </TableCell>\n <TableCell\n align=\"center\">\n <Tooltip\n title=\"Update data in any collection\">\n <Checkbox\n disabled={isAdmin || !editable}\n checked={(isAdmin || defaultEdit) ?? false}\n onCheckedChange={(checked) => setFieldValue(\"defaultPermissions.edit\", checked)}\n />\n </Tooltip>\n </TableCell>\n <TableCell\n align=\"center\">\n <Tooltip\n title=\"Delete data in any collection\">\n <Checkbox\n disabled={isAdmin || !editable}\n checked={(isAdmin || defaultDelete) ?? false}\n onCheckedChange={(checked) => setFieldValue(\"defaultPermissions.delete\", checked)}\n />\n\n </Tooltip>\n </TableCell>\n </TableRow>\n {collections && collections.map((col) => (\n <TableRow key={col.name}>\n <TableCell\n scope=\"row\">\n {col.name}\n </TableCell>\n <TableCell\n align=\"center\">\n <Checkbox\n disabled={isAdmin || defaultCreate || !editable}\n checked={(isAdmin || defaultCreate || getIn(values, `collectionPermissions.${col.path}.create`)) ?? false}\n onCheckedChange={(checked) => setFieldValue(`collectionPermissions.${col.path}.create`, checked)}/>\n </TableCell>\n <TableCell\n align=\"center\">\n <Checkbox\n disabled={isAdmin || defaultRead || !editable}\n checked={(isAdmin || defaultRead || getIn(values, `collectionPermissions.${col.path}.read`)) ?? false}\n onCheckedChange={(checked) => setFieldValue(`collectionPermissions.${col.path}.read`, checked)}/>\n </TableCell>\n <TableCell\n align=\"center\">\n <Checkbox\n disabled={isAdmin || defaultEdit || !editable}\n checked={(isAdmin || defaultEdit || getIn(values, `collectionPermissions.${col.path}.edit`)) ?? false}\n onCheckedChange={(checked) => setFieldValue(`collectionPermissions.${col.path}.edit`, checked)}/>\n </TableCell>\n <TableCell\n align=\"center\">\n <Checkbox\n disabled={isAdmin || defaultDelete || !editable}\n checked={(isAdmin || defaultDelete || getIn(values, `collectionPermissions.${col.path}.delete`)) ?? false}\n onCheckedChange={(checked) => setFieldValue(`collectionPermissions.${col.path}.delete`, checked)}/>\n </TableCell>\n </TableRow>\n ))}\n </TableBody>\n </Table>\n </Paper>\n <FieldCaption>\n You can customise the permissions\n that the users related to this\n role can perform in the entities\n of each collection\n </FieldCaption>\n </div>\n\n <div className={\"col-span-12 md:col-span-4\"}>\n <Select\n error={touched.config && Boolean(errors.config)}\n id=\"createCollections\"\n name=\"createCollections\"\n label=\"Create collections\"\n position={\"item-aligned\"}\n disabled={isAdmin || !editable}\n onChange={(event) => setFieldValue(\"config.createCollections\", event.target.value === \"true\")}\n value={isAdmin || values.config?.createCollections ? \"true\" : \"false\"}\n renderValue={(value: any) => value === \"true\" ? \"Yes\" : \"No\"}\n >\n <SelectItem\n value={\"true\"}> Yes </SelectItem>\n <SelectItem\n value={\"false\"}> No </SelectItem>\n </Select>\n\n <FieldCaption>\n {touched.config && Boolean(errors.config) ? errors.config : \"Can the user create collections\"}\n </FieldCaption>\n </div>\n\n <div className={\"col-span-12 md:col-span-4\"}>\n <Select\n error={touched.config && Boolean(errors.config)}\n id=\"editCollections\"\n name=\"editCollections\"\n label=\"Edit collections\"\n disabled={isAdmin || !editable}\n position={\"item-aligned\"}\n onChange={(event) => setFieldValue(\"config.editCollections\", event.target.value === \"own\" ? \"own\" : event.target.value === \"true\")}\n value={isAdmin ? \"true\" : (values.config?.editCollections === \"own\" ? \"own\" : (values.config?.editCollections ? \"true\" : \"false\"))}\n renderValue={(value: any) => value === \"own\" ? \"Own\" : (value === \"true\" ? \"Yes\" : \"No\")}\n >\n <SelectItem\n value={\"true\"}> Yes </SelectItem>\n <SelectItem\n value={\"false\"}> No </SelectItem>\n <SelectItem\n value={\"own\"}> Only\n his/her own </SelectItem>\n </Select>\n\n <FieldCaption>\n {touched.config && Boolean(errors.config) ? errors.config : \"Can the user edit collections\"}\n </FieldCaption>\n </div>\n\n <div className={\"col-span-12 md:col-span-4\"}>\n <Select\n error={touched.config && Boolean(errors.config)}\n id=\"deleteCollections\"\n name=\"deleteCollections\"\n label=\"Delete collections\"\n disabled={isAdmin || !editable}\n position={\"item-aligned\"}\n onChange={(event) => setFieldValue(\"config.deleteCollections\", event.target.value === \"own\" ? \"own\" : event.target.value === \"true\")}\n value={isAdmin ? \"true\" : (values.config?.deleteCollections === \"own\" ? \"own\" : (values.config?.deleteCollections ? \"true\" : \"false\"))}\n renderValue={(value: any) => value === \"own\" ? \"Own\" : (value === \"true\" ? \"Yes\" : \"No\")}\n >\n <SelectItem\n value={\"true\"}> Yes </SelectItem>\n <SelectItem\n value={\"false\"}> No </SelectItem>\n <SelectItem\n value={\"own\"}> Only\n his/her own </SelectItem>\n </Select>\n\n <FieldCaption>\n {touched.config && Boolean(errors.config) ? errors.config : \"Can the user delete collections\"}\n </FieldCaption>\n\n </div>\n\n </div>\n </DialogContent>\n\n <DialogActions position={\"sticky\"}>\n {savingError && <Typography className={\"text-red-500\"}>\n There was an error saving this role\n </Typography>}\n <Button variant={\"text\"}\n onClick={() => {\n handleClose();\n }}>\n Cancel\n </Button>\n <LoadingButton\n variant=\"filled\"\n color=\"primary\"\n type=\"submit\"\n disabled={!dirty}\n loading={isSubmitting}\n startIcon={<DoneIcon/>}\n >\n {isNewRole ? \"Create role\" : \"Update\"}\n </LoadingButton>\n </DialogActions>\n </form>\n\n </Formex>\n </Dialog>\n );\n}\n","import { Role } from \"@firecms/core\";\n\nexport const DEFAULT_ROLES: Role[] = [\n {\n id: \"admin\",\n name: \"Admin\",\n isAdmin: true\n },\n {\n id: \"editor\",\n name: \"Editor\",\n isAdmin: false,\n defaultPermissions: {\n read: true,\n create: true,\n edit: true,\n delete: true\n },\n config: {\n createCollections: true,\n editCollections: \"own\",\n deleteCollections: \"own\"\n }\n },\n {\n id: \"viewer\",\n name: \"Viewer\",\n isAdmin: false,\n defaultPermissions: {\n read: true,\n create: false,\n edit: false,\n delete: false\n }\n }\n];\n","import { useState } from \"react\";\nimport {\n Button,\n CenteredView,\n Checkbox,\n DeleteIcon,\n IconButton,\n Table,\n TableBody,\n TableCell,\n TableHeader,\n TableRow,\n Tooltip,\n Typography\n} from \"@firecms/ui\";\nimport { DeleteConfirmationDialog, Role } from \"@firecms/core\";\nimport { useUserManagement } from \"../../hooks\";\nimport { RoleChip } from \"./RoleChip\";\nimport { DEFAULT_ROLES } from \"./default_roles\";\n\nexport function RolesTable({\n onRoleClicked,\n editable\n }: {\n onRoleClicked: (role: Role) => void;\n editable: boolean;\n}) {\n\n const {\n roles,\n saveRole,\n deleteRole,\n allowDefaultRolesCreation\n } = useUserManagement();\n\n const [roleToBeDeleted, setRoleToBeDeleted] = useState<Role | undefined>(undefined);\n const [deleteInProgress, setDeleteInProgress] = useState<boolean>(false);\n\n return <div\n className=\"w-full overflow-auto\">\n <Table>\n <TableHeader>\n <TableCell header={true} className=\"w-16\"></TableCell>\n <TableCell header={true}>Role</TableCell>\n <TableCell header={true} className={\"items-center\"}>Is Admin</TableCell>\n <TableCell header={true}>Default permissions</TableCell>\n </TableHeader>\n\n <TableBody>\n {roles && roles.map((role) => {\n const canCreateAll = role.isAdmin || role.defaultPermissions?.create;\n const canReadAll = role.isAdmin || role.defaultPermissions?.read;\n const canUpdateAll = role.isAdmin || role.defaultPermissions?.edit;\n const canDeleteAll = role.isAdmin || role.defaultPermissions?.delete;\n return (\n <TableRow\n key={role.name}\n onClick={() => {\n onRoleClicked(role);\n }}\n >\n <TableCell style={{ width: \"64px\" }}>\n {!role.isAdmin &&\n <Tooltip title={\"Delete this role\"}>\n <IconButton\n size={\"small\"}\n disabled={!editable}\n onClick={(event) => {\n event.stopPropagation();\n return setRoleToBeDeleted(role);\n }}>\n <DeleteIcon/>\n </IconButton>\n </Tooltip>}\n </TableCell>\n <TableCell>\n <RoleChip role={role}/>\n </TableCell>\n <TableCell className={\"items-center\"}>\n <Checkbox checked={role.isAdmin ?? false}/>\n </TableCell>\n <TableCell>\n <ul>\n {canCreateAll && <li>Create</li>}\n {canReadAll && <li>Read</li>}\n {canUpdateAll && <li>Update</li>}\n {canDeleteAll && <li>Delete</li>}\n </ul>\n </TableCell>\n </TableRow>\n );\n })}\n\n {(!roles || roles.length === 0) && <TableRow>\n <TableCell colspan={4}>\n <CenteredView className={\"flex flex-col gap-4 my-8 items-center\"}>\n <Typography variant={\"label\"}>\n You don't have any roles yet.\n </Typography>\n {allowDefaultRolesCreation && <Button variant={\"outlined\"}\n onClick={() => {\n DEFAULT_ROLES.forEach((role) => {\n saveRole(role);\n });\n }}>\n Create default roles\n </Button>}\n </CenteredView>\n </TableCell>\n </TableRow>}\n\n </TableBody>\n\n </Table>\n\n <DeleteConfirmationDialog\n open={Boolean(roleToBeDeleted)}\n loading={deleteInProgress}\n onAccept={() => {\n if (roleToBeDeleted) {\n setDeleteInProgress(true);\n deleteRole(roleToBeDeleted)\n .then(() => {\n setRoleToBeDeleted(undefined);\n })\n .finally(() => {\n setDeleteInProgress(false);\n })\n }\n }}\n onCancel={() => {\n setRoleToBeDeleted(undefined);\n }}\n title={<>Delete?</>}\n body={<>Are you sure you want to delete this role?</>}/>\n\n </div>;\n}\n","import React, { useCallback, useState } from \"react\";\n\nimport { Role, useNavigationController } from \"@firecms/core\";\nimport { AddIcon, Button, Container, Tooltip, Typography } from \"@firecms/ui\";\nimport { RolesTable } from \"./RolesTable\";\nimport { RolesDetailsForm } from \"./RolesDetailsForm\";\nimport { useUserManagement } from \"../../hooks\";\n\nexport const RolesView = React.memo(\n function RolesView({ children }: { children?: React.ReactNode }) {\n\n const { collections } = useNavigationController();\n const [dialogOpen, setDialogOpen] = useState(false);\n const [selectedRole, setSelectedRole] = useState<Role | undefined>();\n\n const { canEditRoles } = useUserManagement();\n\n const onRoleClicked = useCallback((user: Role) => {\n setDialogOpen(true);\n setSelectedRole(user);\n }, []);\n\n const handleClose = () => {\n setSelectedRole(undefined);\n setDialogOpen(false);\n };\n\n return (\n <Container className=\"w-full flex flex-col py-4 gap-4\" maxWidth={\"6xl\"}>\n\n {children}\n\n <div className=\"flex items-center mt-12\">\n <Typography gutterBottom variant=\"h4\"\n className=\"flex-grow\"\n component=\"h4\">\n Roles\n </Typography>\n <Tooltip title={!canEditRoles ? \"Update plans to customise roles\" : undefined}>\n <Button\n size={\"large\"}\n disabled={!canEditRoles}\n startIcon={<AddIcon/>}\n onClick={() => setDialogOpen(true)}>\n Add role\n </Button>\n </Tooltip>\n </div>\n\n <RolesTable onRoleClicked={onRoleClicked} editable={Boolean(canEditRoles)}/>\n\n <RolesDetailsForm\n key={selectedRole?.id ?? \"new\"}\n open={dialogOpen}\n role={selectedRole}\n editable={canEditRoles}\n collections={collections}\n handleClose={handleClose}/>\n\n </Container>\n )\n });\n","import React, { useCallback } from \"react\";\nimport * as Yup from \"yup\";\nimport {\n Button,\n Dialog,\n DialogActions,\n DialogContent,\n DoneIcon,\n LoadingButton,\n MultiSelect,\n MultiSelectItem,\n TextField,\n Typography,\n} from \"@firecms/ui\";\nimport { FieldCaption, Role, useAuthController, User, useSnackbarController } from \"@firecms/core\";\nimport { Formex, useCreateFormex } from \"@firecms/formex\";\n\nimport { areRolesEqual } from \"../../utils\";\nimport { useUserManagement } from \"../../hooks\";\nimport { RoleChip } from \"../roles\";\n\nexport const UserYupSchema = Yup.object().shape({\n displayName: Yup.string().required(\"Required\"),\n email: Yup.string().email().required(\"Required\"),\n roles: Yup.array().min(1)\n});\n\nfunction canUserBeEdited(loggedUser: User, user: User, users: User[], roles: Role[], prevUser?: User) {\n const admins = users.filter(u => u.roles?.map(r => r.id).includes(\"admin\"));\n const loggedUserIsAdmin = loggedUser.roles?.map(r => r.id).includes(\"admin\");\n const didRolesChange = !prevUser || !areRolesEqual(prevUser.roles ?? [], user.roles ?? []);\n\n if (didRolesChange && !loggedUserIsAdmin) {\n throw new Error(\"Only admins can change roles\");\n }\n\n // was the admin role removed\n const adminRoleRemoved = prevUser && prevUser.roles?.map(r => r.id).includes(\"admin\") && !user.roles?.map(r => r.id).includes(\"admin\");\n\n // avoid removing the last admin\n if (adminRoleRemoved && admins.length === 1) {\n throw new Error(\"There must be at least one admin\");\n }\n return true;\n}\n\nexport function UserDetailsForm({\n open,\n user: userProp,\n handleClose\n }: {\n open: boolean,\n user?: User,\n handleClose: () => void\n}) {\n\n const snackbarController = useSnackbarController();\n const {\n user: loggedInUser\n } = useAuthController();\n const {\n saveUser,\n users,\n roles,\n } = useUserManagement();\n const isNewUser = !userProp;\n\n const onUserUpdated = useCallback((savedUser: User): Promise<User> => {\n if (!loggedInUser) {\n throw new Error(\"Logged user not found\");\n }\n try {\n canUserBeEdited(loggedInUser, savedUser, users, roles, userProp);\n return saveUser(savedUser);\n } catch (e: any) {\n return Promise.reject(e);\n }\n }, [roles, saveUser, userProp, users, loggedInUser]);\n\n const formex = useCreateFormex({\n initialValues: userProp ?? {\n displayName: \"\",\n email: \"\",\n roles: roles.filter(r => r.id === \"editor\")\n } as User,\n validation: (values) => {\n return UserYupSchema.validate(values, { abortEarly: false })\n .then(() => {\n return {};\n }).catch((e) => {\n return e.inner.reduce((acc: any, error: any) => {\n acc[error.path] = error.message;\n return acc;\n }, {});\n });\n },\n onSubmit: (user: User, formexController) => {\n\n return onUserUpdated(user)\n .then(() => {\n handleClose();\n formexController.resetForm({\n values: user\n });\n }).catch((e) => {\n snackbarController.open({\n type: \"error\",\n message: e.message\n });\n });\n }\n });\n\n const {\n isSubmitting,\n touched,\n handleChange,\n values,\n errors,\n setFieldValue,\n dirty,\n handleSubmit,\n submitCount\n } = formex;\n\n return (\n <Dialog\n open={open}\n onOpenChange={(open) => !open ? handleClose() : undefined}\n maxWidth={\"4xl\"}\n >\n <Formex value={formex}>\n <form\n onSubmit={handleSubmit}\n autoComplete={\"off\"}\n noValidate\n style={{\n display: \"flex\",\n flexDirection: \"column\",\n position: \"relative\",\n height: \"100%\"\n }}>\n <DialogContent className=\"h-full flex-grow\">\n <div\n className=\"flex flex-row pt-4 pb-4\">\n <Typography variant={\"h4\"}\n className=\"flex-grow\">\n User\n </Typography>\n </div>\n\n <div className={\"grid grid-cols-12 gap-8\"}>\n\n <div className={\"col-span-12\"}>\n <TextField\n name=\"displayName\"\n required\n error={submitCount > 0 && Boolean(errors.displayName)}\n value={values.displayName ?? \"\"}\n onChange={handleChange}\n aria-describedby=\"name-helper-text\"\n label=\"Name\"\n />\n <FieldCaption>\n {submitCount > 0 && Boolean(errors.displayName) ? errors.displayName : \"Name of this user\"}\n </FieldCaption>\n </div>\n <div className={\"col-span-12\"}>\n <TextField\n required\n error={submitCount > 0 && Boolean(errors.email)}\n name=\"email\"\n value={values.email ?? \"\"}\n onChange={handleChange}\n aria-describedby=\"email-helper-text\"\n label=\"Email\"\n />\n <FieldCaption>\n {submitCount > 0 && Boolean(errors.email) ? errors.email : \"Email of this user\"}\n </FieldCaption>\n </div>\n <div className={\"col-span-12\"}>\n <MultiSelect\n label=\"Roles\"\n value={values.roles?.map(r => r.id) ?? []}\n onMultiValueChange={(value: string[]) => setFieldValue(\"roles\", value.map(id => roles.find(r => r.id === id) as Role))}\n renderValue={(value: string) => {\n const userRole = roles\n .find((role) => role.id === value);\n if (!userRole) return null;\n return <div className=\"flex flex-wrap space-x-2 space-y-2\">\n <RoleChip key={userRole?.id} role={userRole}/>\n </div>;\n }}>\n {roles.map(userRole => <MultiSelectItem key={userRole.id}\n value={userRole.id}>\n <RoleChip key={userRole?.id} role={userRole}/>\n </MultiSelectItem>)}\n </MultiSelect>\n </div>\n\n </div>\n\n </DialogContent>\n\n <DialogActions>\n\n <Button variant={\"text\"}\n onClick={() => {\n handleClose();\n }}>\n Cancel\n </Button>\n\n <LoadingButton\n variant=\"filled\"\n color=\"primary\"\n type=\"submit\"\n disabled={!dirty}\n loading={isSubmitting}\n startIcon={<DoneIcon/>}\n >\n {isNewUser ? \"Create user\" : \"Update\"}\n </LoadingButton>\n </DialogActions>\n </form>\n </Formex>\n\n </Dialog>\n );\n}\n","import { useState } from \"react\";\n\nimport { format } from \"date-fns\";\nimport * as locales from \"date-fns/locale\";\n\nimport {\n defaultDateFormat,\n DeleteConfirmationDialog, Role,\n useAuthController,\n useCustomizationController, User,\n useSnackbarController\n} from \"@firecms/core\";\nimport {\n Button,\n CenteredView,\n DeleteIcon,\n IconButton,\n Table,\n TableBody,\n TableCell,\n TableHeader,\n TableRow,\n Tooltip,\n Typography,\n} from \"@firecms/ui\";\nimport { useUserManagement } from \"../../hooks\";\nimport { RoleChip } from \"../roles\";\nimport { PersistedUser } from \"../../types\";\n\nexport function UsersTable({ onUserClicked }: {\n onUserClicked: (user: User) => void;\n}) {\n\n const {\n users,\n saveUser,\n deleteUser\n } = useUserManagement<PersistedUser>();\n\n const authController = useAuthController();\n const snackbarController = useSnackbarController();\n\n const customizationController = useCustomizationController();\n const dateUtilsLocale = customizationController?.locale ? locales[customizationController?.locale as keyof typeof locales] : undefined;\n const dateFormat: string = customizationController?.dateTimeFormat ?? defaultDateFormat;\n\n const [userToBeDeleted, setUserToBeDeleted] = useState<User | undefined>(undefined);\n const [deleteInProgress, setDeleteInProgress] = useState<boolean>(false);\n\n return (\n <div className=\"overflow-auto\">\n\n <Table>\n\n <TableHeader>\n <TableCell className=\"truncate w-16\"></TableCell>\n <TableCell>ID</TableCell>\n <TableCell>Email</TableCell>\n <TableCell>Name</TableCell>\n <TableCell>Roles</TableCell>\n <TableCell>Created on</TableCell>\n </TableHeader>\n <TableBody>\n {users && users.map((user) => {\n\n const userRoles: Role[] | undefined = user.roles;\n\n const formattedDate = user.created_on ? format(user.created_on, dateFormat, { locale: dateUtilsLocale }) : \"\";\n\n return (\n <TableRow\n key={\"row_\" + user.uid}\n onClick={() => {\n onUserClicked(user);\n }}\n >\n <TableCell className={\"w-10\"}>\n <Tooltip title={\"Delete this user\"}>\n <IconButton\n size={\"small\"}\n onClick={(event) => {\n event.stopPropagation();\n return setUserToBeDeleted(user);\n }}>\n <DeleteIcon/>\n </IconButton>\n </Tooltip>\n </TableCell>\n <TableCell>{user.uid}</TableCell>\n <TableCell>{user.email}</TableCell>\n <TableCell className={\"font-medium align-left\"}>{user.displayName}</TableCell>\n <TableCell className=\"align-left\">\n {userRoles\n ? <div className=\"flex flex-wrap gap-2\">\n {userRoles.map(userRole =>\n <RoleChip key={userRole?.id} role={userRole}/>\n )}\n </div>\n : null}\n </TableCell>\n <TableCell>{formattedDate}</TableCell>\n </TableRow>\n );\n })}\n\n {(!users || users.length === 0) && <TableRow>\n <TableCell colspan={6}>\n <CenteredView className={\"flex flex-col gap-4 my-8 items-center\"}>\n <Typography variant={\"label\"}>\n There are no users yet\n </Typography>\n <Button variant={\"outlined\"}\n onClick={() => {\n if (!authController.user?.uid) {\n throw Error(\"UsersTable, authController misconfiguration\");\n }\n saveUser({\n uid: authController.user?.uid,\n email: authController.user?.email,\n displayName: authController.user?.displayName,\n photoURL: authController.user?.photoURL,\n providerId: authController.user?.providerId,\n isAnonymous: authController.user?.isAnonymous,\n roles: [{ id: \"admin\", name: \"Admin\" }],\n created_on: new Date()\n })\n .then(() => {\n snackbarController.open({\n type: \"success\",\n message: \"User added successfully\"\n })\n })\n .catch((error) => {\n snackbarController.open({\n type: \"error\",\n message: \"Error adding user: \" + error.message,\n })\n });\n }}>\n\n Add the logged user as an admin\n </Button>\n </CenteredView>\n </TableCell>\n </TableRow>}\n\n </TableBody>\n </Table>\n\n <DeleteConfirmationDialog\n open={Boolean(userToBeDeleted)}\n loading={deleteInProgress}\n onAccept={() => {\n if (userToBeDeleted) {\n setDeleteInProgress(true);\n deleteUser(userToBeDeleted)\n .then(() => {\n setUserToBeDeleted(undefined);\n })\n .catch((error) => {\n snackbarController.open({\n type: \"error\",\n message: \"Error deleting user: \" + error.message,\n })\n })\n .finally(() => {\n setDeleteInProgress(false);\n })\n }\n }}\n onCancel={() => {\n setUserToBeDeleted(undefined);\n }}\n title={<>Delete?</>}\n body={<>Are you sure you want to delete this user?</>}/>\n </div>);\n}\n","import { AddIcon, Button, Container, Typography } from \"@firecms/ui\";\n\nimport { UsersTable } from \"./UsersTable\";\nimport { UserDetailsForm } from \"./UserDetailsForm\";\nimport React, { useCallback, useState } from \"react\";\nimport { useUserManagement } from \"../../hooks/useUserManagement\";\nimport { User } from \"@firecms/core\";\n\nexport const UsersView = function UsersView({ children }: { children?: React.ReactNode }) {\n\n const [dialogOpen, setDialogOpen] = useState<boolean>();\n const [selectedUser, setSelectedUser] = useState<User | undefined>();\n\n const { users, usersLimit } = useUserManagement();\n\n const reachedUsersLimit = usersLimit !== undefined && (users && users.length >= usersLimit);\n\n const onUserClicked = useCallback((user: User) => {\n setSelectedUser(user);\n setDialogOpen(true);\n }, []);\n\n const handleClose = useCallback(() => {\n setDialogOpen(false);\n setSelectedUser(undefined);\n }, []);\n\n return (\n <Container className=\"w-full flex flex-col py-4 gap-4\" maxWidth={\"6xl\"}>\n\n {children}\n\n <div\n className=\"flex items-center mt-12\">\n <Typography gutterBottom variant=\"h4\"\n className=\"flex-grow\"\n component=\"h4\">\n Users\n </Typography>\n <Button\n size={\"large\"}\n disabled={reachedUsersLimit}\n startIcon={<AddIcon/>}\n onClick={() => setDialogOpen(true)}>\n Add user\n </Button>\n </div>\n\n <UsersTable onUserClicked={onUserClicked}/>\n\n <UserDetailsForm\n key={selectedUser?.uid ?? \"new\"}\n open={dialogOpen ?? false}\n user={selectedUser}\n handleClose={handleClose}/>\n\n </Container>\n )\n};\n","import { FireCMSPlugin } from \"@firecms/core\";\nimport { UserManagementProvider } from \"./UserManagementProvider\";\nimport { UserManagement } from \"./types\";\n\nexport function useUserManagementPlugin({ userManagement }: {\n userManagement: UserManagement,\n}): FireCMSPlugin {\n return {\n key: \"user_management\",\n loading: userManagement.loading,\n provider: {\n Component: UserManagementProvider,\n props: {\n userManagement\n }\n }\n }\n}\n","import { CMSView } from \"@firecms/core\";\nimport { RolesView, UsersView } from \"./components\";\n\nexport const userManagementAdminViews: CMSView[] = [\n {\n path: \"users\",\n name: \"CMS Users\",\n group: \"Admin\",\n icon: \"face\",\n view: <UsersView/>\n },\n {\n path: \"roles\",\n name: \"Roles\",\n group: \"Admin\",\n icon: \"gpp_good\",\n view: <RolesView/>\n }\n]\n"],"names":["RESERVED_GROUPS","DEFAULT_PERMISSIONS","resolveUserRolePermissions","collection","user","roles","basePermissions","role","resolveCollectionRole","mergePermissions","id","permA","permB","getUserRoles","fireCMSUser","r","areRolesEqual","rolesA","rolesB","rolesAIds","rolesBIds","cacheDelegatedLoginToken","projectId","delegatedToken","data","parseJwt","expiry","getDelegatedLoginTokenFromCache","entry","clearDelegatedLoginTokensCache","i","key","token","base64","jsonPayload","c","darkenColor","hexColor","darkenBy","color","g","b","hexToRgbaWithOpacity","opacity","alpha","useBuildFirestoreUserManagement","firebaseApp","usersPath","rolesPath","usersLimit","canEditRoles","allowDefaultRolesCreation","includeCollectionConfigPermissions","firestoreRef","useRef","rolesLoading","setRolesLoading","React","usersLoading","setUsersLoading","setRoles","usersWithRoleIds","setUsersWithRoleIds","users","u","rolesError","setRolesError","usersError","setUsersError","loading","useEffect","getFirestore","firestore","onSnapshot","snapshot","newRoles","docsToRoles","e","newUsers","docsToUsers","saveUser","useCallback","roleIds","uid","userData","setDoc","doc","saveRole","roleData","ref","deleteUser","deleteDoc","deleteRole","collectionPermissions","userIds","defineRolesFor","docs","UserManagementContext","UserManagementProvider","children","userManagement","useUserManagement","useContext","RoleChip","colorScheme","getColorSchemeForSeed","jsx","Chip","RoleYupSchema","Yup","RolesDetailsForm","open","editable","handleClose","collections","isNewRole","savingError","setSavingError","useState","onRoleUpdated","formex","useCreateFormex","formexController","values","errors","error","isSubmitting","touched","handleChange","setFieldValue","dirty","setFieldTouched","isAdmin","defaultCreate","defaultRead","defaultEdit","defaultDelete","getIn","toSnakeCase","Dialog","Formex","jsxs","DialogContent","Typography","TextField","FieldCaption","Paper","Table","TableHeader","TableCell","TableBody","TableRow","Tooltip","Checkbox","checked","col","Select","event","value","SelectItem","DialogActions","Button","LoadingButton","DoneIcon","DEFAULT_ROLES","RolesTable","onRoleClicked","roleToBeDeleted","setRoleToBeDeleted","deleteInProgress","setDeleteInProgress","canCreateAll","canReadAll","canUpdateAll","canDeleteAll","IconButton","DeleteIcon","CenteredView","DeleteConfirmationDialog","RolesView","useNavigationController","dialogOpen","setDialogOpen","selectedRole","setSelectedRole","Container","AddIcon","UserYupSchema","canUserBeEdited","loggedUser","prevUser","admins","loggedUserIsAdmin","UserDetailsForm","userProp","snackbarController","useSnackbarController","loggedInUser","useAuthController","isNewUser","onUserUpdated","savedUser","acc","handleSubmit","submitCount","MultiSelect","userRole","MultiSelectItem","UsersTable","onUserClicked","authController","customizationController","useCustomizationController","dateUtilsLocale","locales","dateFormat","defaultDateFormat","userToBeDeleted","setUserToBeDeleted","userRoles","formattedDate","format","UsersView","selectedUser","setSelectedUser","reachedUsersLimit","useUserManagementPlugin","userManagementAdminViews"],"mappings":"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"}
|
1
|
+
{"version":3,"file":"index.umd.js","sources":["../src/utils/permissions.ts","../src/utils/local_storage.ts","../src/utils/colors.ts","../src/hooks/useBuildUserManagement.tsx","../src/UserManagementProvider.tsx","../src/hooks/useUserManagement.tsx","../src/components/roles/RoleChip.tsx","../src/components/roles/RolesDetailsForm.tsx","../src/components/roles/default_roles.tsx","../src/components/roles/RolesTable.tsx","../src/components/roles/RolesView.tsx","../src/components/users/UserDetailsForm.tsx","../src/components/users/UsersTable.tsx","../src/components/users/UsersView.tsx","../src/useUserManagementPlugin.tsx","../src/admin_views.tsx"],"sourcesContent":["import { EntityCollection, Permissions, Role, User } from \"@firecms/core\";\n\nexport const RESERVED_GROUPS = [\"Admin\"];\n\nconst DEFAULT_PERMISSIONS = {\n read: false,\n edit: false,\n create: false,\n delete: false\n};\n\nexport function resolveUserRolePermissions<UserType extends User>\n({\n collection,\n user\n }: {\n collection: EntityCollection<any>,\n user: UserType | null\n}): Permissions {\n\n const roles = user?.roles;\n if (!roles) {\n return DEFAULT_PERMISSIONS;\n } else if (collection.ownerId === user?.uid) {\n return {\n read: true,\n create: true,\n edit: true,\n delete: true\n };\n } else {\n const basePermissions = {\n read: false,\n create: false,\n edit: false,\n delete: false\n };\n\n return roles\n .map(role => resolveCollectionRole(role, collection.id))\n .reduce(mergePermissions, basePermissions);\n }\n}\n\nfunction resolveCollectionRole(role: Role, id: string): Permissions {\n\n const basePermissions = {\n read: (role.isAdmin || role.defaultPermissions?.read) ?? false,\n create: (role.isAdmin || role.defaultPermissions?.create) ?? false,\n edit: (role.isAdmin || role.defaultPermissions?.edit) ?? false,\n delete: (role.isAdmin || role.defaultPermissions?.delete) ?? false\n };\n const thisCollectionPermissions = role.collectionPermissions?.[id];\n if (thisCollectionPermissions) {\n return mergePermissions(thisCollectionPermissions, basePermissions);\n } else if (role.defaultPermissions) {\n return mergePermissions(role.defaultPermissions, basePermissions);\n } else {\n return basePermissions;\n }\n}\n\nconst mergePermissions = (permA: Permissions, permB: Permissions) => {\n return {\n read: permA.read || permB.read,\n create: permA.create || permB.create,\n edit: permA.edit || permB.edit,\n delete: permA.delete || permB.delete\n };\n}\n\nexport function getUserRoles(roles: Role[], fireCMSUser: User): Role[] | undefined {\n return !roles\n ? undefined\n : (fireCMSUser.roles\n ? fireCMSUser.roles\n .map(role => roles.find((r) => r.id === role.id))\n .filter(Boolean) as Role[]\n : []);\n}\n\nexport const areRolesEqual = (rolesA: Role[], rolesB: Role[]) => {\n const rolesAIds = rolesA.map(r => r.id);\n const rolesBIds = rolesB.map(r => r.id);\n return rolesAIds.length === rolesB.length && rolesAIds.every((role) => rolesBIds.includes(role));\n}\n","// const tokens = new Map<string, {\n// token: string,\n// expiry: Date\n// }>();\n\nexport function cacheDelegatedLoginToken(projectId: string, delegatedToken?: string) {\n if (!delegatedToken) {\n return;\n }\n\n const data = parseJwt(delegatedToken);\n // @ts-ignore\n const expiry = new Date(data.exp * 1000);\n localStorage.setItem(`auth_token::${projectId}`, JSON.stringify({\n token: delegatedToken,\n expiry\n }));\n\n}\n\nexport function getDelegatedLoginTokenFromCache(projectId: string) {\n const entry = localStorage.getItem(`auth_token::${projectId}`);\n if (entry) {\n const data = JSON.parse(entry);\n data.expiry = new Date(data.expiry);\n if (data.expiry > new Date()) {\n return data.token;\n }\n }\n return undefined;\n}\n\nexport function clearDelegatedLoginTokensCache() {\n for (let i = 0; i < localStorage.length; i++) {\n const key = localStorage.key(i);\n if (key?.startsWith(\"auth_token::\")) {\n localStorage.removeItem(key);\n }\n }\n}\n\nfunction parseJwt(token?: string): object {\n if (!token) {\n throw new Error(\"No JWT token\");\n }\n const base64Url = token.split(\".\")[1];\n const base64 = base64Url.replace(/-/g, \"+\").replace(/_/g, \"/\");\n const jsonPayload = decodeURIComponent(window.atob(base64).split(\"\").map(function (c) {\n return \"%\" + (\"00\" + c.charCodeAt(0).toString(16)).slice(-2);\n }).join(\"\"));\n\n return JSON.parse(jsonPayload);\n}\n","export function darkenColor(hexColor: string, darkenBy = 10): string {\n // Check input validity\n if (!/^#([0-9A-Fa-f]{3}){1,2}$/.test(hexColor)) {\n throw new Error(\"Invalid color format\");\n }\n\n // If shorthand form, convert to full form\n let color = hexColor.substring(1).split(\"\");\n if (color.length === 3) {\n color = [color[0], color[0], color[1], color[1], color[2], color[2]];\n }\n\n // Convert to RGB values\n let r = parseInt(color[0] + color[1], 16);\n let g = parseInt(color[2] + color[3], 16);\n let b = parseInt(color[4] + color[5], 16);\n\n // Reduce each color component by the specified percentage (darkenBy)\n r = Math.floor(r * (1 - darkenBy / 100));\n g = Math.floor(g * (1 - darkenBy / 100));\n b = Math.floor(b * (1 - darkenBy / 100));\n\n // Recombine into hex and return\n return \"#\" +\n (r < 16 ? \"0\" : \"\") + r.toString(16) +\n (g < 16 ? \"0\" : \"\") + g.toString(16) +\n (b < 16 ? \"0\" : \"\") + b.toString(16);\n}\n\nexport function hexToRgbaWithOpacity(hexColor: string, opacity = 10): string {\n // Check input validity\n if (!/^#([0-9A-Fa-f]{3}){1,2}$/.test(hexColor)) {\n throw new Error(\"Invalid color format\");\n }\n\n // If shorthand form, convert to full form\n let color = hexColor.substring(1).split(\"\");\n if (color.length === 3) {\n color = [color[0], color[0], color[1], color[1], color[2], color[2]];\n }\n\n // Convert to RGB values\n const r = parseInt(color[0] + color[1], 16);\n const g = parseInt(color[2] + color[3], 16);\n const b = parseInt(color[4] + color[5], 16);\n\n // Convert opacity to a decimal for CSS\n const alpha = opacity / 100;\n\n // Construct and return the RGBA color\n return `rgba(${r}, ${g}, ${b}, ${alpha})`;\n}\n","import React, { useCallback, useEffect } from \"react\";\nimport equal from \"react-fast-compare\"\n\nimport { UserManagement } from \"../types\";\nimport {\n Authenticator,\n DataSourceDelegate,\n Entity,\n PermissionsBuilder,\n removeUndefined,\n Role,\n User\n} from \"@firecms/core\";\nimport { resolveUserRolePermissions } from \"../utils\";\n\ntype UserWithRoleIds = Omit<User, \"roles\"> & { roles: string[] };\n\nexport interface UserManagementParams {\n\n /**\n * The delegate in charge of persisting the data.\n */\n dataSourceDelegate?: DataSourceDelegate;\n\n /**\n * Path where the plugin users configuration is stored.\n * Default: __FIRECMS/config/users\n * You can specify a different path if you want to store the user management configuration in a different place.\n * Please keep in mind that the FireCMS users are not necessarily the same as the Firebase users (but they can be).\n * The path should be relative to the root of the database, and should always have an odd number of segments.\n */\n usersPath?: string;\n\n /**\n * Path where the plugin roles configuration is stored.\n * Default: __FIRECMS/config/roles\n */\n rolesPath?: string;\n\n /**\n * Maximum number of users that can be created.\n */\n usersLimit?: number;\n\n /**\n * Can the logged user edit roles\n */\n canEditRoles?: boolean;\n\n /**\n * If there are no roles in the database, provide a button to create the default roles.\n */\n allowDefaultRolesCreation?: boolean;\n\n /**\n * Include the collection config permissions in the user management system.\n */\n includeCollectionConfigPermissions?: boolean;\n\n}\n\n/**\n * This hook is used to build a user management object that can be used to\n * manage users and roles in a Firestore backend.\n * @param dataSourceDelegate\n * @param usersPath\n * @param rolesPath\n * @param usersLimit\n * @param canEditRoles\n * @param allowDefaultRolesCreation\n * @param includeCollectionConfigPermissions\n */\nexport function useBuildUserManagement({\n dataSourceDelegate,\n usersPath = \"__FIRECMS/config/users\",\n rolesPath = \"__FIRECMS/config/roles\",\n usersLimit,\n canEditRoles = true,\n allowDefaultRolesCreation,\n includeCollectionConfigPermissions\n }: UserManagementParams): UserManagement {\n\n const [rolesLoading, setRolesLoading] = React.useState<boolean>(true);\n const [usersLoading, setUsersLoading] = React.useState<boolean>(true);\n const [roles, setRoles] = React.useState<Role[]>([]);\n const [usersWithRoleIds, setUsersWithRoleIds] = React.useState<UserWithRoleIds[]>([]);\n\n const users = usersWithRoleIds.map(u => ({\n ...u,\n roles: roles.filter(r => u.roles?.includes(r.id))\n }) as User);\n\n const [rolesError, setRolesError] = React.useState<Error | undefined>();\n const [usersError, setUsersError] = React.useState<Error | undefined>();\n\n const loading = rolesLoading || usersLoading;\n\n useEffect(() => {\n if (!dataSourceDelegate || !rolesPath) return;\n if (dataSourceDelegate.initialised !== undefined && !dataSourceDelegate.initialised) return;\n if (dataSourceDelegate.authenticated !== undefined && !dataSourceDelegate.authenticated) {\n setRolesLoading(false);\n return;\n }\n\n setRolesLoading(true);\n return dataSourceDelegate.listenCollection?.({\n path: rolesPath,\n onUpdate(entities: Entity<any>[]): void {\n setRolesError(undefined);\n try {\n const newRoles = entityToRoles(entities);\n if (!equal(newRoles, roles))\n setRoles(newRoles);\n } catch (e) {\n setRoles([]);\n console.error(\"Error loading roles\", e);\n setRolesError(e as Error);\n }\n setRolesLoading(false);\n },\n onError(e: any): void {\n setRoles([]);\n console.error(\"Error loading roles\", e);\n setRolesError(e);\n setRolesLoading(false);\n }\n });\n\n }, [dataSourceDelegate?.initialised, dataSourceDelegate?.authenticated, rolesPath]);\n\n useEffect(() => {\n if (!dataSourceDelegate || !usersPath) return;\n if (dataSourceDelegate.initialised !== undefined && !dataSourceDelegate.initialised) return;\n if (dataSourceDelegate.authenticated !== undefined && !dataSourceDelegate.authenticated) {\n setUsersLoading(false);\n return;\n }\n\n setUsersLoading(true);\n return dataSourceDelegate.listenCollection?.({\n path: usersPath,\n onUpdate(entities: Entity<any>[]): void {\n setUsersError(undefined);\n try {\n const newUsers = entitiesToUsers(entities);\n if (!equal(newUsers, usersWithRoleIds))\n setUsersWithRoleIds(newUsers);\n } catch (e) {\n setUsersWithRoleIds([]);\n console.error(\"Error loading users\", e);\n setUsersError(e as Error);\n }\n setUsersLoading(false);\n },\n onError(e: any): void {\n setUsersWithRoleIds([]);\n console.error(\"Error loading users\", e);\n setUsersError(e);\n setUsersLoading(false);\n }\n });\n\n }, [dataSourceDelegate?.initialised, dataSourceDelegate?.authenticated, usersPath]);\n\n const saveUser = useCallback(async (user: User): Promise<User> => {\n if (!dataSourceDelegate) throw Error(\"useBuildUserManagement Firebase not initialised\");\n if (!usersPath) throw Error(\"useBuildUserManagement Firestore not initialised\");\n\n console.debug(\"Persisting user\", user);\n\n const roleIds = user.roles?.map(r => r.id);\n const email = user.email?.toLowerCase().trim();\n if (!email) throw Error(\"Email is required\");\n\n const userExists = users.find(u => u.email?.toLowerCase() === email);\n const data = {\n ...user,\n roles: roleIds ?? []\n };\n if (!userExists) {\n // @ts-ignore\n data.created_on = new Date();\n }\n\n return dataSourceDelegate.saveEntity({\n status: \"existing\",\n path: usersPath,\n entityId: email,\n values: removeUndefined(data)\n }).then(() => user);\n }, [usersPath, dataSourceDelegate?.initialised]);\n\n const saveRole = useCallback((role: Role): Promise<void> => {\n if (!dataSourceDelegate) throw Error(\"useBuildUserManagement Firebase not initialised\");\n if (!rolesPath) throw Error(\"useBuildUserManagement Firestore not initialised\");\n console.debug(\"Persisting role\", role);\n const {\n id,\n ...roleData\n } = role;\n return dataSourceDelegate.saveEntity({\n status: \"existing\",\n path: rolesPath,\n entityId: id,\n values: removeUndefined(roleData)\n }).then(() => {\n return;\n });\n }, [rolesPath, dataSourceDelegate?.initialised]);\n\n const deleteUser = useCallback(async (user: User): Promise<void> => {\n if (!dataSourceDelegate) throw Error(\"useBuildUserManagement Firebase not initialised\");\n if (!usersPath) throw Error(\"useBuildUserManagement Firestore not initialised\");\n console.debug(\"Deleting\", user);\n const { uid } = user;\n const entity: Entity<any> = {\n path: usersPath,\n id: uid,\n values: {}\n };\n await dataSourceDelegate.deleteEntity({ entity })\n }, [usersPath, dataSourceDelegate?.initialised]);\n\n const deleteRole = useCallback(async (role: Role): Promise<void> => {\n if (!dataSourceDelegate) throw Error(\"useBuildUserManagement Firebase not initialised\");\n if (!rolesPath) throw Error(\"useBuildUserManagement Firestore not initialised\");\n console.debug(\"Deleting\", role);\n const { id } = role;\n const entity: Entity<any> = {\n path: rolesPath,\n id: id,\n values: {}\n };\n await dataSourceDelegate.deleteEntity({ entity })\n }, [rolesPath, dataSourceDelegate?.initialised]);\n\n const collectionPermissions: PermissionsBuilder = useCallback(({\n collection,\n user\n }) => resolveUserRolePermissions({\n collection,\n user\n }), []);\n\n const defineRolesFor: ((user: User) => Role[] | undefined) = useCallback((user) => {\n if (!users) throw Error(\"Users not loaded\");\n const mgmtUser = users.find(u => u.email?.toLowerCase() === user?.email?.toLowerCase());\n return mgmtUser?.roles;\n }, [users]);\n\n const authenticator: Authenticator = useCallback(({ user }) => {\n console.debug(\"Authenticating user\", user);\n\n if (loading) {\n console.warn(\"User management is still loading\");\n return false;\n }\n\n // This is an example of how you can link the access system to the user management plugin\n if (users.length === 0) {\n return true; // If there are no users created yet, we allow access to every user\n }\n\n const mgmtUser = users.find(u => u.email?.toLowerCase() === user?.email?.toLowerCase());\n if (mgmtUser) {\n return true;\n }\n\n throw Error(\"Could not find a user with the provided email in the user management system.\");\n }, [loading, users, usersError, rolesError]);\n\n const isAdmin = roles.some(r => r.id === \"admin\");\n\n return {\n loading,\n roles,\n users,\n saveUser,\n saveRole,\n rolesError,\n deleteUser,\n deleteRole,\n usersLimit,\n usersError,\n isAdmin,\n canEditRoles: canEditRoles === undefined ? true : canEditRoles,\n allowDefaultRolesCreation: allowDefaultRolesCreation === undefined ? true : allowDefaultRolesCreation,\n includeCollectionConfigPermissions: Boolean(includeCollectionConfigPermissions),\n collectionPermissions,\n defineRolesFor,\n authenticator\n }\n}\n\nconst entitiesToUsers = (docs: Entity<Omit<UserWithRoleIds, \"id\">>[]): (UserWithRoleIds)[] => {\n return docs.map((doc) => {\n const data = doc.values as any;\n const newVar = {\n uid: doc.id,\n ...data,\n created_on: data?.created_on,\n updated_on: data?.updated_on\n };\n return newVar as (UserWithRoleIds);\n });\n}\n\nconst entityToRoles = (entities: Entity<Omit<Role, \"id\">>[]): Role[] => {\n return entities.map((doc) => ({\n id: doc.id,\n ...doc.values\n } as Role));\n}\n","import React, { PropsWithChildren } from \"react\";\nimport { UserManagement } from \"./types\";\nimport { User } from \"@firecms/core\";\n\nexport const UserManagementContext = React.createContext<UserManagement<any>>({} as any);\n\nexport interface UserManagementProviderProps<U extends User = User> {\n userManagement: UserManagement<U>\n}\n\nexport function UserManagementProvider<U extends User = User>({\n children,\n userManagement\n }: PropsWithChildren<UserManagementProviderProps<U>>) {\n return (\n <UserManagementContext.Provider value={userManagement}>\n {children}\n </UserManagementContext.Provider>\n );\n};\n","import { useContext } from \"react\";\nimport { UserManagement } from \"../types\";\nimport { UserManagementContext } from \"../UserManagementProvider\";\nimport { User } from \"@firecms/core\";\nexport const useUserManagement = <USER extends User>() => useContext<UserManagement<USER>>(UserManagementContext);\n","import { Chip, getColorSchemeForSeed } from \"@firecms/ui\";\nimport { Role } from \"@firecms/core\";\n\nexport type RoleChipProps = {\n role: Role;\n}\n\nexport function RoleChip({ role }: RoleChipProps) {\n let colorScheme;\n if (role.isAdmin) {\n colorScheme = \"blueDarker\";\n } else if (role.id === \"editor\") {\n colorScheme = \"yellowLight\";\n } else if (role.id === \"viewer\") {\n colorScheme = \"grayLight\";\n } else {\n colorScheme = getColorSchemeForSeed(role.id);\n }\n\n return (\n <Chip\n colorScheme={colorScheme}\n key={role.id}>\n {role.name}\n </Chip>\n );\n\n}\n","import React, { useCallback, useState } from \"react\";\nimport * as Yup from \"yup\";\n\nimport { EntityCollection, FieldCaption, Role, toSnakeCase, useAuthController, User, } from \"@firecms/core\";\nimport {\n Button,\n Checkbox,\n Dialog,\n DialogActions,\n DialogContent,\n DoneIcon,\n LoadingButton,\n Paper,\n Select,\n SelectItem,\n Table,\n TableBody,\n TableCell,\n TableHeader,\n TableRow,\n TextField,\n Tooltip,\n Typography\n} from \"@firecms/ui\";\nimport { useUserManagement } from \"../../hooks\";\nimport { Formex, getIn, useCreateFormex } from \"@firecms/formex\";\n\nexport const RoleYupSchema = Yup.object().shape({\n id: Yup.string().required(\"Required\"),\n name: Yup.string().required(\"Required\")\n});\n\nfunction canRoleBeEdited(loggedUser: User) {\n const loggedUserIsAdmin = loggedUser.roles?.map(r => r.id).includes(\"admin\");\n if (!loggedUserIsAdmin) {\n throw new Error(\"Only admins can edit roles\");\n }\n\n return true;\n}\n\nexport function RolesDetailsForm({\n open,\n role,\n editable,\n handleClose,\n collections\n }: {\n open: boolean,\n editable?: boolean,\n role?: Role,\n handleClose: () => void,\n collections?: EntityCollection[]\n}) {\n\n const { saveRole } = useUserManagement();\n const isNewRole = !role;\n const {\n user: loggedInUser\n } = useAuthController();\n\n const [savingError, setSavingError] = useState<Error | undefined>();\n\n const onRoleUpdated = useCallback((role: Role) => {\n setSavingError(undefined);\n if (!loggedInUser) throw new Error(\"User not found\");\n canRoleBeEdited(loggedInUser);\n return saveRole(role);\n }, [saveRole, loggedInUser]);\n\n const formex = useCreateFormex({\n initialValues: role ?? {\n name: \"\"\n } as Role,\n onSubmit: (role: Role, formexController) => {\n try {\n return onRoleUpdated(role)\n .then(() => {\n formexController.resetForm({\n values: role\n });\n handleClose();\n })\n .catch(e => {\n setSavingError(e);\n });\n } catch (e: any) {\n setSavingError(e);\n return Promise.resolve();\n }\n },\n validation: (values) => {\n return RoleYupSchema.validate(values, { abortEarly: false })\n .then(() => ({}))\n .catch((e) => {\n const errors: Record<string, string> = {};\n e.inner.forEach((error: any) => {\n errors[error.path] = error.message;\n });\n return errors;\n });\n }\n\n });\n\n const {\n isSubmitting,\n touched,\n values,\n errors,\n handleChange,\n setFieldValue,\n dirty,\n setFieldTouched\n } = formex;\n\n const isAdmin = values.isAdmin ?? false;\n const defaultCreate = values.defaultPermissions?.create ?? false;\n const defaultRead = values.defaultPermissions?.read ?? false;\n const defaultEdit = values.defaultPermissions?.edit ?? false;\n const defaultDelete = values.defaultPermissions?.delete ?? false;\n\n React.useEffect(() => {\n const idTouched = getIn(touched, \"id\");\n if (!idTouched && values.name) {\n setFieldValue(\"id\", toSnakeCase(values.name))\n }\n }, [touched, values.name]);\n\n return (\n <Dialog\n open={open}\n maxWidth={\"4xl\"}\n >\n <Formex value={formex}>\n <form noValidate\n autoComplete={\"off\"}\n onSubmit={formex.handleSubmit}\n style={{\n display: \"flex\",\n flexDirection: \"column\",\n position: \"relative\",\n height: \"100%\"\n }}>\n <DialogContent className=\"flex-grow\">\n <div\n className=\"flex flex-row pt-12 pb-8\">\n <Typography variant={\"h4\"}\n className=\"flex-grow\">\n Role\n </Typography>\n </div>\n\n <div className={\"grid grid-cols-12 gap-8\"}>\n\n <div className={\"col-span-12 md:col-span-8\"}>\n <TextField\n name=\"name\"\n required\n error={touched.name && Boolean(errors.name)}\n value={values.name}\n disabled={isAdmin || !editable}\n onChange={handleChange}\n aria-describedby=\"name-helper-text\"\n label=\"Name\"\n />\n <FieldCaption>\n {touched.name && Boolean(errors.name) ? errors.name : \"Name of this role\"}\n </FieldCaption>\n </div>\n\n <div className={\"col-span-12 md:col-span-4\"}>\n <TextField\n name=\"id\"\n required\n error={touched.id && Boolean(errors.id)}\n value={values.id}\n disabled={!isNewRole || !editable}\n onChange={(e) => {\n handleChange(e);\n setFieldTouched(\"id\", true)\n }}\n aria-describedby=\"id-helper-text\"\n label=\"ID\"\n />\n <FieldCaption>\n {touched.id && Boolean(errors.id) ? errors.id : \"ID of this role\"}\n </FieldCaption>\n </div>\n\n <div className={\"col-span-12\"}>\n <Paper className=\"bg-inherit overflow-hidden\">\n <Table className={\"w-full rounded-md\"}>\n <TableHeader className={\"rounded-md\"}>\n <TableCell></TableCell>\n <TableCell\n align=\"center\">Create\n entities\n </TableCell>\n <TableCell\n align=\"center\">Read\n entities\n </TableCell>\n <TableCell\n align=\"center\">Update\n entities\n </TableCell>\n <TableCell\n align=\"center\">Delete\n entities\n </TableCell>\n <TableCell\n align=\"center\">\n </TableCell>\n </TableHeader>\n\n <TableBody>\n <TableRow>\n <TableCell\n scope=\"row\">\n <strong>All\n collections</strong>\n </TableCell>\n <TableCell\n align=\"center\">\n <Tooltip\n title=\"Create entities in collections\">\n <Checkbox\n disabled={isAdmin || !editable}\n checked={(isAdmin || defaultCreate) ?? false}\n onCheckedChange={(checked) => setFieldValue(\"defaultPermissions.create\", checked)}\n />\n </Tooltip>\n </TableCell>\n\n <TableCell\n align=\"center\">\n <Tooltip\n title=\"Access all data in every collection\">\n <Checkbox\n disabled={isAdmin || !editable}\n checked={(isAdmin || defaultRead) ?? false}\n onCheckedChange={(checked) => setFieldValue(\"defaultPermissions.read\", checked)}\n />\n </Tooltip>\n </TableCell>\n <TableCell\n align=\"center\">\n <Tooltip\n title=\"Update data in any collection\">\n <Checkbox\n disabled={isAdmin || !editable}\n checked={(isAdmin || defaultEdit) ?? false}\n onCheckedChange={(checked) => setFieldValue(\"defaultPermissions.edit\", checked)}\n />\n </Tooltip>\n </TableCell>\n <TableCell\n align=\"center\">\n <Tooltip\n title=\"Delete data in any collection\">\n <Checkbox\n disabled={isAdmin || !editable}\n checked={(isAdmin || defaultDelete) ?? false}\n onCheckedChange={(checked) => setFieldValue(\"defaultPermissions.delete\", checked)}\n />\n\n </Tooltip>\n </TableCell>\n <TableCell\n align=\"center\">\n </TableCell>\n </TableRow>\n {collections && collections.map((col) => (\n <TableRow key={col.name}>\n <TableCell\n scope=\"row\">\n {col.name}\n </TableCell>\n <TableCell\n align=\"center\">\n <Checkbox\n disabled={isAdmin || defaultCreate || !editable}\n checked={(isAdmin || defaultCreate || getIn(values, `collectionPermissions.${col.id}.create`)) ?? false}\n onCheckedChange={(checked) => setFieldValue(`collectionPermissions.${col.id}.create`, checked)}/>\n </TableCell>\n <TableCell\n align=\"center\">\n <Checkbox\n disabled={isAdmin || defaultRead || !editable}\n checked={(isAdmin || defaultRead || getIn(values, `collectionPermissions.${col.id}.read`)) ?? false}\n onCheckedChange={(checked) => setFieldValue(`collectionPermissions.${col.id}.read`, checked)}/>\n </TableCell>\n <TableCell\n align=\"center\">\n <Checkbox\n disabled={isAdmin || defaultEdit || !editable}\n checked={(isAdmin || defaultEdit || getIn(values, `collectionPermissions.${col.id}.edit`)) ?? false}\n onCheckedChange={(checked) => setFieldValue(`collectionPermissions.${col.id}.edit`, checked)}/>\n </TableCell>\n <TableCell\n align=\"center\">\n <Checkbox\n disabled={isAdmin || defaultDelete || !editable}\n checked={(isAdmin || defaultDelete || getIn(values, `collectionPermissions.${col.id}.delete`)) ?? false}\n onCheckedChange={(checked) => setFieldValue(`collectionPermissions.${col.id}.delete`, checked)}/>\n </TableCell>\n\n <TableCell\n align=\"center\">\n <Tooltip\n title=\"Allow all permissions in this collections\">\n <Button\n className={\"color-inherit\"}\n onClick={() => {\n setFieldValue(`collectionPermissions.${col.id}.create`, true);\n setFieldValue(`collectionPermissions.${col.id}.read`, true);\n setFieldValue(`collectionPermissions.${col.id}.edit`, true);\n setFieldValue(`collectionPermissions.${col.id}.delete`, true);\n }}\n disabled={isAdmin || !editable}\n variant={\"text\"}>\n All\n </Button>\n\n </Tooltip>\n </TableCell>\n </TableRow>\n ))}\n </TableBody>\n </Table>\n </Paper>\n <FieldCaption>\n You can customise the permissions\n that the users related to this\n role can perform in the entities\n of each collection\n </FieldCaption>\n </div>\n\n <div className={\"col-span-12 md:col-span-4\"}>\n <Select\n error={touched.config && Boolean(errors.config)}\n id=\"createCollections\"\n name=\"createCollections\"\n label=\"Create collections\"\n position={\"item-aligned\"}\n disabled={isAdmin || !editable}\n onChange={(event) => setFieldValue(\"config.createCollections\", event.target.value === \"true\")}\n value={isAdmin || values.config?.createCollections ? \"true\" : \"false\"}\n renderValue={(value: any) => value === \"true\" ? \"Yes\" : \"No\"}\n >\n <SelectItem\n value={\"true\"}> Yes </SelectItem>\n <SelectItem\n value={\"false\"}> No </SelectItem>\n </Select>\n\n <FieldCaption>\n {touched.config && Boolean(errors.config) ? errors.config : \"Can the user create collections\"}\n </FieldCaption>\n </div>\n\n <div className={\"col-span-12 md:col-span-4\"}>\n <Select\n error={touched.config && Boolean(errors.config)}\n id=\"editCollections\"\n name=\"editCollections\"\n label=\"Edit collections\"\n disabled={isAdmin || !editable}\n position={\"item-aligned\"}\n onChange={(event) => setFieldValue(\"config.editCollections\", event.target.value === \"own\" ? \"own\" : event.target.value === \"true\")}\n value={isAdmin ? \"true\" : (values.config?.editCollections === \"own\" ? \"own\" : (values.config?.editCollections ? \"true\" : \"false\"))}\n renderValue={(value: any) => value === \"own\" ? \"Own\" : (value === \"true\" ? \"Yes\" : \"No\")}\n >\n <SelectItem\n value={\"true\"}> Yes </SelectItem>\n <SelectItem\n value={\"false\"}> No </SelectItem>\n <SelectItem\n value={\"own\"}> Only\n his/her own </SelectItem>\n </Select>\n\n <FieldCaption>\n {touched.config && Boolean(errors.config) ? errors.config : \"Can the user edit collections\"}\n </FieldCaption>\n </div>\n\n <div className={\"col-span-12 md:col-span-4\"}>\n <Select\n error={touched.config && Boolean(errors.config)}\n id=\"deleteCollections\"\n name=\"deleteCollections\"\n label=\"Delete collections\"\n disabled={isAdmin || !editable}\n position={\"item-aligned\"}\n onChange={(event) => setFieldValue(\"config.deleteCollections\", event.target.value === \"own\" ? \"own\" : event.target.value === \"true\")}\n value={isAdmin ? \"true\" : (values.config?.deleteCollections === \"own\" ? \"own\" : (values.config?.deleteCollections ? \"true\" : \"false\"))}\n renderValue={(value: any) => value === \"own\" ? \"Own\" : (value === \"true\" ? \"Yes\" : \"No\")}\n >\n <SelectItem\n value={\"true\"}> Yes </SelectItem>\n <SelectItem\n value={\"false\"}> No </SelectItem>\n <SelectItem\n value={\"own\"}> Only\n his/her own </SelectItem>\n </Select>\n\n <FieldCaption>\n {touched.config && Boolean(errors.config) ? errors.config : \"Can the user delete collections\"}\n </FieldCaption>\n\n </div>\n\n </div>\n </DialogContent>\n\n <DialogActions position={\"sticky\"}>\n {savingError && <Typography className={\"text-red-500 dark:text-red-500\"}>\n {savingError.message ?? \"There was an error saving this role\"}\n </Typography>}\n <Button variant={\"text\"}\n onClick={() => {\n handleClose();\n }}>\n Cancel\n </Button>\n <LoadingButton\n variant=\"filled\"\n color=\"primary\"\n type=\"submit\"\n disabled={!dirty}\n loading={isSubmitting}\n startIcon={<DoneIcon/>}\n >\n {isNewRole ? \"Create role\" : \"Update\"}\n </LoadingButton>\n </DialogActions>\n </form>\n\n </Formex>\n </Dialog>\n );\n}\n","import { Role } from \"@firecms/core\";\n\nexport const DEFAULT_ROLES: Role[] = [\n {\n id: \"admin\",\n name: \"Admin\",\n isAdmin: true\n },\n {\n id: \"editor\",\n name: \"Editor\",\n isAdmin: false,\n defaultPermissions: {\n read: true,\n create: true,\n edit: true,\n delete: true\n },\n config: {\n createCollections: true,\n editCollections: \"own\",\n deleteCollections: \"own\"\n }\n },\n {\n id: \"viewer\",\n name: \"Viewer\",\n isAdmin: false,\n defaultPermissions: {\n read: true,\n create: false,\n edit: false,\n delete: false\n }\n }\n];\n","import { useState } from \"react\";\nimport {\n Button,\n CenteredView,\n Checkbox,\n DeleteIcon,\n IconButton,\n Table,\n TableBody,\n TableCell,\n TableHeader,\n TableRow,\n Tooltip,\n Typography\n} from \"@firecms/ui\";\nimport { ConfirmationDialog, Role } from \"@firecms/core\";\nimport { useUserManagement } from \"../../hooks\";\nimport { RoleChip } from \"./RoleChip\";\nimport { DEFAULT_ROLES } from \"./default_roles\";\n\nexport function RolesTable({\n onRoleClicked,\n editable\n }: {\n onRoleClicked: (role: Role) => void;\n editable: boolean;\n}) {\n\n const {\n roles,\n saveRole,\n deleteRole,\n allowDefaultRolesCreation\n } = useUserManagement();\n\n const [roleToBeDeleted, setRoleToBeDeleted] = useState<Role | undefined>(undefined);\n const [deleteInProgress, setDeleteInProgress] = useState<boolean>(false);\n\n return <div\n className=\"w-full overflow-auto\">\n <Table className={\"w-full\"}>\n <TableHeader>\n <TableCell header={true} className=\"w-16\"></TableCell>\n <TableCell header={true}>Role</TableCell>\n <TableCell header={true} className={\"items-center\"}>Is Admin</TableCell>\n <TableCell header={true}>Default permissions</TableCell>\n </TableHeader>\n\n <TableBody>\n {roles && roles.map((role) => {\n const canCreateAll = role.isAdmin || role.defaultPermissions?.create;\n const canReadAll = role.isAdmin || role.defaultPermissions?.read;\n const canUpdateAll = role.isAdmin || role.defaultPermissions?.edit;\n const canDeleteAll = role.isAdmin || role.defaultPermissions?.delete;\n return (\n <TableRow\n key={role.name}\n onClick={() => {\n onRoleClicked(role);\n }}\n >\n <TableCell style={{ width: \"64px\" }}>\n {!role.isAdmin &&\n <Tooltip\n asChild={true}\n title={\"Delete this role\"}>\n <IconButton\n size={\"small\"}\n disabled={!editable}\n onClick={(event) => {\n event.stopPropagation();\n return setRoleToBeDeleted(role);\n }}>\n <DeleteIcon/>\n </IconButton>\n </Tooltip>}\n </TableCell>\n <TableCell>\n <RoleChip role={role}/>\n </TableCell>\n <TableCell className={\"items-center\"}>\n <Checkbox checked={role.isAdmin ?? false}/>\n </TableCell>\n <TableCell>\n <ul>\n {canCreateAll && <li>Create</li>}\n {canReadAll && <li>Read</li>}\n {canUpdateAll && <li>Update</li>}\n {canDeleteAll && <li>Delete</li>}\n </ul>\n </TableCell>\n </TableRow>\n );\n })}\n\n {(!roles || roles.length === 0) && <TableRow>\n <TableCell colspan={4}>\n <CenteredView className={\"flex flex-col gap-4 my-8 items-center\"}>\n <Typography variant={\"label\"}>\n You don't have any roles yet.\n </Typography>\n {allowDefaultRolesCreation && <Button variant={\"outlined\"}\n onClick={() => {\n DEFAULT_ROLES.forEach((role) => {\n saveRole(role);\n });\n }}>\n Create default roles\n </Button>}\n </CenteredView>\n </TableCell>\n </TableRow>}\n\n </TableBody>\n\n </Table>\n\n <ConfirmationDialog\n open={Boolean(roleToBeDeleted)}\n loading={deleteInProgress}\n onAccept={() => {\n if (roleToBeDeleted) {\n setDeleteInProgress(true);\n deleteRole(roleToBeDeleted)\n .then(() => {\n setRoleToBeDeleted(undefined);\n })\n .finally(() => {\n setDeleteInProgress(false);\n })\n }\n }}\n onCancel={() => {\n setRoleToBeDeleted(undefined);\n }}\n title={<>Delete?</>}\n body={<>Are you sure you want to delete this role?</>}/>\n\n </div>;\n}\n","import React, { useCallback, useState } from \"react\";\n\nimport { Role, useNavigationController } from \"@firecms/core\";\nimport { AddIcon, Button, Container, Tooltip, Typography } from \"@firecms/ui\";\nimport { RolesTable } from \"./RolesTable\";\nimport { RolesDetailsForm } from \"./RolesDetailsForm\";\nimport { useUserManagement } from \"../../hooks\";\n\nexport const RolesView = React.memo(\n function RolesView({ children }: { children?: React.ReactNode }) {\n\n const { collections } = useNavigationController();\n const [dialogOpen, setDialogOpen] = useState(false);\n const [selectedRole, setSelectedRole] = useState<Role | undefined>();\n\n const { canEditRoles } = useUserManagement();\n\n const onRoleClicked = useCallback((user: Role) => {\n setDialogOpen(true);\n setSelectedRole(user);\n }, []);\n\n const handleClose = () => {\n setSelectedRole(undefined);\n setDialogOpen(false);\n };\n\n return (\n <Container className=\"w-full flex flex-col py-4 gap-4\" maxWidth={\"6xl\"}>\n\n {children}\n\n <div className=\"flex items-center mt-12\">\n <Typography gutterBottom variant=\"h4\"\n className=\"flex-grow\"\n component=\"h4\">\n Roles\n </Typography>\n <Tooltip\n asChild={true}\n title={!canEditRoles ? \"Update plans to customise roles\" : undefined}>\n <Button\n size={\"large\"}\n disabled={!canEditRoles}\n startIcon={<AddIcon/>}\n onClick={() => setDialogOpen(true)}>\n Add role\n </Button>\n </Tooltip>\n </div>\n\n <RolesTable onRoleClicked={onRoleClicked} editable={Boolean(canEditRoles)}/>\n\n <RolesDetailsForm\n key={selectedRole?.id ?? \"new\"}\n open={dialogOpen}\n role={selectedRole}\n editable={canEditRoles}\n collections={collections}\n handleClose={handleClose}/>\n\n </Container>\n )\n });\n","import React, { useCallback } from \"react\";\nimport * as Yup from \"yup\";\nimport {\n Button,\n Dialog,\n DialogActions,\n DialogContent,\n DoneIcon,\n LoadingButton,\n MultiSelect,\n MultiSelectItem,\n TextField,\n Typography,\n} from \"@firecms/ui\";\nimport { FieldCaption, Role, useAuthController, User, useSnackbarController } from \"@firecms/core\";\nimport { Formex, useCreateFormex } from \"@firecms/formex\";\n\nimport { areRolesEqual } from \"../../utils\";\nimport { useUserManagement } from \"../../hooks\";\nimport { RoleChip } from \"../roles\";\n\nexport const UserYupSchema = Yup.object().shape({\n displayName: Yup.string().required(\"Required\"),\n email: Yup.string().email().required(\"Required\"),\n roles: Yup.array().min(1)\n});\n\nfunction canUserBeEdited(loggedUser: User, user: User, users: User[], roles: Role[], prevUser?: User) {\n const admins = users.filter(u => u.roles?.map(r => r.id).includes(\"admin\"));\n const loggedUserIsAdmin = loggedUser.roles?.map(r => r.id).includes(\"admin\");\n const didRolesChange = !prevUser || !areRolesEqual(prevUser.roles ?? [], user.roles ?? []);\n\n if (didRolesChange && !loggedUserIsAdmin) {\n throw new Error(\"Only admins can change roles\");\n }\n\n // was the admin role removed\n const adminRoleRemoved = prevUser && prevUser.roles?.map(r => r.id).includes(\"admin\") && !user.roles?.map(r => r.id).includes(\"admin\");\n\n // avoid removing the last admin\n if (adminRoleRemoved && admins.length === 1) {\n throw new Error(\"There must be at least one admin\");\n }\n return true;\n}\n\nexport function UserDetailsForm({\n open,\n user: userProp,\n handleClose\n }: {\n open: boolean,\n user?: User,\n handleClose: () => void\n}) {\n\n const snackbarController = useSnackbarController();\n const {\n user: loggedInUser\n } = useAuthController();\n const {\n saveUser,\n users,\n roles,\n } = useUserManagement();\n const isNewUser = !userProp;\n\n const onUserUpdated = useCallback((savedUser: User): Promise<User> => {\n if (!loggedInUser) {\n throw new Error(\"Logged user not found\");\n }\n try {\n canUserBeEdited(loggedInUser, savedUser, users, roles, userProp);\n return saveUser(savedUser);\n } catch (e: any) {\n return Promise.reject(e);\n }\n }, [roles, saveUser, userProp, users, loggedInUser]);\n\n const formex = useCreateFormex({\n initialValues: userProp ?? {\n displayName: \"\",\n email: \"\",\n roles: roles.filter(r => r.id === \"editor\")\n } as User,\n validation: (values) => {\n return UserYupSchema.validate(values, { abortEarly: false })\n .then(() => {\n return {};\n }).catch((e) => {\n return e.inner.reduce((acc: any, error: any) => {\n acc[error.path] = error.message;\n return acc;\n }, {});\n });\n },\n onSubmit: (user: User, formexController) => {\n\n return onUserUpdated(user)\n .then(() => {\n handleClose();\n formexController.resetForm({\n values: user\n });\n }).catch((e) => {\n snackbarController.open({\n type: \"error\",\n message: e.message\n });\n });\n }\n });\n\n const {\n isSubmitting,\n touched,\n handleChange,\n values,\n errors,\n setFieldValue,\n dirty,\n handleSubmit,\n submitCount\n } = formex;\n\n return (\n <Dialog\n open={open}\n onOpenChange={(open) => !open ? handleClose() : undefined}\n maxWidth={\"4xl\"}\n >\n <Formex value={formex}>\n <form\n onSubmit={handleSubmit}\n autoComplete={\"off\"}\n noValidate\n style={{\n display: \"flex\",\n flexDirection: \"column\",\n position: \"relative\",\n height: \"100%\"\n }}>\n <DialogContent className=\"h-full flex-grow\">\n <div\n className=\"flex flex-row pt-4 pb-4\">\n <Typography variant={\"h4\"}\n className=\"flex-grow\">\n User\n </Typography>\n </div>\n\n <div className={\"grid grid-cols-12 gap-8\"}>\n\n <div className={\"col-span-12\"}>\n <TextField\n name=\"displayName\"\n required\n error={submitCount > 0 && Boolean(errors.displayName)}\n value={values.displayName ?? \"\"}\n onChange={handleChange}\n aria-describedby=\"name-helper-text\"\n label=\"Name\"\n />\n <FieldCaption>\n {submitCount > 0 && Boolean(errors.displayName) ? errors.displayName : \"Name of this user\"}\n </FieldCaption>\n </div>\n <div className={\"col-span-12\"}>\n <TextField\n required\n error={submitCount > 0 && Boolean(errors.email)}\n name=\"email\"\n value={values.email ?? \"\"}\n onChange={handleChange}\n aria-describedby=\"email-helper-text\"\n label=\"Email\"\n />\n <FieldCaption>\n {submitCount > 0 && Boolean(errors.email) ? errors.email : \"Email of this user\"}\n </FieldCaption>\n </div>\n <div className={\"col-span-12\"}>\n <MultiSelect\n className={\"w-full\"}\n label=\"Roles\"\n value={values.roles?.map(r => r.id) ?? []}\n onValueChange={(value: string[]) => setFieldValue(\"roles\", value.map(id => roles.find(r => r.id === id) as Role))}\n // renderValue={(value: string) => {\n // const userRole = roles\n // .find((role) => role.id === value);\n // if (!userRole) return null;\n // return <div className=\"flex flex-wrap space-x-2 space-y-2\">\n // <RoleChip key={userRole?.id} role={userRole}/>\n // </div>;\n // }}\n >\n {roles.map(userRole => <MultiSelectItem key={userRole.id}\n value={userRole.id}>\n <RoleChip key={userRole?.id} role={userRole}/>\n </MultiSelectItem>)}\n </MultiSelect>\n </div>\n\n </div>\n\n </DialogContent>\n\n <DialogActions>\n\n <Button variant={\"text\"}\n onClick={() => {\n handleClose();\n }}>\n Cancel\n </Button>\n\n <LoadingButton\n variant=\"filled\"\n color=\"primary\"\n type=\"submit\"\n disabled={!dirty}\n loading={isSubmitting}\n startIcon={<DoneIcon/>}\n >\n {isNewUser ? \"Create user\" : \"Update\"}\n </LoadingButton>\n </DialogActions>\n </form>\n </Formex>\n\n </Dialog>\n );\n}\n","import { useState } from \"react\";\n\nimport { format } from \"date-fns\";\nimport * as locales from \"date-fns/locale\";\n\nimport {\n defaultDateFormat,\n ConfirmationDialog, Role,\n useAuthController,\n useCustomizationController, User,\n useSnackbarController\n} from \"@firecms/core\";\nimport {\n Button,\n CenteredView,\n DeleteIcon,\n IconButton,\n Table,\n TableBody,\n TableCell,\n TableHeader,\n TableRow,\n Tooltip,\n Typography,\n} from \"@firecms/ui\";\nimport { useUserManagement } from \"../../hooks\";\nimport { RoleChip } from \"../roles\";\nimport { PersistedUser } from \"../../types\";\n\nexport function UsersTable({ onUserClicked }: {\n onUserClicked: (user: User) => void;\n}) {\n\n const {\n users,\n saveUser,\n deleteUser\n } = useUserManagement<PersistedUser>();\n\n const authController = useAuthController();\n const snackbarController = useSnackbarController();\n\n const customizationController = useCustomizationController();\n const dateUtilsLocale = customizationController?.locale ? locales[customizationController?.locale as keyof typeof locales] : undefined;\n const dateFormat: string = customizationController?.dateTimeFormat ?? defaultDateFormat;\n\n const [userToBeDeleted, setUserToBeDeleted] = useState<User | undefined>(undefined);\n const [deleteInProgress, setDeleteInProgress] = useState<boolean>(false);\n\n return (\n <div className=\"overflow-auto\">\n\n <Table className={\"w-full\"}>\n\n <TableHeader>\n <TableCell className=\"truncate w-16\"></TableCell>\n <TableCell>ID</TableCell>\n <TableCell>Email</TableCell>\n <TableCell>Name</TableCell>\n <TableCell>Roles</TableCell>\n <TableCell>Created on</TableCell>\n </TableHeader>\n <TableBody>\n {users && users.map((user) => {\n\n const userRoles: Role[] | undefined = user.roles;\n\n const formattedDate = user.created_on ? format(user.created_on, dateFormat, { locale: dateUtilsLocale }) : \"\";\n\n return (\n <TableRow\n key={\"row_\" + user.uid}\n onClick={() => {\n onUserClicked(user);\n }}\n >\n <TableCell className={\"w-10\"}>\n <Tooltip\n asChild={true}\n title={\"Delete this user\"}>\n <IconButton\n size={\"small\"}\n onClick={(event) => {\n event.stopPropagation();\n return setUserToBeDeleted(user);\n }}>\n <DeleteIcon/>\n </IconButton>\n </Tooltip>\n </TableCell>\n <TableCell>{user.uid}</TableCell>\n <TableCell>{user.email}</TableCell>\n <TableCell className={\"font-medium align-left\"}>{user.displayName}</TableCell>\n <TableCell className=\"align-left\">\n {userRoles\n ? <div className=\"flex flex-wrap gap-2\">\n {userRoles.map(userRole =>\n <RoleChip key={userRole?.id} role={userRole}/>\n )}\n </div>\n : null}\n </TableCell>\n <TableCell>{formattedDate}</TableCell>\n </TableRow>\n );\n })}\n\n {(!users || users.length === 0) && <TableRow>\n <TableCell colspan={6}>\n <CenteredView className={\"flex flex-col gap-4 my-8 items-center\"}>\n <Typography variant={\"label\"}>\n There are no users yet\n </Typography>\n <Button variant={\"outlined\"}\n onClick={() => {\n if (!authController.user?.uid) {\n throw Error(\"UsersTable, authController misconfiguration\");\n }\n saveUser({\n uid: authController.user?.uid,\n email: authController.user?.email,\n displayName: authController.user?.displayName,\n photoURL: authController.user?.photoURL,\n providerId: authController.user?.providerId,\n isAnonymous: authController.user?.isAnonymous,\n roles: [{ id: \"admin\", name: \"Admin\" }],\n created_on: new Date()\n })\n .then(() => {\n snackbarController.open({\n type: \"success\",\n message: \"User added successfully\"\n })\n })\n .catch((error) => {\n snackbarController.open({\n type: \"error\",\n message: \"Error adding user: \" + error.message,\n })\n });\n }}>\n\n Add the logged user as an admin\n </Button>\n </CenteredView>\n </TableCell>\n </TableRow>}\n\n </TableBody>\n </Table>\n\n <ConfirmationDialog\n open={Boolean(userToBeDeleted)}\n loading={deleteInProgress}\n onAccept={() => {\n if (userToBeDeleted) {\n setDeleteInProgress(true);\n deleteUser(userToBeDeleted)\n .then(() => {\n setUserToBeDeleted(undefined);\n })\n .catch((error) => {\n snackbarController.open({\n type: \"error\",\n message: \"Error deleting user: \" + error.message,\n })\n })\n .finally(() => {\n setDeleteInProgress(false);\n })\n }\n }}\n onCancel={() => {\n setUserToBeDeleted(undefined);\n }}\n title={<>Delete?</>}\n body={<>Are you sure you want to delete this user?</>}/>\n </div>);\n}\n","import { AddIcon, Button, Container, Typography } from \"@firecms/ui\";\n\nimport { UsersTable } from \"./UsersTable\";\nimport { UserDetailsForm } from \"./UserDetailsForm\";\nimport React, { useCallback, useState } from \"react\";\nimport { useUserManagement } from \"../../hooks/useUserManagement\";\nimport { User } from \"@firecms/core\";\n\nexport const UsersView = function UsersView({ children }: { children?: React.ReactNode }) {\n\n const [dialogOpen, setDialogOpen] = useState<boolean>();\n const [selectedUser, setSelectedUser] = useState<User | undefined>();\n\n const { users, usersLimit } = useUserManagement();\n\n const reachedUsersLimit = usersLimit !== undefined && (users && users.length >= usersLimit);\n\n const onUserClicked = useCallback((user: User) => {\n setSelectedUser(user);\n setDialogOpen(true);\n }, []);\n\n const handleClose = useCallback(() => {\n setDialogOpen(false);\n setSelectedUser(undefined);\n }, []);\n\n return (\n <Container className=\"w-full flex flex-col py-4 gap-4\" maxWidth={\"6xl\"}>\n\n {children}\n\n <div\n className=\"flex items-center mt-12\">\n <Typography gutterBottom variant=\"h4\"\n className=\"flex-grow\"\n component=\"h4\">\n Users\n </Typography>\n <Button\n size={\"large\"}\n disabled={reachedUsersLimit}\n startIcon={<AddIcon/>}\n onClick={() => setDialogOpen(true)}>\n Add user\n </Button>\n </div>\n\n <UsersTable onUserClicked={onUserClicked}/>\n\n <UserDetailsForm\n key={selectedUser?.uid ?? \"new\"}\n open={dialogOpen ?? false}\n user={selectedUser}\n handleClose={handleClose}/>\n\n </Container>\n )\n};\n","import { FireCMSPlugin, useAuthController, useSnackbarController } from \"@firecms/core\";\nimport { UserManagementProvider } from \"./UserManagementProvider\";\nimport { PersistedUser, UserManagement } from \"./types\";\nimport { AddIcon, Button, Paper, Typography } from \"@firecms/ui\";\nimport { DEFAULT_ROLES } from \"./components/roles/default_roles\";\n\nexport function useUserManagementPlugin({ userManagement }: {\n userManagement: UserManagement,\n}): FireCMSPlugin {\n\n const noUsers = userManagement.users.length === 0;\n const noRoles = userManagement.roles.length === 0;\n\n return {\n key: \"user_management\",\n loading: userManagement.loading,\n\n homePage: {\n additionalChildrenStart: noUsers || noRoles\n ? <IntroWidget\n noUsers={noUsers}\n noRoles={noRoles}\n userManagement={userManagement}/>\n : undefined\n },\n provider: {\n Component: UserManagementProvider,\n props: {\n userManagement\n }\n }\n }\n}\n\nexport function IntroWidget({\n noUsers,\n noRoles,\n userManagement\n }: {\n noUsers: boolean;\n noRoles: boolean;\n userManagement: UserManagement<PersistedUser>;\n}) {\n\n const authController = useAuthController();\n const snackbarController = useSnackbarController();\n\n const buttonLabel = noUsers && noRoles\n ? \"Create default roles and add current user as admin\"\n : noUsers\n ? \"Add current user as admin\"\n : noRoles ? \"Create default roles\" : undefined;\n\n return (\n <Paper\n className={\"my-4 flex flex-col px-4 py-6 bg-white dark:bg-slate-800 gap-2\"}>\n <Typography variant={\"subtitle2\"} className={\"uppercase\"}>Create your users and roles</Typography>\n <Typography>\n You have no users or roles defined. You can create default roles and add the current user as admin.\n </Typography>\n <Button onClick={() => {\n if (!authController.user?.uid) {\n throw Error(\"UsersTable, authController misconfiguration\");\n }\n if (noUsers) {\n userManagement.saveUser({\n uid: authController.user?.uid,\n email: authController.user?.email,\n displayName: authController.user?.displayName,\n photoURL: authController.user?.photoURL,\n providerId: authController.user?.providerId,\n isAnonymous: authController.user?.isAnonymous,\n roles: [{\n id: \"admin\",\n name: \"Admin\"\n }],\n created_on: new Date()\n })\n .then(() => {\n snackbarController.open({\n type: \"success\",\n message: \"User added successfully\"\n })\n })\n .catch((error) => {\n snackbarController.open({\n type: \"error\",\n message: \"Error adding user: \" + error.message\n })\n });\n }\n if (noRoles) {\n DEFAULT_ROLES.forEach((role) => {\n userManagement.saveRole(role);\n });\n }\n }}>\n <AddIcon/>\n {buttonLabel}\n </Button>\n </Paper>\n );\n\n}\n","import { CMSView } from \"@firecms/core\";\nimport { RolesView, UsersView } from \"./components\";\n\nexport const userManagementAdminViews: CMSView[] = [\n {\n path: \"users\",\n name: \"CMS Users\",\n group: \"Admin\",\n icon: \"face\",\n view: <UsersView/>\n },\n {\n path: \"roles\",\n name: \"Roles\",\n group: \"Admin\",\n icon: \"gpp_good\",\n view: <RolesView/>\n }\n]\n"],"names":["useEffect","useCallback","removeUndefined","useContext","getColorSchemeForSeed","jsx","Chip","Yup","useAuthController","useState","role","formex","useCreateFormex","values","errors","getIn","toSnakeCase","Dialog","Formex","jsxs","DialogContent","Typography","TextField","FieldCaption","Paper","Table","TableHeader","TableCell","TableBody","TableRow","Tooltip","Checkbox","Button","Select","SelectItem","DialogActions","LoadingButton","DoneIcon","IconButton","DeleteIcon","CenteredView","ConfirmationDialog","RolesView","useNavigationController","Container","AddIcon","useSnackbarController","open","MultiSelect","MultiSelectItem","useCustomizationController","locales","defaultDateFormat","format","UsersView"],"mappings":"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"}
|
@@ -1,4 +1,4 @@
|
|
1
|
-
import { PermissionsBuilder, Role, User } from "@firecms/core";
|
1
|
+
import { Authenticator, PermissionsBuilder, Role, User } from "@firecms/core";
|
2
2
|
export type UserManagement<USER extends User = User> = {
|
3
3
|
loading: boolean;
|
4
4
|
users: USER[];
|
@@ -15,6 +15,10 @@ export type UserManagement<USER extends User = User> = {
|
|
15
15
|
* Can the logged user edit roles?
|
16
16
|
*/
|
17
17
|
canEditRoles?: boolean;
|
18
|
+
/**
|
19
|
+
* Is the logged user Admin?
|
20
|
+
*/
|
21
|
+
isAdmin?: boolean;
|
18
22
|
/**
|
19
23
|
* Include a button to create default roles, in case there are no roles in the system.
|
20
24
|
*/
|
@@ -29,8 +33,15 @@ export type UserManagement<USER extends User = User> = {
|
|
29
33
|
*/
|
30
34
|
collectionPermissions: PermissionsBuilder;
|
31
35
|
/**
|
32
|
-
* Define the roles for a given user.
|
36
|
+
* Define the roles for a given user. You will typically want to plug this into your auth controller.
|
33
37
|
* @param user
|
34
38
|
*/
|
35
|
-
defineRolesFor: (user: User) => Promise<Role[]> | Role[] | undefined;
|
39
|
+
defineRolesFor: (user: User) => Promise<Role[] | undefined> | Role[] | undefined;
|
40
|
+
/**
|
41
|
+
* You can build an authenticator callback from the current configuration of the user management.
|
42
|
+
* It will only allow access to users with the required roles.
|
43
|
+
*/
|
44
|
+
authenticator?: Authenticator;
|
45
|
+
rolesError?: Error;
|
46
|
+
usersError?: Error;
|
36
47
|
};
|
@@ -1,5 +1,10 @@
|
|
1
1
|
import { FireCMSPlugin } from "@firecms/core";
|
2
|
-
import { UserManagement } from "./types";
|
2
|
+
import { PersistedUser, UserManagement } from "./types";
|
3
3
|
export declare function useUserManagementPlugin({ userManagement }: {
|
4
4
|
userManagement: UserManagement;
|
5
5
|
}): FireCMSPlugin;
|
6
|
+
export declare function IntroWidget({ noUsers, noRoles, userManagement }: {
|
7
|
+
noUsers: boolean;
|
8
|
+
noRoles: boolean;
|
9
|
+
userManagement: UserManagement<PersistedUser>;
|
10
|
+
}): import("react/jsx-runtime").JSX.Element;
|