import Button from "@/components/Low/Button"; import {PERMISSIONS} from "@/constants/userPermissions"; import useGroups from "@/hooks/useGroups"; import useUsers from "@/hooks/useUsers"; import {Type, User, userTypes, CorporateUser, Group} from "@/interfaces/user"; import {Popover, Transition} from "@headlessui/react"; import {createColumnHelper, flexRender, getCoreRowModel, useReactTable} from "@tanstack/react-table"; import axios from "axios"; import clsx from "clsx"; import {capitalize, reverse} from "lodash"; import moment from "moment"; import {Fragment, useEffect, useState} from "react"; import {BsArrowDown, BsArrowDownUp, BsArrowUp, BsCheck, BsCheckCircle, BsEye, BsFillExclamationOctagonFill, BsPerson, BsTrash} from "react-icons/bs"; import {toast} from "react-toastify"; import {countries, TCountries} from "countries-list"; import countryCodes from "country-codes-list"; import Modal from "@/components/Modal"; import UserCard from "@/components/UserCard"; import {getUserCompanyName, isAgentUser, USER_TYPE_LABELS} from "@/resources/user"; import useFilterStore from "@/stores/listFilterStore"; import {useRouter} from "next/router"; import {isCorporateUser} from "@/resources/user"; import {useListSearch} from "@/hooks/useListSearch"; import {getUserCorporate} from "@/utils/groups"; import {asyncSorter} from "@/utils"; import {exportListToExcel, UserListRow} from "@/utils/users"; import {checkAccess} from "@/utils/permissions"; import {PermissionType} from "@/interfaces/permissions"; import usePermissions from "@/hooks/usePermissions"; import useUserBalance from "@/hooks/useUserBalance"; const columnHelper = createColumnHelper(); const searchFields = [["name"], ["email"], ["corporateInformation", "companyInformation", "name"]]; const CompanyNameCell = ({users, user, groups}: {user: User; users: User[]; groups: Group[]}) => { const [companyName, setCompanyName] = useState(""); const [isLoading, setIsLoading] = useState(false); useEffect(() => { const name = getUserCompanyName(user, users, groups); setCompanyName(name); }, [user, users, groups]); return isLoading ? Loading... : <>{companyName}; }; export default function UserList({ user, filters = [], renderHeader, }: { user: User; filters?: ((user: User) => boolean)[]; renderHeader?: (total: number) => JSX.Element; }) { const [showDemographicInformation, setShowDemographicInformation] = useState(false); const [sorter, setSorter] = useState(); const [displayUsers, setDisplayUsers] = useState([]); const [selectedUser, setSelectedUser] = useState(); const {users, reload} = useUsers(); const {permissions} = usePermissions(user?.id || ""); const {balance} = useUserBalance(); const {groups} = useGroups({ admin: user && ["corporate", "teacher", "mastercorporate"].includes(user?.type) ? user.id : undefined, userType: user?.type, }); const appendUserFilters = useFilterStore((state) => state.appendUserFilter); const router = useRouter(); const expirationDateColor = (date: Date) => { const momentDate = moment(date); const today = moment(new Date()); if (today.isAfter(momentDate)) return "!text-mti-red-light font-bold line-through"; if (today.add(1, "weeks").isAfter(momentDate)) return "!text-mti-red-light"; if (today.add(2, "weeks").isAfter(momentDate)) return "!text-mti-rose-light"; if (today.add(1, "months").isAfter(momentDate)) return "!text-mti-orange-light"; }; useEffect(() => { (async () => { if (users) { const filteredUsers = filters.reduce((d, f) => d.filter(f), users); const sortedUsers = await asyncSorter(filteredUsers, sortFunction); setDisplayUsers([...sortedUsers]); } })(); // eslint-disable-next-line react-hooks/exhaustive-deps }, [users, sorter]); const deleteAccount = (user: User) => { if (!confirm(`Are you sure you want to delete ${user.name}'s account?`)) return; axios .delete<{ok: boolean}>(`/api/user?id=${user.id}`) .then(() => { toast.success("User deleted successfully!"); reload(); }) .catch(() => { toast.error("Something went wrong!", {toastId: "delete-error"}); }) .finally(reload); }; const verifyAccount = (user: User) => { axios .post<{user?: User; ok?: boolean}>(`/api/users/update?id=${user.id}`, { ...user, isVerified: true, }) .then(() => { toast.success("User verified successfully!"); reload(); }) .catch(() => { toast.error("Something went wrong!", {toastId: "update-error"}); }); }; const toggleDisableAccount = (user: User) => { if ( !confirm( `Are you sure you want to ${user.status === "disabled" ? "enable" : "disable"} ${ user.name }'s account? This change is usually related to their payment state.`, ) ) return; axios .post<{user?: User; ok?: boolean}>(`/api/users/update?id=${user.id}`, { ...user, status: user.status === "disabled" ? "active" : "disabled", }) .then(() => { toast.success(`User ${user.status === "disabled" ? "enabled" : "disabled"} successfully!`); reload(); }) .catch(() => { toast.error("Something went wrong!", {toastId: "update-error"}); }); }; const SorterArrow = ({name}: {name: string}) => { if (sorter === name) return ; if (sorter === reverseString(name)) return ; return ; }; const actionColumn = ({row}: {row: {original: User}}) => { const updateUserPermission = PERMISSIONS.updateUser[row.original.type] as { list: Type[]; perm: PermissionType; }; const deleteUserPermission = PERMISSIONS.deleteUser[row.original.type] as { list: Type[]; perm: PermissionType; }; return (
{!row.original.isVerified && checkAccess(user, updateUserPermission.list, permissions, updateUserPermission.perm) && (
verifyAccount(row.original)}>
)} {checkAccess(user, updateUserPermission.list, permissions, updateUserPermission.perm) && (
toggleDisableAccount(row.original)}> {row.original.status === "disabled" ? ( ) : ( )}
)} {checkAccess(user, deleteUserPermission.list, permissions, deleteUserPermission.perm) && (
deleteAccount(row.original)}>
)}
); }; const demographicColumns = [ columnHelper.accessor("name", { header: ( ) as any, cell: ({row, getValue}) => (
checkAccess(user, ["admin", "corporate", "developer", "mastercorporate"]) ? setSelectedUser(row.original) : null }> {getValue()}
), }), columnHelper.accessor("demographicInformation.country", { header: ( ) as any, cell: (info) => info.getValue() ? `${countryCodes.findOne("countryCode" as any, info.getValue())?.flag} ${ countries[info.getValue() as unknown as keyof TCountries]?.name } (+${countryCodes.findOne("countryCode" as any, info.getValue())?.countryCallingCode})` : "N/A", }), columnHelper.accessor("demographicInformation.phone", { header: ( ) as any, cell: (info) => info.getValue() || "N/A", enableSorting: true, }), columnHelper.accessor( (x) => x.type === "corporate" || x.type === "mastercorporate" ? x.demographicInformation?.position : x.demographicInformation?.employment, { id: "employment", header: ( ) as any, cell: (info) => (info.row.original.type === "corporate" ? info.getValue() : capitalize(info.getValue())) || "N/A", enableSorting: true, }, ), columnHelper.accessor("lastLogin", { header: ( ) as any, cell: (info) => (!!info.getValue() ? moment(info.getValue()).format("YYYY-MM-DD HH:mm") : "N/A"), }), columnHelper.accessor("demographicInformation.gender", { header: ( ) as any, cell: (info) => capitalize(info.getValue()) || "N/A", enableSorting: true, }), { header: ( setShowDemographicInformation((prev) => !prev)}> Switch ), id: "actions", cell: actionColumn, }, ]; const defaultColumns = [ columnHelper.accessor("name", { header: ( ) as any, cell: ({row, getValue}) => (
checkAccess(user, ["admin", "corporate", "developer", "mastercorporate"]) ? setSelectedUser(row.original) : null }> {getValue()}
), }), columnHelper.accessor("email", { header: ( ) as any, cell: ({row, getValue}) => (
(PERMISSIONS.updateExpiryDate[row.original.type]?.includes(user.type) ? setSelectedUser(row.original) : null)}> {getValue()}
), }), columnHelper.accessor("type", { header: ( ) as any, cell: (info) => USER_TYPE_LABELS[info.getValue()], }), columnHelper.accessor("studentID", { header: ( ) as any, cell: (info) => info.getValue() || "N/A", }), columnHelper.accessor("corporateInformation.companyInformation.name", { header: ( ) as any, cell: (info) => , }), columnHelper.accessor("subscriptionExpirationDate", { header: ( ) as any, cell: (info) => ( {!info.getValue() ? "No expiry date" : moment(info.getValue()).format("DD/MM/YYYY")} ), }), columnHelper.accessor("isVerified", { header: ( ) as any, cell: (info) => (
), }), { header: ( setShowDemographicInformation((prev) => !prev)}> Switch ), id: "actions", cell: actionColumn, }, ]; const reverseString = (str: string) => reverse(str.split("")).join(""); const selectSorter = (previous: string | undefined, name: string) => { if (!previous) return name; if (previous === name) return reverseString(name); return undefined; }; const sortFunction = async (a: User, b: User) => { if (sorter === "name" || sorter === reverseString("name")) return sorter === "name" ? a.name.localeCompare(b.name) : b.name.localeCompare(a.name); if (sorter === "email" || sorter === reverseString("email")) return sorter === "email" ? a.email.localeCompare(b.email) : b.email.localeCompare(a.email); if (sorter === "type" || sorter === reverseString("type")) return sorter === "type" ? userTypes.findIndex((t) => a.type === t) - userTypes.findIndex((t) => b.type === t) : userTypes.findIndex((t) => b.type === t) - userTypes.findIndex((t) => a.type === t); if (sorter === "studentID" || sorter === reverseString("studentID")) return sorter === "studentID" ? (a.type === "student" ? a.studentID || "N/A" : "N/A").localeCompare(b.type === "student" ? b.studentID || "N/A" : "N/A") : (b.type === "student" ? b.studentID || "N/A" : "N/A").localeCompare(a.type === "student" ? a.studentID || "N/A" : "N/A"); if (sorter === "verification" || sorter === reverseString("verification")) return sorter === "verification" ? a.isVerified.toString().localeCompare(b.isVerified.toString()) : b.isVerified.toString().localeCompare(a.isVerified.toString()); if (sorter === "expiryDate" || sorter === reverseString("expiryDate")) { if (!a.subscriptionExpirationDate && b.subscriptionExpirationDate) return sorter === "expiryDate" ? -1 : 1; if (a.subscriptionExpirationDate && !b.subscriptionExpirationDate) return sorter === "expiryDate" ? 1 : -1; if (!a.subscriptionExpirationDate && !b.subscriptionExpirationDate) return 0; if (moment(a.subscriptionExpirationDate).isAfter(b.subscriptionExpirationDate)) return sorter === "expiryDate" ? -1 : 1; if (moment(b.subscriptionExpirationDate).isAfter(a.subscriptionExpirationDate)) return sorter === "expiryDate" ? 1 : -1; return 0; } if (sorter === "lastLogin" || sorter === reverseString("lastLogin")) { if (!a.lastLogin && b.lastLogin) return sorter === "lastLogin" ? -1 : 1; if (a.lastLogin && !b.lastLogin) return sorter === "lastLogin" ? 1 : -1; if (!a.lastLogin && !b.lastLogin) return 0; if (moment(a.lastLogin).isAfter(b.lastLogin)) return sorter === "lastLogin" ? -1 : 1; if (moment(b.lastLogin).isAfter(a.lastLogin)) return sorter === "lastLogin" ? 1 : -1; return 0; } if (sorter === "country" || sorter === reverseString("country")) { if (!a.demographicInformation?.country && b.demographicInformation?.country) return sorter === "country" ? -1 : 1; if (a.demographicInformation?.country && !b.demographicInformation?.country) return sorter === "country" ? 1 : -1; if (!a.demographicInformation?.country && !b.demographicInformation?.country) return 0; return sorter === "country" ? a.demographicInformation!.country.localeCompare(b.demographicInformation!.country) : b.demographicInformation!.country.localeCompare(a.demographicInformation!.country); } if (sorter === "phone" || sorter === reverseString("phone")) { if (!a.demographicInformation?.phone && b.demographicInformation?.phone) return sorter === "phone" ? -1 : 1; if (a.demographicInformation?.phone && !b.demographicInformation?.phone) return sorter === "phone" ? 1 : -1; if (!a.demographicInformation?.phone && !b.demographicInformation?.phone) return 0; return sorter === "phone" ? a.demographicInformation!.phone.localeCompare(b.demographicInformation!.phone) : b.demographicInformation!.phone.localeCompare(a.demographicInformation!.phone); } if (sorter === "employment" || sorter === reverseString("employment")) { const aSortingItem = a.type === "corporate" || a.type === "mastercorporate" ? a.demographicInformation?.position : a.demographicInformation?.employment; const bSortingItem = b.type === "corporate" || b.type === "mastercorporate" ? b.demographicInformation?.position : b.demographicInformation?.employment; if (!aSortingItem && bSortingItem) return sorter === "employment" ? -1 : 1; if (aSortingItem && !bSortingItem) return sorter === "employment" ? 1 : -1; if (!aSortingItem && !bSortingItem) return 0; return sorter === "employment" ? aSortingItem!.localeCompare(bSortingItem!) : bSortingItem!.localeCompare(aSortingItem!); } if (sorter === "gender" || sorter === reverseString("gender")) { if (!a.demographicInformation?.gender && b.demographicInformation?.gender) return sorter === "employment" ? -1 : 1; if (a.demographicInformation?.gender && !b.demographicInformation?.gender) return sorter === "employment" ? 1 : -1; if (!a.demographicInformation?.gender && !b.demographicInformation?.gender) return 0; return sorter === "gender" ? a.demographicInformation!.gender.localeCompare(b.demographicInformation!.gender) : b.demographicInformation!.gender.localeCompare(a.demographicInformation!.gender); } if (sorter === "companyName" || sorter === reverseString("companyName")) { const aCorporateName = getUserCompanyName(a, users, groups); const bCorporateName = getUserCompanyName(b, users, groups); if (!aCorporateName && bCorporateName) return sorter === "companyName" ? -1 : 1; if (aCorporateName && !bCorporateName) return sorter === "companyName" ? 1 : -1; if (!aCorporateName && !bCorporateName) return 0; return sorter === "companyName" ? aCorporateName.localeCompare(bCorporateName) : bCorporateName.localeCompare(aCorporateName); } return a.id.localeCompare(b.id); }; const {rows: filteredRows, renderSearch} = useListSearch(searchFields, displayUsers); const table = useReactTable({ data: filteredRows, columns: (!showDemographicInformation ? defaultColumns : demographicColumns) as any, getCoreRowModel: getCoreRowModel(), }); const downloadExcel = () => { const csv = exportListToExcel(filteredRows, users, groups); const element = document.createElement("a"); const file = new Blob([csv], {type: "text/csv"}); element.href = URL.createObjectURL(file); element.download = "users.csv"; document.body.appendChild(element); element.click(); document.body.removeChild(element); }; const viewStudentFilter = (x: User) => x.type === "student"; const viewTeacherFilter = (x: User) => x.type === "teacher"; const belongsToAdminFilter = (x: User) => { if (!selectedUser) return false; return groups .filter((g) => g.admin === selectedUser.id || g.participants.includes(selectedUser.id)) .flatMap((g) => g.participants) .includes(x.id); }; const viewStudentFilterBelongsToAdmin = (x: User) => x.type === "student" && belongsToAdminFilter(x); const viewTeacherFilterBelongsToAdmin = (x: User) => x.type === "teacher" && belongsToAdminFilter(x); const renderUserCard = (selectedUser: User) => { const studentsFromAdmin = users.filter(viewStudentFilterBelongsToAdmin); const teachersFromAdmin = users.filter(viewTeacherFilterBelongsToAdmin); return (
0 ? () => { appendUserFilters({ id: "view-students", filter: viewStudentFilter, }); appendUserFilters({ id: "belongs-to-admin", filter: belongsToAdminFilter, }); router.push("/list/users"); } : undefined } onViewTeachers={ (selectedUser.type === "corporate" || selectedUser.type === "student") && teachersFromAdmin.length > 0 ? () => { appendUserFilters({ id: "view-teachers", filter: viewTeacherFilter, }); appendUserFilters({ id: "belongs-to-admin", filter: belongsToAdminFilter, }); router.push("/list/users"); } : undefined } onViewCorporate={ selectedUser.type === "teacher" || selectedUser.type === "student" ? () => { appendUserFilters({ id: "view-corporate", filter: (x: User) => x.type === "corporate", }); appendUserFilters({ id: "belongs-to-admin", filter: (x: User) => groups .filter((g) => g.participants.includes(selectedUser.id)) .flatMap((g) => [g.admin, ...g.participants]) .includes(x.id), }); router.push("/list/users"); } : undefined } onClose={(shouldReload) => { setSelectedUser(undefined); if (shouldReload) reload(); }} user={selectedUser} />
); }; return ( <> {renderHeader && renderHeader(displayUsers.length)}
setSelectedUser(undefined)}> {selectedUser && renderUserCard(selectedUser)}
{renderSearch()}
{table.getHeaderGroups().map((headerGroup) => ( {headerGroup.headers.map((header) => ( ))} ))} {table.getRowModel().rows.map((row) => ( {row.getVisibleCells().map((cell) => ( ))} ))}
{header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext())}
{flexRender(cell.column.columnDef.cell, cell.getContext())}
); }