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} 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 {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"; const columnHelper = createColumnHelper(); const searchFields = [ ['name'], ['email'], ['corporateInformation', 'companyInformation', 'name'], ]; export default function UserList({user, filters = []}: {user: User; filters?: ((user: User) => boolean)[]}) { const [showDemographicInformation, setShowDemographicInformation] = useState(false); const [sorter, setSorter] = useState(); const [displayUsers, setDisplayUsers] = useState([]); const [selectedUser, setSelectedUser] = useState(); const {users, reload} = useUsers(); const {groups} = useGroups(user && (user?.type === "corporate" || user?.type === "teacher") ? user.id : undefined); 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(() => { if (user && users) { const filterUsers = user.type === "corporate" || user.type === "teacher" ? users.filter((u) => groups.flatMap((g) => g.participants).includes(u.id)) : users; const filteredUsers = filters.reduce((d, f) => d.filter(f), filterUsers); setDisplayUsers([...filteredUsers.sort(sortFunction)]); } // eslint-disable-next-line react-hooks/exhaustive-deps }, [user, users, sorter, groups]); 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"}); }); }; const updateAccountType = (user: User, type: Type) => { if (!confirm(`Are you sure you want to update ${user.name}'s account from ${capitalize(user.type)} to ${capitalize(type)}?`)) return; axios .post<{user?: User; ok?: boolean}>(`/api/users/update?id=${user.id}`, {...user, type}) .then(() => { toast.success("User type updated successfully!"); reload(); }) .catch(() => { toast.error("Something went wrong!", {toastId: "update-error"}); }); }; 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}}) => { return (
{PERMISSIONS.updateUser[row.original.type].includes(user.type) && (
)} {!row.original.isVerified && PERMISSIONS.updateUser[row.original.type].includes(user.type) && (
verifyAccount(row.original)}>
)} {PERMISSIONS.updateUser[row.original.type].includes(user.type) && (
toggleDisableAccount(row.original)}> {row.original.status === "disabled" ? ( ) : ( )}
)} {PERMISSIONS.deleteUser[row.original.type].includes(user.type) && (
deleteAccount(row.original)}>
)}
); }; const demographicColumns = [ columnHelper.accessor("name", { header: ( ) as any, cell: ({row, getValue}) => (
(PERMISSIONS.updateExpiryDate[row.original.type].includes(user.type) ? 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})` : "Not available", }), columnHelper.accessor("demographicInformation.phone", { header: ( ) as any, cell: (info) => info.getValue() || "Not available", enableSorting: true, }), columnHelper.accessor((x) => (x.type === "corporate" ? x.demographicInformation?.position : x.demographicInformation?.employment), { id: "employment", header: ( ) as any, cell: (info) => (info.row.original.type === "corporate" ? info.getValue() : capitalize(info.getValue())) || "Not available", enableSorting: true, }), columnHelper.accessor("demographicInformation.gender", { header: ( ) as any, cell: (info) => capitalize(info.getValue()) || "Not available", enableSorting: true, }), { header: ( setShowDemographicInformation((prev) => !prev)}> Switch ), id: "actions", cell: actionColumn, }, ]; const defaultColumns = [ columnHelper.accessor("name", { header: ( ) as any, cell: ({row, getValue}) => (
(PERMISSIONS.updateExpiryDate[row.original.type].includes(user.type) ? setSelectedUser(row.original) : null)}> {row.original.type === "corporate" ? row.original.corporateInformation?.companyInformation?.name || getValue() : 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('corporateInformation.companyInformation.name', { header: ( ) as any, cell: (info) => getCorporateName(info.row.original), }), 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 getCorporateName = (user: User) => { if(isCorporateUser(user)) { return user.corporateInformation?.companyInformation?.name } return ''; } const sortFunction = (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 === "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 === "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.demographicInformation?.position : a.demographicInformation?.employment; const bSortingItem = b.type === "corporate" ? 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 = getCorporateName(a); const bCorporateName = getCorporateName(b); 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(), }); return (
setSelectedUser(undefined)}> <> {selectedUser && (
{ appendUserFilters({ id: "view-students", filter: (x: User) => x.type === "student", }); appendUserFilters({ id: "belongs-to-admin", filter: (x: User) => groups .filter((g) => g.admin === selectedUser.id || g.participants.includes(selectedUser.id)) .flatMap((g) => g.participants) .includes(x.id), }); router.push("/list/users"); } : undefined } onViewTeachers={ selectedUser.type === "corporate" || selectedUser.type === "student" ? () => { appendUserFilters({ id: "view-teachers", filter: (x: User) => x.type === "teacher", }); appendUserFilters({ id: "belongs-to-admin", filter: (x: User) => groups .filter((g) => g.admin === selectedUser.id || g.participants.includes(selectedUser.id)) .flatMap((g) => g.participants) .includes(x.id), }); 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} />
)}
{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())}
); }