147 lines
4.7 KiB
TypeScript
147 lines
4.7 KiB
TypeScript
import { GetServerSidePropsContext } from "next";
|
|
import { useState } from "react";
|
|
|
|
import { useLocale } from "@calcom/lib/hooks/useLocale";
|
|
import { trpc } from "@calcom/trpc/react";
|
|
import {
|
|
Button,
|
|
Dialog,
|
|
DialogClose,
|
|
DialogContent,
|
|
DialogFooter,
|
|
Dropdown,
|
|
DropdownMenuContent,
|
|
DropdownMenuItem,
|
|
DropdownMenuTrigger,
|
|
getSettingsLayout as getLayout,
|
|
Icon,
|
|
List,
|
|
ListItem,
|
|
ListItemText,
|
|
ListItemTitle,
|
|
Meta,
|
|
showToast,
|
|
SkeletonContainer,
|
|
SkeletonText,
|
|
} from "@calcom/ui";
|
|
|
|
import { ssrInit } from "@server/lib/ssr";
|
|
|
|
const SkeletonLoader = ({ title, description }: { title: string; description: string }) => {
|
|
return (
|
|
<SkeletonContainer>
|
|
<Meta title={title} description={description} />
|
|
<div className="mt-6 mb-8 space-y-6 divide-y">
|
|
<SkeletonText className="h-8 w-full" />
|
|
<SkeletonText className="h-8 w-full" />
|
|
</div>
|
|
</SkeletonContainer>
|
|
);
|
|
};
|
|
|
|
const ConferencingLayout = () => {
|
|
const { t } = useLocale();
|
|
const utils = trpc.useContext();
|
|
|
|
const { data: apps, isLoading } = trpc.viewer.integrations.useQuery(
|
|
{ variant: "conferencing", onlyInstalled: true },
|
|
{
|
|
suspense: true,
|
|
}
|
|
);
|
|
const deleteAppMutation = trpc.viewer.deleteCredential.useMutation({
|
|
onSuccess: () => {
|
|
showToast("Integration deleted successfully", "success");
|
|
utils.viewer.integrations.invalidate({ variant: "conferencing", onlyInstalled: true });
|
|
setDeleteAppModal(false);
|
|
},
|
|
onError: () => {
|
|
showToast("Error deleting app", "error");
|
|
setDeleteAppModal(false);
|
|
},
|
|
});
|
|
|
|
const [deleteAppModal, setDeleteAppModal] = useState(false);
|
|
const [deleteCredentialId, setDeleteCredentialId] = useState<number>(0);
|
|
|
|
if (isLoading)
|
|
return <SkeletonLoader title={t("conferencing")} description={t("conferencing_description")} />;
|
|
|
|
return (
|
|
<div className="w-full bg-white sm:mx-0 xl:mt-0">
|
|
<Meta title={t("conferencing")} description={t("conferencing_description")} />
|
|
<List>
|
|
{apps?.items &&
|
|
apps.items
|
|
.map((app) => ({ ...app, title: app.title || app.name }))
|
|
.map((app) => (
|
|
<ListItem className="flex-col border-0" key={app.title}>
|
|
<div className="flex w-full flex-1 items-center space-x-2 p-4 rtl:space-x-reverse">
|
|
{
|
|
// eslint-disable-next-line @next/next/no-img-element
|
|
app.logo && <img className="h-10 w-10" src={app.logo} alt={app.title} />
|
|
}
|
|
<div className="flex-grow truncate pl-2">
|
|
<ListItemTitle component="h3" className="mb-1 space-x-2">
|
|
<h3 className="truncate text-sm font-medium text-neutral-900">{app.title}</h3>
|
|
</ListItemTitle>
|
|
<ListItemText component="p">{app.description}</ListItemText>
|
|
</div>
|
|
<div>
|
|
<Dropdown>
|
|
<DropdownMenuTrigger asChild>
|
|
<Button StartIcon={Icon.FiMoreHorizontal} size="icon" color="secondary" />
|
|
</DropdownMenuTrigger>
|
|
<DropdownMenuContent>
|
|
<DropdownMenuItem>
|
|
<Button
|
|
color="destructive"
|
|
StartIcon={Icon.FiTrash}
|
|
disabled={app.isGlobal}
|
|
onClick={() => {
|
|
setDeleteCredentialId(app.credentialIds[0]);
|
|
setDeleteAppModal(true);
|
|
}}>
|
|
{t("remove_app")}
|
|
</Button>
|
|
</DropdownMenuItem>
|
|
</DropdownMenuContent>
|
|
</Dropdown>
|
|
</div>
|
|
</div>
|
|
</ListItem>
|
|
))}
|
|
</List>
|
|
|
|
<Dialog open={deleteAppModal} onOpenChange={setDeleteAppModal}>
|
|
<DialogContent
|
|
title={t("Remove app")}
|
|
description={t("are_you_sure_you_want_to_remove_this_app")}
|
|
type="confirmation"
|
|
Icon={Icon.FiAlertCircle}>
|
|
<DialogFooter>
|
|
<Button color="primary" onClick={() => deleteAppMutation.mutate({ id: deleteCredentialId })}>
|
|
{t("yes_remove_app")}
|
|
</Button>
|
|
<DialogClose />
|
|
</DialogFooter>
|
|
</DialogContent>
|
|
</Dialog>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
ConferencingLayout.getLayout = getLayout;
|
|
|
|
export const getServerSideProps = async (context: GetServerSidePropsContext) => {
|
|
const ssr = await ssrInit(context);
|
|
|
|
return {
|
|
props: {
|
|
trpcState: ssr.dehydrate(),
|
|
},
|
|
};
|
|
};
|
|
|
|
export default ConferencingLayout;
|