import { useAutoAnimate } from "@formkit/auto-animate/react"; import Link from "next/link"; import { useRouter, usePathname } from "next/navigation"; import { useCallback, useState } from "react"; import { BulkEditDefaultForEventsModal } from "@calcom/features/eventtypes/components/BulkEditDefaultForEventsModal"; import { NewScheduleButton, ScheduleListItem } from "@calcom/features/schedules"; import Shell from "@calcom/features/shell/Shell"; import { AvailabilitySliderTable } from "@calcom/features/timezone-buddy/components/AvailabilitySliderTable"; import { useCompatSearchParams } from "@calcom/lib/hooks/useCompatSearchParams"; import { useLocale } from "@calcom/lib/hooks/useLocale"; import { HttpError } from "@calcom/lib/http-error"; import { MembershipRole } from "@calcom/prisma/enums"; import type { RouterOutputs } from "@calcom/trpc/react"; import { trpc } from "@calcom/trpc/react"; import useMeQuery from "@calcom/trpc/react/hooks/useMeQuery"; import { EmptyScreen, showToast, ToggleGroup } from "@calcom/ui"; import { QueryCell } from "@lib/QueryCell"; import PageWrapper from "@components/PageWrapper"; import SkeletonLoader from "@components/availability/SkeletonLoader"; export function AvailabilityList({ schedules }: RouterOutputs["viewer"]["availability"]["list"]) { const { t } = useLocale(); const [bulkUpdateModal, setBulkUpdateModal] = useState(false); const utils = trpc.useUtils(); const meQuery = trpc.viewer.me.useQuery(); const router = useRouter(); const deleteMutation = trpc.viewer.availability.schedule.delete.useMutation({ onMutate: async ({ scheduleId }) => { await utils.viewer.availability.list.cancel(); const previousValue = utils.viewer.availability.list.getData(); if (previousValue) { const filteredValue = previousValue.schedules.filter(({ id }) => id !== scheduleId); utils.viewer.availability.list.setData(undefined, { ...previousValue, schedules: filteredValue }); } return { previousValue }; }, onError: (err, variables, context) => { if (context?.previousValue) { utils.viewer.availability.list.setData(undefined, context.previousValue); } if (err instanceof HttpError) { const message = `${err.statusCode}: ${err.message}`; showToast(message, "error"); } }, onSettled: () => { utils.viewer.availability.list.invalidate(); }, onSuccess: () => { showToast(t("schedule_deleted_successfully"), "success"); }, }); const updateMutation = trpc.viewer.availability.schedule.update.useMutation({ onSuccess: async ({ schedule }) => { await utils.viewer.availability.list.invalidate(); showToast( t("availability_updated_successfully", { scheduleName: schedule.name, }), "success" ); setBulkUpdateModal(true); }, onError: (err) => { if (err instanceof HttpError) { const message = `${err.statusCode}: ${err.message}`; showToast(message, "error"); } }, }); const bulkUpdateDefaultAvailabilityMutation = trpc.viewer.availability.schedule.bulkUpdateToDefaultAvailability.useMutation({ onSuccess: () => { utils.viewer.availability.list.invalidate(); setBulkUpdateModal(false); showToast(t("success"), "success"); }, }); const duplicateMutation = trpc.viewer.availability.schedule.duplicate.useMutation({ onSuccess: async ({ schedule }) => { await router.push(`/availability/${schedule.id}`); showToast(t("schedule_created_successfully", { scheduleName: schedule.name }), "success"); }, onError: (err) => { if (err instanceof HttpError) { const message = `${err.statusCode}: ${err.message}`; showToast(message, "error"); } }, }); // Adds smooth delete button - item fades and old item slides into place const [animationParentRef] = useAutoAnimate(); return ( <> {schedules.length === 0 ? (
} />
) : ( <>
{t("temporarily_out_of_office")}{" "} {t("add_a_redirect")}
{bulkUpdateModal && ( )} )} ); } function AvailabilityListWithQuery() { const query = trpc.viewer.availability.list.useQuery(); return ( } customLoader={} /> ); } export default function AvailabilityPage() { const { t } = useLocale(); const searchParams = useCompatSearchParams(); const router = useRouter(); const pathname = usePathname(); const me = useMeQuery(); const { data } = trpc.viewer.organizations.listCurrent.useQuery(); // Get a new searchParams string by merging the current // searchParams with a provided key/value pair const createQueryString = useCallback( (name: string, value: string) => { const params = new URLSearchParams(searchParams ?? undefined); params.set(name, value); return params.toString(); }, [searchParams] ); const isOrgAdminOrOwner = data && (data.user.role === MembershipRole.OWNER || data.user.role === MembershipRole.ADMIN); const isOrgAndPrivate = data?.isOrganization && data.isPrivate; const toggleGroupOptions = [{ value: "mine", label: t("my_availability") }]; if (!isOrgAndPrivate || isOrgAdminOrOwner) { toggleGroupOptions.push({ value: "team", label: t("team_availability") }); } return (
{ if (!value) return; router.push(`${pathname}?${createQueryString("type", value)}`); }} options={toggleGroupOptions} />
}> {searchParams?.get("type") === "team" && (!isOrgAndPrivate || isOrgAdminOrOwner) ? ( ) : ( )} ); } AvailabilityPage.PageWrapper = PageWrapper;