feat: filter users by name or email

This commit is contained in:
pit
2023-10-06 15:48:05 +03:00
committed by Mythie
parent 02b6f6a7b7
commit 9682f8ea36
7 changed files with 192 additions and 45 deletions

View File

@ -10,10 +10,10 @@ import { useSession } from 'next-auth/react';
import { useUpdateSearchParams } from '@documenso/lib/client-only/hooks/use-update-search-params'; import { useUpdateSearchParams } from '@documenso/lib/client-only/hooks/use-update-search-params';
import { FindResultSet } from '@documenso/lib/types/find-result-set'; import { FindResultSet } from '@documenso/lib/types/find-result-set';
import { Document, Recipient, User } from '@documenso/prisma/client'; import { Document, Recipient, User } from '@documenso/prisma/client';
import { Avatar, AvatarFallback } from '@documenso/ui/primitives/avatar';
import { DataTable } from '@documenso/ui/primitives/data-table'; import { DataTable } from '@documenso/ui/primitives/data-table';
import { DataTablePagination } from '@documenso/ui/primitives/data-table-pagination'; import { DataTablePagination } from '@documenso/ui/primitives/data-table-pagination';
import { StackAvatarsWithTooltip } from '~/components/(dashboard)/avatar/stack-avatars-with-tooltip';
import { DocumentStatus } from '~/components/formatter/document-status'; import { DocumentStatus } from '~/components/formatter/document-status';
import { LocaleDate } from '~/components/formatter/locale-date'; import { LocaleDate } from '~/components/formatter/locale-date';
@ -68,15 +68,11 @@ export const DocumentsDataTable = ({ results }: DocumentsDataTableProps) => {
cell: ({ row }) => { cell: ({ row }) => {
return ( return (
<Link href={`/admin/users/${row.original.User.id}`}> <Link href={`/admin/users/${row.original.User.id}`}>
<StackAvatarsWithTooltip <Avatar className="dark:border-border h-12 w-12 border-2 border-solid border-white">
recipients={[ <AvatarFallback className="text-gray-400">
{ <span className="text-xs">{row.original.User.name}</span>
id: row.original.User.id, </AvatarFallback>
email: row.original.User.email, </Avatar>
name: row.original.User.name,
},
]}
/>
</Link> </Link>
); );
}, },

View File

@ -11,12 +11,10 @@ export type DocumentsPageProps = {
}; };
export default async function Documents({ searchParams = {} }: DocumentsPageProps) { export default async function Documents({ searchParams = {} }: DocumentsPageProps) {
const user = await getRequiredServerComponentSession();
const page = Number(searchParams.page) || 1; const page = Number(searchParams.page) || 1;
const perPage = Number(searchParams.perPage) || 20; const perPage = Number(searchParams.perPage) || 20;
const results = await findDocuments({ const results = await findDocuments({
userId: user.id,
orderBy: { orderBy: {
column: 'createdAt', column: 'createdAt',
direction: 'desc', direction: 'desc',

View File

@ -7,7 +7,7 @@ import Link from 'next/link';
import { Edit, Loader } from 'lucide-react'; import { Edit, Loader } from 'lucide-react';
import { useUpdateSearchParams } from '@documenso/lib/client-only/hooks/use-update-search-params'; import { useUpdateSearchParams } from '@documenso/lib/client-only/hooks/use-update-search-params';
import { Role } from '@documenso/prisma/client'; import { Document, Role, Subscription } from '@documenso/prisma/client';
import { Button } from '@documenso/ui/primitives/button'; import { Button } from '@documenso/ui/primitives/button';
import { DataTable } from '@documenso/ui/primitives/data-table'; import { DataTable } from '@documenso/ui/primitives/data-table';
import { DataTablePagination } from '@documenso/ui/primitives/data-table-pagination'; import { DataTablePagination } from '@documenso/ui/primitives/data-table-pagination';
@ -17,18 +17,16 @@ interface User {
name: string | null; name: string | null;
email: string; email: string;
roles: Role[]; roles: Role[];
Subscription: Subscription[]; Subscription: SubscriptionLite[];
Document: Document[]; Document: DocumentLite[];
} }
interface Subscription { type SubscriptionLite = Pick<
id: number; Subscription,
status: string; 'id' | 'status' | 'planId' | 'priceId' | 'createdAt' | 'periodEnd'
planId: string | null; >;
priceId: string | null;
createdAt: Date | null; type DocumentLite = Pick<Document, 'id'>;
periodEnd: Date | null;
}
type UsersDataTableProps = { type UsersDataTableProps = {
users: User[]; users: User[];
@ -37,10 +35,6 @@ type UsersDataTableProps = {
totalPages: number; totalPages: number;
}; };
type Document = {
id: number;
};
export const UsersDataTable = ({ users, perPage, page, totalPages }: UsersDataTableProps) => { export const UsersDataTable = ({ users, perPage, page, totalPages }: UsersDataTableProps) => {
const [isPending, startTransition] = useTransition(); const [isPending, startTransition] = useTransition();
const updateSearchParams = useUpdateSearchParams(); const updateSearchParams = useUpdateSearchParams();
@ -97,7 +91,7 @@ export const UsersDataTable = ({ users, perPage, page, totalPages }: UsersDataTa
if (row.original.Subscription && row.original.Subscription.length > 0) { if (row.original.Subscription && row.original.Subscription.length > 0) {
return ( return (
<> <>
{row.original.Subscription.map((subscription: Subscription, i: number) => { {row.original.Subscription.map((subscription: SubscriptionLite, i: number) => {
return <span key={i}>{subscription.status}</span>; return <span key={i}>{subscription.status}</span>;
})} })}
</> </>

View File

@ -1,10 +1,6 @@
import { findUsers } from '@documenso/lib/server-only/user/get-all-users'; import { findUsers } from '@documenso/lib/server-only/user/get-all-users';
/* import { Users } from './users';
1. retrieve all users from the db
2. display them in a table
*/
import { UsersDataTable } from './data-table-users';
type AdminManageUsersProps = { type AdminManageUsersProps = {
searchParams?: { searchParams?: {
@ -13,23 +9,22 @@ type AdminManageUsersProps = {
}; };
}; };
export default async function AdminManageUsers({ searchParams = {} }: AdminManageUsersProps) { export default function AdminManageUsers({ searchParams = {} }: AdminManageUsersProps) {
const page = Number(searchParams.page) || 1; const page = Number(searchParams.page) || 1;
const perPage = Number(searchParams.perPage) || 10; const perPage = Number(searchParams.perPage) || 10;
const results = await findUsers({ page, perPage }); async function search(search: string) {
'use server';
const results = await findUsers({ username: search, email: search, page, perPage });
return results;
}
return ( return (
<div> <div>
<h2 className="text-4xl font-semibold">Manage users</h2> <h2 className="text-4xl font-semibold">Manage users</h2>
<div className="mt-8"> <Users search={search} page={page} perPage={perPage} />
<UsersDataTable
users={results.users}
perPage={perPage}
page={page}
totalPages={results.totalPages}
/>
</div>
</div> </div>
); );
} }

View File

@ -0,0 +1,78 @@
'use client';
import { useEffect, useState } from 'react';
import { Document, Role, Subscription } from '@documenso/prisma/client';
import { Button } from '@documenso/ui/primitives/button';
import { Input } from '@documenso/ui/primitives/input';
import { UsersDataTable } from './data-table-users';
export type SubscriptionLite = Pick<
Subscription,
'id' | 'status' | 'planId' | 'priceId' | 'createdAt' | 'periodEnd'
>;
export type DocumentLite = Pick<Document, 'id'>;
export type User = {
id: number;
name: string | null;
email: string;
roles: Role[];
Subscription: SubscriptionLite[];
Document: DocumentLite[];
};
export type UsersProps = {
search: (search: string) => Promise<{ users: User[]; totalPages: number }>;
perPage: number;
page: number;
};
export const Users = ({ search, perPage, page }: UsersProps) => {
const [data, setData] = useState<User[]>([]);
const [totalPages, setTotalPages] = useState<number>(0);
const [searchString, setSearchString] = useState('');
useEffect(() => {
const fetchData = async () => {
try {
const result = await search(searchString);
setData(result.users);
setTotalPages(result.totalPages);
} catch (err) {
throw new Error(err);
}
};
fetchData();
}, [searchString, search]);
const onSubmit = async (e: React.FormEvent<HTMLFormElement>) => {
e.preventDefault();
const result = await search(searchString);
setData(result.users);
};
const handleChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setSearchString(e.target.value);
};
return (
<>
<form className="my-6 flex flex-row gap-4" onSubmit={onSubmit}>
<Input
type="text"
placeholder="Search by name or email and press enter"
value={searchString}
onChange={handleChange}
/>
<Button type="submit">Search</Button>
</form>
<div className="mt-8">
<UsersDataTable users={data} perPage={perPage} page={page} totalPages={totalPages} />
</div>
</>
);
};

View File

@ -0,0 +1,57 @@
import { useTransition } from 'react';
import { ColumnDef } from '@tanstack/react-table';
import { Loader } from 'lucide-react';
import { useUpdateSearchParams } from '@documenso/lib/client-only/hooks/use-update-search-params';
import { DataTable } from '@documenso/ui/primitives/data-table';
import { DataTablePagination } from '@documenso/ui/primitives/data-table-pagination';
type GenericDataTableProps<TData, TValue> = {
columns: ColumnDef<TData, TValue>[];
data: TData[];
perPage: number;
currentPage: number;
totalPages: number;
};
export function GenericDataTable<TData, TValue>({
columns,
data,
perPage,
currentPage,
totalPages,
}: GenericDataTableProps<TData, TValue>) {
const [isPending, startTransition] = useTransition();
const updateSearchParams = useUpdateSearchParams();
const onPaginationChange = (page: number, perPage: number) => {
startTransition(() => {
updateSearchParams({
page: page.toString(),
perPage: perPage.toString(),
});
});
};
return (
<div className="relative">
<DataTable
columns={columns}
data={data}
perPage={perPage}
currentPage={currentPage}
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

@ -1,11 +1,37 @@
import { Prisma } from '@prisma/client';
import { prisma } from '@documenso/prisma'; import { prisma } from '@documenso/prisma';
type getAllUsersProps = { type getAllUsersProps = {
username: string;
email: string;
page: number; page: number;
perPage: number; perPage: number;
}; };
export const findUsers = async ({ page = 1, perPage = 10 }: getAllUsersProps) => { export const findUsers = async ({
username = '',
email = '',
page = 1,
perPage = 10,
}: getAllUsersProps) => {
const whereClause = Prisma.validator<Prisma.UserWhereInput>()({
OR: [
{
name: {
contains: username,
mode: 'insensitive',
},
},
{
email: {
contains: email,
mode: 'insensitive',
},
},
],
});
const [users, count] = await Promise.all([ const [users, count] = await Promise.all([
await prisma.user.findMany({ await prisma.user.findMany({
select: { select: {
@ -29,10 +55,13 @@ export const findUsers = async ({ page = 1, perPage = 10 }: getAllUsersProps) =>
}, },
}, },
}, },
where: whereClause,
skip: Math.max(page - 1, 0) * perPage, skip: Math.max(page - 1, 0) * perPage,
take: perPage, take: perPage,
}), }),
await prisma.user.count(), await prisma.user.count({
where: whereClause,
}),
]); ]);
return { return {