101 lines
2.8 KiB
TypeScript
101 lines
2.8 KiB
TypeScript
import { useMutation } from "@tanstack/react-query";
|
|
import { useState } from "react";
|
|
|
|
import { classNames } from "@calcom/lib";
|
|
import { useLocale } from "@calcom/lib/hooks/useLocale";
|
|
import { trpc } from "@calcom/trpc/react";
|
|
import { showToast, Switch } from "@calcom/ui";
|
|
import { ArrowLeft, RotateCcw } from "@calcom/ui/components/icon";
|
|
|
|
interface ICalendarSwitchProps {
|
|
title: string;
|
|
externalId: string;
|
|
type: string;
|
|
isChecked: boolean;
|
|
name: string;
|
|
isLastItemInList?: boolean;
|
|
destination?: boolean;
|
|
}
|
|
const CalendarSwitch = (props: ICalendarSwitchProps) => {
|
|
const { title, externalId, type, isChecked, name, isLastItemInList = false } = props;
|
|
const [checkedInternal, setCheckedInternal] = useState(isChecked);
|
|
const utils = trpc.useContext();
|
|
const { t } = useLocale();
|
|
const mutation = useMutation<
|
|
unknown,
|
|
unknown,
|
|
{
|
|
isOn: boolean;
|
|
}
|
|
>(
|
|
async ({ isOn }: { isOn: boolean }) => {
|
|
const body = {
|
|
integration: type,
|
|
externalId: externalId,
|
|
};
|
|
|
|
if (isOn) {
|
|
const res = await fetch("/api/availability/calendar", {
|
|
method: "POST",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
},
|
|
body: JSON.stringify(body),
|
|
});
|
|
|
|
if (!res.ok) {
|
|
throw new Error("Something went wrong");
|
|
}
|
|
} else {
|
|
const res = await fetch("/api/availability/calendar", {
|
|
method: "DELETE",
|
|
headers: {
|
|
"Content-Type": "application/json",
|
|
},
|
|
body: JSON.stringify(body),
|
|
});
|
|
|
|
if (!res.ok) {
|
|
throw new Error("Something went wrong");
|
|
}
|
|
}
|
|
},
|
|
{
|
|
async onSettled() {
|
|
await utils.viewer.integrations.invalidate();
|
|
await utils.viewer.connectedCalendars.invalidate();
|
|
},
|
|
onError() {
|
|
setCheckedInternal(false);
|
|
showToast(`Something went wrong when toggling "${title}""`, "error");
|
|
},
|
|
}
|
|
);
|
|
return (
|
|
<div className={classNames("my-2 flex flex-row items-center")}>
|
|
<div className="flex pl-2">
|
|
<Switch
|
|
id={externalId}
|
|
checked={checkedInternal}
|
|
onCheckedChange={(isOn: boolean) => {
|
|
setCheckedInternal(isOn);
|
|
mutation.mutate({ isOn });
|
|
}}
|
|
/>
|
|
</div>
|
|
<label className="ml-3 text-sm font-medium leading-5" htmlFor={externalId}>
|
|
{name}
|
|
</label>
|
|
{!!props.destination && (
|
|
<span className="bg-subtle text-default ml-8 inline-flex items-center gap-1 rounded-md px-2 py-1 text-sm font-normal sm:ml-4">
|
|
<ArrowLeft className="h-4 w-4" />
|
|
{t("adding_events_to")}
|
|
</span>
|
|
)}
|
|
{mutation.isLoading && <RotateCcw className="text-muted h-4 w-4 animate-spin ltr:ml-1 rtl:mr-1" />}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export { CalendarSwitch };
|