import { useEffect } from "react"; import { shallow } from "zustand/shallow"; import type { IFromUser, IToUser } from "@calcom/core/getUserAvailability"; import type { Dayjs } from "@calcom/dayjs"; import dayjs from "@calcom/dayjs"; import { useEmbedStyles } from "@calcom/embed-core/embed-iframe"; import { useBookerStore } from "@calcom/features/bookings/Booker/store"; import { getAvailableDatesInMonth } from "@calcom/features/calendars/lib/getAvailableDatesInMonth"; import classNames from "@calcom/lib/classNames"; import { daysInMonth, yyyymmdd } from "@calcom/lib/date-fns"; import { useLocale } from "@calcom/lib/hooks/useLocale"; import { weekdayNames } from "@calcom/lib/weekday"; import { Button, SkeletonText } from "@calcom/ui"; export type DatePickerProps = { /** which day of the week to render the calendar. Usually Sunday (=0) or Monday (=1) - default: Sunday */ weekStart?: 0 | 1 | 2 | 3 | 4 | 5 | 6; /** Fires whenever a selected date is changed. */ onChange: (date: Dayjs | null) => void; /** Fires when the month is changed. */ onMonthChange?: (date: Dayjs) => void; /** which date or dates are currently selected (not tracked from here) */ selected?: Dayjs | Dayjs[] | null; /** defaults to current date. */ minDate?: Date; /** Furthest date selectable in the future, default = UNLIMITED */ maxDate?: Date; /** locale, any IETF language tag, e.g. "hu-HU" - defaults to Browser settings */ locale: string; /** Defaults to [], which dates are not bookable. Array of valid dates like: ["2022-04-23", "2022-04-24"] */ excludedDates?: string[]; /** defaults to all, which dates are bookable (inverse of excludedDates) */ includedDates?: string[]; /** allows adding classes to the container */ className?: string; /** Shows a small loading spinner next to the month name */ isPending?: boolean; /** used to query the multiple selected dates */ eventSlug?: string; /** To identify days that are not available and should display OOO and redirect if toUser exists */ slots?: Record< string, { time: string; userIds?: number[]; away?: boolean; fromUser?: IFromUser; toUser?: IToUser; reason?: string; emoji?: string; }[] >; }; export const Day = ({ date, active, disabled, away, emoji, customClassName, ...props }: JSX.IntrinsicElements["button"] & { active: boolean; date: Dayjs; away?: boolean; emoji?: string | null; customClassName?: { dayContainer?: string; dayActive?: string; }; }) => { const { t } = useLocale(); const enabledDateButtonEmbedStyles = useEmbedStyles("enabledDateButton"); const disabledDateButtonEmbedStyles = useEmbedStyles("disabledDateButton"); return ( ); }; const NoAvailabilityOverlay = ({ month, nextMonthButton, }: { month: string | null; nextMonthButton: () => void; }) => { const { t } = useLocale(); return (

{t("no_availability_in_month", { month: month })}

); }; const ReschedulingNotPossibleOverlay = () => { const { t } = useLocale(); return (

{t("rescheduling_not_possible")}

); }; const Days = ({ minDate, excludedDates = [], browsingDate, weekStart, DayComponent = Day, selected, month, nextMonthButton, eventSlug, slots, customClassName, isBookingInPast, ...props }: Omit & { DayComponent?: React.FC>; browsingDate: Dayjs; weekStart: number; month: string | null; nextMonthButton: () => void; customClassName?: { datePickerDate?: string; datePickerDateActive?: string; }; scrollToTimeSlots?: () => void; isBookingInPast: boolean; }) => { // Create placeholder elements for empty days in first week const weekdayOfFirst = browsingDate.date(1).day(); const includedDates = getAvailableDatesInMonth({ browsingDate: browsingDate.toDate(), minDate, includedDates: props.includedDates, }); const days: (Dayjs | null)[] = Array((weekdayOfFirst - weekStart + 7) % 7).fill(null); for (let day = 1, dayCount = daysInMonth(browsingDate); day <= dayCount; day++) { const date = browsingDate.set("date", day); days.push(date); } const [selectedDatesAndTimes] = useBookerStore((state) => [state.selectedDatesAndTimes], shallow); const isActive = (day: dayjs.Dayjs) => { // for selecting a range of dates if (Array.isArray(selected)) { return Array.isArray(selected) && selected?.some((e) => yyyymmdd(e) === yyyymmdd(day)); } if (selected && yyyymmdd(selected) === yyyymmdd(day)) { return true; } // for selecting multiple dates for an event if ( eventSlug && selectedDatesAndTimes && selectedDatesAndTimes[eventSlug as string] && Object.keys(selectedDatesAndTimes[eventSlug as string]).length > 0 ) { return Object.keys(selectedDatesAndTimes[eventSlug as string]).some((date) => { return yyyymmdd(dayjs(date)) === yyyymmdd(day); }); } return false; }; const daysToRenderForTheMonth = days.map((day) => { if (!day) return { day: null, disabled: true }; const dateKey = yyyymmdd(day); const oooInfo = slots && slots?.[dateKey] ? slots?.[dateKey]?.find((slot) => slot.away) : null; const included = includedDates?.includes(dateKey); const excluded = excludedDates.includes(dateKey); const isOOOAllDay = !!(slots && slots[dateKey] && slots[dateKey].every((slot) => slot.away)); const away = isOOOAllDay; const disabled = away ? !oooInfo?.toUser : !included || excluded; return { day: day, disabled, away, emoji: oooInfo?.emoji, }; }); /** * Takes care of selecting a valid date in the month if the selected date is not available in the month */ const useHandleInitialDateSelection = () => { // Let's not do something for now in case of multiple selected dates as behaviour is unclear and it's not needed at the moment if (selected instanceof Array) { return; } const firstAvailableDateOfTheMonth = daysToRenderForTheMonth.find((day) => !day.disabled)?.day; const isSelectedDateAvailable = selected ? daysToRenderForTheMonth.some(({ day, disabled }) => { if (day && yyyymmdd(day) === yyyymmdd(selected) && !disabled) return true; }) : false; if (!isSelectedDateAvailable && firstAvailableDateOfTheMonth) { // If selected date not available in the month, select the first available date of the month props.onChange(firstAvailableDateOfTheMonth); } if (isSelectedDateAvailable) { props.onChange(dayjs(selected)); } if (!firstAvailableDateOfTheMonth) { props.onChange(null); } }; useEffect(useHandleInitialDateSelection); return ( <> {daysToRenderForTheMonth.map(({ day, disabled, away, emoji }, idx) => (
{day === null ? (
) : props.isPending ? ( ) : ( { props.onChange(day); props?.scrollToTimeSlots?.(); }} disabled={disabled} active={isActive(day)} away={away} emoji={emoji} /> )}
))} {isBookingInPast && } {!props.isPending && !isBookingInPast && includedDates && includedDates?.length === 0 && ( )} ); }; const DatePicker = ({ weekStart = 0, className, locale, selected, onMonthChange, slots, customClassNames, includedDates, ...passThroughProps }: DatePickerProps & Partial> & { customClassNames?: { datePickerTitle?: string; datePickerDays?: string; datePickersDates?: string; datePickerDatesActive?: string; datePickerToggle?: string; }; scrollToTimeSlots?: () => void; }) => { const browsingDate = passThroughProps.browsingDate || dayjs().startOf("month"); const { i18n } = useLocale(); const bookingData = useBookerStore((state) => state.bookingData); const isBookingInPast = bookingData ? new Date(bookingData.endTime) < new Date() : false; const changeMonth = (newMonth: number) => { if (onMonthChange) { onMonthChange(browsingDate.add(newMonth, "month")); } }; const month = browsingDate ? new Intl.DateTimeFormat(i18n.language, { month: "long" }).format( new Date(browsingDate.year(), browsingDate.month()) ) : null; return (
{browsingDate ? ( <> {month} {" "} {browsingDate.format("YYYY")} ) : ( )}
{weekdayNames(locale, weekStart, "short").map((weekDay) => (
{weekDay}
))}
changeMonth(+1)} slots={!isBookingInPast ? slots : {}} includedDates={!isBookingInPast ? includedDates : []} isBookingInPast={isBookingInPast} />
); }; export default DatePicker;