chore: leaderboard table

This commit is contained in:
Ephraim Atta-Duncan
2024-09-19 13:08:29 +00:00
parent 670393d0d0
commit 894e857826
6 changed files with 147 additions and 331 deletions

View File

@ -8,20 +8,16 @@ import { ChevronDownIcon as CaretSortIcon, Loader } from 'lucide-react';
import { useDebouncedValue } from '@documenso/lib/client-only/hooks/use-debounced-value';
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';
import { DataTable } from '@documenso/ui/primitives/data-table';
import { DataTablePagination } from '@documenso/ui/primitives/data-table-pagination';
import { Input } from '@documenso/ui/primitives/input';
export type SigningVolume = {
customer_id: number;
customer_type: 'User' | 'Team';
customer_created_at: Date;
total_documents: bigint;
completed_documents: bigint;
customer_email: string;
customer_name: string;
id: number;
name: string;
signingVolume: number;
createdAt: Date;
};
type LeaderboardTableProps = {
@ -46,57 +42,53 @@ export const LeaderboardTable = ({
const columns = useMemo(() => {
return [
{
header: 'ID',
accessorKey: 'id',
cell: ({ row }) => <div>{row.original.id}</div>,
size: 10,
},
{
header: ({ column }) => (
<Button
variant="ghost"
<div
className="flex cursor-pointer items-center"
onClick={() => column.toggleSorting(column.getIsSorted() === 'asc')}
>
{_(msg`Name`)}
<CaretSortIcon className="ml-2 h-4 w-4" />
</Button>
</div>
),
accessorKey: 'customer_name',
cell: ({ row }) => <div>{row.getValue('customer_name')}</div>,
accessorKey: 'name',
cell: ({ row }) => <div>{row.getValue('name')}</div>,
size: 250,
},
{
header: ({ column }) => (
<Button
variant="ghost"
onClick={() => column.toggleSorting(column.getIsSorted() === 'asc')}
>
{_(msg`Email`)}
<CaretSortIcon className="ml-2 h-4 w-4" />
</Button>
),
accessorKey: 'customer_email',
cell: ({ row }) => <div>{row.getValue('customer_email')}</div>,
},
{
header: ({ column }) => (
<Button
variant="ghost"
<div
className="flex cursor-pointer items-center"
onClick={() => column.toggleSorting(column.getIsSorted() === 'asc')}
>
{_(msg`Signing Volume`)}
<CaretSortIcon className="ml-2 h-4 w-4" />
</Button>
</div>
),
accessorKey: 'completed_documents',
cell: ({ row }) => <div>{Number(row.getValue('completed_documents'))}</div>,
accessorKey: 'signingVolume',
cell: ({ row }) => <div>{Number(row.getValue('signingVolume'))}</div>,
},
{
header: ({ column }) => (
<Button
variant="ghost"
onClick={() => column.toggleSorting(column.getIsSorted() === 'asc')}
>
{_(msg`Customer Type`)}
<CaretSortIcon className="ml-2 h-4 w-4" />
</Button>
),
accessorKey: 'customer_type',
cell: ({ row }) => <div>{row.getValue('customer_type')}</div>,
header: ({ column }) => {
return (
<div
className="flex cursor-pointer items-center"
onClick={() => column.toggleSorting(column.getIsSorted() === 'asc')}
>
{_(msg`Created`)}
<CaretSortIcon className="ml-2 h-4 w-4" />
</div>
);
},
accessorKey: 'createdAt',
cell: ({ row }) => <div>{row.original.createdAt.toLocaleDateString()}</div>,
},
] satisfies DataTableColumnDef<SigningVolume>[];
}, []);

View File

@ -1,155 +0,0 @@
'use client';
import { useEffect, useMemo, useState, useTransition } from 'react';
import { msg } from '@lingui/macro';
import { useLingui } from '@lingui/react';
import { ChevronDownIcon as CaretSortIcon, Loader } from 'lucide-react';
import { useDebouncedValue } from '@documenso/lib/client-only/hooks/use-debounced-value';
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';
import { DataTable } from '@documenso/ui/primitives/data-table';
import { DataTablePagination } from '@documenso/ui/primitives/data-table-pagination';
import { Input } from '@documenso/ui/primitives/input';
export type SigningVolume = {
customer_id: number;
customer_type: 'User' | 'Team';
customer_created_at: Date;
total_documents: bigint;
completed_documents: bigint;
customer_email: string;
customer_name: string;
};
type LeaderboardTableProps = {
signingVolume: SigningVolume[];
totalPages: number;
perPage: number;
page: number;
};
export const LeaderboardTable = ({
signingVolume,
totalPages,
perPage,
page,
}: LeaderboardTableProps) => {
const { _ } = useLingui();
const [isPending, startTransition] = useTransition();
const updateSearchParams = useUpdateSearchParams();
const [searchString, setSearchString] = useState('');
const debouncedSearchString = useDebouncedValue(searchString, 1000);
const columns = useMemo(() => {
return [
{
header: ({ column }) => (
<Button
variant="ghost"
onClick={() => column.toggleSorting(column.getIsSorted() === 'asc')}
>
{_(msg`Name`)}
<CaretSortIcon className="ml-2 h-4 w-4" />
</Button>
),
accessorKey: 'customer_name',
cell: ({ row }) => <div>{row.getValue('customer_name')}</div>,
},
{
header: ({ column }) => (
<Button
variant="ghost"
onClick={() => column.toggleSorting(column.getIsSorted() === 'asc')}
>
{_(msg`Email`)}
<CaretSortIcon className="ml-2 h-4 w-4" />
</Button>
),
accessorKey: 'customer_email',
cell: ({ row }) => <div>{row.getValue('customer_email')}</div>,
},
{
header: ({ column }) => (
<Button
variant="ghost"
onClick={() => column.toggleSorting(column.getIsSorted() === 'asc')}
>
{_(msg`Signing Volume`)}
<CaretSortIcon className="ml-2 h-4 w-4" />
</Button>
),
accessorKey: 'completed_documents',
cell: ({ row }) => <div>{Number(row.getValue('completed_documents'))}</div>,
},
{
header: ({ column }) => (
<Button
variant="ghost"
onClick={() => column.toggleSorting(column.getIsSorted() === 'asc')}
>
{_(msg`Customer Type`)}
<CaretSortIcon className="ml-2 h-4 w-4" />
</Button>
),
accessorKey: 'customer_type',
cell: ({ row }) => <div>{row.getValue('customer_type')}</div>,
},
] satisfies DataTableColumnDef<SigningVolume>[];
}, []);
useEffect(() => {
startTransition(() => {
updateSearchParams({
search: debouncedSearchString,
page: 1,
perPage,
});
});
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [debouncedSearchString]);
const onPaginationChange = (page: number, perPage: number) => {
startTransition(() => {
updateSearchParams({
page,
perPage,
});
});
};
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setSearchString(e.target.value);
};
return (
<div className="relative">
<Input
className="my-6 flex flex-row gap-4"
type="text"
placeholder={_(msg`Search by name or email`)}
value={searchString}
onChange={handleChange}
/>
<DataTable
columns={columns}
data={signingVolume}
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>
);
};

View File

@ -28,7 +28,8 @@ export default async function Leaderboard({ searchParams = {} }: AdminLeaderboar
const perPage = Number(searchParams.perPage) || 10;
const searchString = searchParams.search || '';
const { signingVolume, totalPages } = await search(searchString, page, perPage);
// todo: change the name
const { leaderboard: signingVolume, totalPages } = await search(searchString, page, perPage);
return (
<div>