Files
sign/apps/remix/app/components/tables/admin-dashboard-users-table.tsx

164 lines
4.5 KiB
TypeScript
Raw Normal View History

import { useEffect, useMemo, useState, useTransition } from 'react';
2023-09-21 12:43:36 +01:00
2025-01-02 15:33:37 +11:00
import { msg } from '@lingui/core/macro';
2024-08-27 20:34:39 +09:00
import { useLingui } from '@lingui/react';
2025-01-02 15:33:37 +11:00
import type { Document, Role, Subscription } from '@prisma/client';
2023-09-21 12:43:36 +01:00
import { Edit, Loader } from 'lucide-react';
2025-01-02 15:33:37 +11:00
import { Link } from 'react-router';
2023-09-21 12:43:36 +01:00
2023-10-10 13:57:07 +03:00
import { useDebouncedValue } from '@documenso/lib/client-only/hooks/use-debounced-value';
2023-09-21 12:43:36 +01:00
import { useUpdateSearchParams } from '@documenso/lib/client-only/hooks/use-update-search-params';
import { Button } from '@documenso/ui/primitives/button';
import type { DataTableColumnDef } from '@documenso/ui/primitives/data-table';
2023-09-21 12:43:36 +01:00
import { DataTable } from '@documenso/ui/primitives/data-table';
import { DataTablePagination } from '@documenso/ui/primitives/data-table-pagination';
import { Input } from '@documenso/ui/primitives/input';
2023-09-21 12:43:36 +01:00
type UserData = {
2023-09-21 12:43:36 +01:00
id: number;
name: string | null;
email: string;
roles: Role[];
2025-01-13 13:41:53 +11:00
subscriptions?: SubscriptionLite[] | null;
documents: DocumentLite[];
};
2023-09-21 12:43:36 +01:00
2023-10-06 15:48:05 +03:00
type SubscriptionLite = Pick<
Subscription,
'id' | 'status' | 'planId' | 'priceId' | 'createdAt' | 'periodEnd'
>;
type DocumentLite = Pick<Document, 'id'>;
2023-09-21 12:43:36 +01:00
2025-01-02 15:33:37 +11:00
type AdminDashboardUsersTableProps = {
users: UserData[];
2023-10-11 16:20:04 +03:00
totalPages: number;
2023-09-21 12:43:36 +01:00
perPage: number;
page: number;
individualPriceIds: string[];
2023-09-21 12:43:36 +01:00
};
2025-01-02 15:33:37 +11:00
export const AdminDashboardUsersTable = ({
users,
totalPages,
perPage,
page,
individualPriceIds,
2025-01-02 15:33:37 +11:00
}: AdminDashboardUsersTableProps) => {
2024-08-27 20:34:39 +09:00
const { _ } = useLingui();
2023-09-21 12:43:36 +01:00
const [isPending, startTransition] = useTransition();
const updateSearchParams = useUpdateSearchParams();
const [searchString, setSearchString] = useState('');
2023-10-11 16:20:04 +03:00
const debouncedSearchString = useDebouncedValue(searchString, 1000);
2023-09-21 12:43:36 +01:00
const columns = useMemo(() => {
return [
{
header: 'ID',
accessorKey: 'id',
cell: ({ row }) => <div>{row.original.id}</div>,
},
{
header: _(msg`Name`),
accessorKey: 'name',
cell: ({ row }) => <div>{row.original.name}</div>,
},
{
header: _(msg`Email`),
accessorKey: 'email',
cell: ({ row }) => <div>{row.original.email}</div>,
},
{
header: _(msg`Roles`),
accessorKey: 'roles',
cell: ({ row }) => row.original.roles.join(', '),
},
{
header: _(msg`Subscription`),
accessorKey: 'subscription',
cell: ({ row }) => {
2025-01-13 13:41:53 +11:00
const foundIndividualSubscription = (row.original.subscriptions ?? []).find((sub) =>
individualPriceIds.includes(sub.priceId),
);
return foundIndividualSubscription?.status ?? 'NONE';
},
},
{
header: _(msg`Documents`),
accessorKey: 'documents',
cell: ({ row }) => {
2025-01-13 13:41:53 +11:00
return <div>{row.original.documents?.length}</div>;
},
},
{
header: '',
accessorKey: 'edit',
cell: ({ row }) => {
return (
<Button className="w-24" asChild>
2025-01-02 15:33:37 +11:00
<Link to={`/admin/users/${row.original.id}`}>
<Edit className="-ml-1 mr-2 h-4 w-4" />
Edit
</Link>
</Button>
);
},
},
] satisfies DataTableColumnDef<(typeof users)[number]>[];
}, [individualPriceIds]);
2023-10-11 16:20:04 +03:00
useEffect(() => {
2023-09-21 12:43:36 +01:00
startTransition(() => {
updateSearchParams({
2023-10-11 16:20:04 +03:00
search: debouncedSearchString,
page: 1,
2023-09-21 12:43:36 +01:00
perPage,
});
});
// eslint-disable-next-line react-hooks/exhaustive-deps
2023-10-11 16:20:04 +03:00
}, [debouncedSearchString]);
2023-10-11 16:20:04 +03:00
const onPaginationChange = (page: number, perPage: number) => {
startTransition(() => {
updateSearchParams({
page,
perPage,
});
});
2023-10-11 16:20:04 +03:00
};
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setSearchString(e.target.value);
};
2023-09-21 12:43:36 +01:00
return (
<div className="relative">
<Input
className="my-6 flex flex-row gap-4"
type="text"
2024-08-27 20:34:39 +09:00
placeholder={_(msg`Search by name or email`)}
value={searchString}
onChange={handleChange}
/>
2023-09-21 12:43:36 +01:00
<DataTable
columns={columns}
2023-10-11 16:20:04 +03:00
data={users}
2023-09-21 12:43:36 +01:00
perPage={perPage}
currentPage={page}
totalPages={totalPages}
onPaginationChange={onPaginationChange}
>
{(table) => <DataTablePagination additionalInformation="VisibleCount" table={table} />}
</DataTable>
{isPending && (
<div className="absolute inset-0 flex items-center justify-center bg-white/50">
<Loader className="h-8 w-8 animate-spin text-gray-500" />
</div>
)}
</div>
);
};