Files
sign/apps/web/src/app/(dashboard)/admin/users/data-table-users.tsx

144 lines
4.0 KiB
TypeScript
Raw Normal View History

2023-09-21 12:43:36 +01:00
'use client';
import { useEffect, useState, useTransition } from 'react';
2023-09-21 12:43:36 +01:00
import Link from 'next/link';
import { Edit, Loader } from 'lucide-react';
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';
2023-10-06 15:48:05 +03:00
import { Document, Role, Subscription } from '@documenso/prisma/client';
2023-09-21 12:43:36 +01:00
import { Button } from '@documenso/ui/primitives/button';
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
interface User {
id: number;
name: string | null;
email: string;
roles: Role[];
2023-10-16 17:38:41 +11:00
Subscription?: SubscriptionLite | null;
2023-10-06 15:48:05 +03:00
Document: 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
type UsersDataTableProps = {
2023-10-11 16:20:04 +03:00
users: User[];
totalPages: number;
2023-09-21 12:43:36 +01:00
perPage: number;
page: number;
};
2023-10-11 16:20:04 +03:00
export const UsersDataTable = ({ users, totalPages, perPage, page }: UsersDataTableProps) => {
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
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,
});
});
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"
placeholder="Search by name or email"
value={searchString}
onChange={handleChange}
/>
2023-09-21 12:43:36 +01:00
<DataTable
columns={[
{
header: 'ID',
accessorKey: 'id',
cell: ({ row }) => <div>{row.original.id}</div>,
},
{
header: 'Name',
accessorKey: 'name',
cell: ({ row }) => <div>{row.original.name}</div>,
},
{
header: 'Email',
accessorKey: 'email',
cell: ({ row }) => <div>{row.original.email}</div>,
},
{
header: 'Roles',
accessorKey: 'roles',
2023-10-13 13:08:39 +00:00
cell: ({ row }) => row.original.roles.join(', '),
2023-09-21 12:43:36 +01:00
},
{
2023-09-29 17:12:02 +01:00
header: 'Subscription',
2023-09-21 12:43:36 +01:00
accessorKey: 'subscription',
2023-10-16 17:38:41 +11:00
cell: ({ row }) => row.original.Subscription?.status ?? 'NONE',
2023-09-29 17:12:02 +01:00
},
{
header: 'Documents',
accessorKey: 'documents',
2023-09-21 12:43:36 +01:00
cell: ({ row }) => {
2023-10-09 13:30:28 +03:00
return <div>{row.original.Document.length}</div>;
2023-09-21 12:43:36 +01:00
},
},
{
2023-10-13 13:08:39 +00:00
header: '',
2023-09-21 12:43:36 +01:00
accessorKey: 'edit',
cell: ({ row }) => {
return (
2023-10-13 13:08:39 +00:00
<Button className="w-24" asChild>
<Link href={`/admin/users/${row.original.id}`}>
<Edit className="-ml-1 mr-2 h-4 w-4" />
Edit
</Link>
</Button>
2023-09-21 12:43:36 +01:00
);
},
},
]}
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>
);
};