2023-07-07 12:55:21 +00:00
|
|
|
import { useState, useRef, useMemo, useCallback, useEffect } from "react";
|
2023-04-15 00:04:48 +00:00
|
|
|
|
2023-07-07 12:55:21 +00:00
|
|
|
import { WEBAPP_URL } from "@calcom/lib/constants";
|
|
|
|
import { useDebounce } from "@calcom/lib/hooks/useDebounce";
|
2023-04-15 00:04:48 +00:00
|
|
|
import { trpc } from "@calcom/trpc/react";
|
2023-07-07 12:55:21 +00:00
|
|
|
import {
|
|
|
|
Badge,
|
|
|
|
ConfirmationDialogContent,
|
|
|
|
Dialog,
|
|
|
|
DropdownActions,
|
|
|
|
showToast,
|
|
|
|
Table,
|
|
|
|
TextField,
|
|
|
|
Avatar,
|
|
|
|
} from "@calcom/ui";
|
|
|
|
import { Edit, Trash, Lock } from "@calcom/ui/components/icon";
|
2023-04-15 00:04:48 +00:00
|
|
|
|
2023-04-19 14:15:08 +00:00
|
|
|
import { withLicenseRequired } from "../../common/components/LicenseRequired";
|
2023-04-15 00:04:48 +00:00
|
|
|
|
2023-07-07 12:55:21 +00:00
|
|
|
const { Cell, ColumnTitle, Header, Row } = Table;
|
|
|
|
|
|
|
|
const FETCH_LIMIT = 25;
|
2023-04-15 00:04:48 +00:00
|
|
|
|
|
|
|
function UsersTableBare() {
|
2023-07-07 12:55:21 +00:00
|
|
|
const tableContainerRef = useRef<HTMLDivElement>(null);
|
2023-04-15 00:04:48 +00:00
|
|
|
const utils = trpc.useContext();
|
2023-07-07 12:55:21 +00:00
|
|
|
const [searchTerm, setSearchTerm] = useState<string>("");
|
|
|
|
const debouncedSearchTerm = useDebounce(searchTerm, 500);
|
|
|
|
|
2023-04-15 00:04:48 +00:00
|
|
|
const mutation = trpc.viewer.users.delete.useMutation({
|
|
|
|
onSuccess: async () => {
|
|
|
|
showToast("User has been deleted", "success");
|
2023-07-07 12:55:21 +00:00
|
|
|
// Lets not invalidated the whole cache, just remove the user from the cache.
|
|
|
|
// usefull cause in prod this will be fetching 100k+ users
|
2023-08-02 09:35:48 +00:00
|
|
|
// FIXME: Tested locally and it doesnt't work, need to investigate
|
|
|
|
utils.viewer.admin.listPaginated.setInfiniteData({ limit: FETCH_LIMIT }, (cachedData) => {
|
|
|
|
if (!cachedData) {
|
2023-07-07 12:55:21 +00:00
|
|
|
return {
|
|
|
|
pages: [],
|
|
|
|
pageParams: [],
|
|
|
|
};
|
|
|
|
}
|
|
|
|
return {
|
2023-08-02 09:35:48 +00:00
|
|
|
...cachedData,
|
|
|
|
pages: cachedData.pages.map((page) => ({
|
2023-07-07 12:55:21 +00:00
|
|
|
...page,
|
|
|
|
rows: page.rows.filter((row) => row.id !== userToDelete),
|
|
|
|
})),
|
|
|
|
};
|
|
|
|
});
|
2023-04-15 00:04:48 +00:00
|
|
|
},
|
|
|
|
onError: (err) => {
|
|
|
|
console.error(err.message);
|
|
|
|
showToast("There has been an error deleting this user.", "error");
|
|
|
|
},
|
|
|
|
onSettled: () => {
|
|
|
|
setUserToDelete(null);
|
|
|
|
},
|
|
|
|
});
|
2023-07-07 12:55:21 +00:00
|
|
|
|
|
|
|
const { data, fetchNextPage, isFetching } = trpc.viewer.admin.listPaginated.useInfiniteQuery(
|
|
|
|
{
|
|
|
|
limit: FETCH_LIMIT,
|
|
|
|
searchTerm: debouncedSearchTerm,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
getNextPageParam: (lastPage) => lastPage.nextCursor,
|
|
|
|
keepPreviousData: true,
|
|
|
|
refetchOnWindowFocus: false,
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
const sendPasswordResetEmail = trpc.viewer.admin.sendPasswordReset.useMutation({
|
|
|
|
onSuccess: () => {
|
|
|
|
showToast("Password reset email has been sent", "success");
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
//we must flatten the array of arrays from the useInfiniteQuery hook
|
|
|
|
const flatData = useMemo(() => data?.pages?.flatMap((page) => page.rows) ?? [], [data]);
|
|
|
|
const totalDBRowCount = data?.pages?.[0]?.meta?.totalRowCount ?? 0;
|
|
|
|
const totalFetched = flatData.length;
|
|
|
|
|
|
|
|
//called on scroll and possibly on mount to fetch more data as the user scrolls and reaches bottom of table
|
|
|
|
const fetchMoreOnBottomReached = useCallback(
|
|
|
|
(containerRefElement?: HTMLDivElement | null) => {
|
|
|
|
if (containerRefElement) {
|
|
|
|
const { scrollHeight, scrollTop, clientHeight } = containerRefElement;
|
|
|
|
//once the user has scrolled within 300px of the bottom of the table, fetch more data if there is any
|
|
|
|
if (scrollHeight - scrollTop - clientHeight < 300 && !isFetching && totalFetched < totalDBRowCount) {
|
|
|
|
fetchNextPage();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
[fetchNextPage, isFetching, totalFetched, totalDBRowCount]
|
|
|
|
);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
fetchMoreOnBottomReached(tableContainerRef.current);
|
|
|
|
}, [fetchMoreOnBottomReached]);
|
|
|
|
|
2023-04-15 00:04:48 +00:00
|
|
|
const [userToDelete, setUserToDelete] = useState<number | null>(null);
|
|
|
|
|
|
|
|
return (
|
2023-07-07 12:55:21 +00:00
|
|
|
<>
|
|
|
|
<TextField
|
|
|
|
placeholder="username or email"
|
|
|
|
label="Search"
|
|
|
|
onChange={(e) => setSearchTerm(e.target.value)}
|
2023-04-15 00:04:48 +00:00
|
|
|
/>
|
2023-07-07 12:55:21 +00:00
|
|
|
<div
|
|
|
|
className="rounded-md border"
|
|
|
|
ref={tableContainerRef}
|
|
|
|
onScroll={() => fetchMoreOnBottomReached()}
|
|
|
|
style={{
|
|
|
|
height: "calc(100vh - 30vh)",
|
|
|
|
overflow: "auto",
|
|
|
|
}}>
|
|
|
|
<Table>
|
|
|
|
<Header>
|
|
|
|
<ColumnTitle widthClassNames="w-auto">User</ColumnTitle>
|
|
|
|
<ColumnTitle>Timezone</ColumnTitle>
|
|
|
|
<ColumnTitle>Role</ColumnTitle>
|
|
|
|
<ColumnTitle widthClassNames="w-auto">
|
|
|
|
<span className="sr-only">Edit</span>
|
|
|
|
</ColumnTitle>
|
|
|
|
</Header>
|
|
|
|
|
|
|
|
<tbody className="divide-subtle divide-y rounded-md">
|
|
|
|
{flatData.map((user) => (
|
|
|
|
<Row key={user.email}>
|
|
|
|
<Cell widthClassNames="w-auto">
|
|
|
|
<div className="min-h-10 flex ">
|
|
|
|
<Avatar
|
|
|
|
size="md"
|
|
|
|
alt={`Avatar of ${user.username || "Nameless"}`}
|
|
|
|
gravatarFallbackMd5=""
|
2023-08-03 15:32:38 +00:00
|
|
|
imageSrc={`${WEBAPP_URL}/${user.username}/avatar.png?orgId=${user.organizationId}`}
|
2023-07-07 12:55:21 +00:00
|
|
|
/>
|
|
|
|
|
|
|
|
<div className="text-subtle ml-4 font-medium">
|
|
|
|
<span className="text-default">{user.name}</span>
|
|
|
|
<span className="ml-3">/{user.username}</span>
|
|
|
|
<br />
|
|
|
|
<span className="break-all">{user.email}</span>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</Cell>
|
|
|
|
<Cell>{user.timeZone}</Cell>
|
|
|
|
<Cell>
|
|
|
|
<Badge className="capitalize" variant={user.role === "ADMIN" ? "red" : "gray"}>
|
|
|
|
{user.role.toLowerCase()}
|
|
|
|
</Badge>
|
|
|
|
</Cell>
|
|
|
|
<Cell widthClassNames="w-auto">
|
|
|
|
<div className="flex w-full justify-end">
|
|
|
|
<DropdownActions
|
|
|
|
actions={[
|
|
|
|
{
|
|
|
|
id: "edit",
|
|
|
|
label: "Edit",
|
|
|
|
href: `/settings/admin/users/${user.id}/edit`,
|
|
|
|
icon: Edit,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: "reset-password",
|
|
|
|
label: "Reset Password",
|
|
|
|
onClick: () => sendPasswordResetEmail.mutate({ userId: user.id }),
|
|
|
|
icon: Lock,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: "delete",
|
|
|
|
label: "Delete",
|
|
|
|
color: "destructive",
|
|
|
|
onClick: () => setUserToDelete(user.id),
|
|
|
|
icon: Trash,
|
|
|
|
},
|
|
|
|
]}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</Cell>
|
|
|
|
</Row>
|
|
|
|
))}
|
|
|
|
</tbody>
|
|
|
|
</Table>
|
|
|
|
<DeleteUserDialog
|
|
|
|
user={userToDelete}
|
|
|
|
onClose={() => setUserToDelete(null)}
|
|
|
|
onConfirm={() => {
|
|
|
|
if (!userToDelete) return;
|
|
|
|
mutation.mutate({ userId: userToDelete });
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</>
|
2023-04-15 00:04:48 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
const DeleteUserDialog = ({
|
|
|
|
user,
|
|
|
|
onConfirm,
|
|
|
|
onClose,
|
|
|
|
}: {
|
|
|
|
user: number | null;
|
|
|
|
onConfirm: () => void;
|
|
|
|
onClose: () => void;
|
|
|
|
}) => {
|
|
|
|
return (
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-empty-function -- noop
|
|
|
|
<Dialog name="delete-user" open={!!user} onOpenChange={(open) => (open ? () => {} : onClose())}>
|
|
|
|
<ConfirmationDialogContent
|
|
|
|
title="Delete User"
|
|
|
|
confirmBtnText="Delete"
|
|
|
|
cancelBtnText="Cancel"
|
|
|
|
variety="danger"
|
|
|
|
onConfirm={onConfirm}>
|
|
|
|
<p>Are you sure you want to delete this user?</p>
|
|
|
|
</ConfirmationDialogContent>
|
|
|
|
</Dialog>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const UsersTable = withLicenseRequired(UsersTableBare);
|