import classNames from "classnames"; import { useEffect, useState } from "react"; import { components, OptionProps, SingleValueProps } from "react-select"; import { useLocale } from "@calcom/lib/hooks/useLocale"; import { DestinationCalendar } from "@calcom/prisma/client"; import { trpc } from "@calcom/trpc/react"; import { Select } from "@calcom/ui"; interface Props { onChange: (value: { externalId: string; integration: string }) => void; isLoading?: boolean; hidePlaceholder?: boolean; /** The external Id of the connected calendar */ destinationCalendar?: DestinationCalendar | null; value: string | undefined; maxWidth?: number; } interface Option { label: string; value: string; subtitle: string; } const SingleValueComponent = ({ ...props }: SingleValueProps) => { const { label, subtitle } = props.data; return ( {label} {subtitle} ); }; const OptionComponent = ({ ...props }: OptionProps) => { const { label } = props.data; return ( {label} ); }; const DestinationCalendarSelector = ({ onChange, isLoading, value, hidePlaceholder, maxWidth, destinationCalendar, }: Props): JSX.Element | null => { const { t } = useLocale(); const query = trpc.viewer.connectedCalendars.useQuery(); const [selectedOption, setSelectedOption] = useState<{ value: string; label: string; subtitle: string; } | null>(null); // Extra styles to show prefixed text in react-select const content = (hidePlaceholder = false) => { if (!hidePlaceholder) { return { alignItems: "center", width: "100%", display: "flex", ":before": { content: `'${t("select_destination_calendar")}:'`, display: "block", marginRight: 8, }, }; } return {}; }; useEffect(() => { const selected = query.data?.connectedCalendars .map((connected) => connected.calendars ?? []) .flat() .find((cal) => cal.externalId === value); if (selected) { const selectedIntegration = query.data?.connectedCalendars.find((integration) => integration.calendars?.some((calendar) => calendar.externalId === selected.externalId) ); setSelectedOption({ value: `${selected.integration}:${selected.externalId}`, label: `${selected.name} ` || "", subtitle: `(${selectedIntegration?.integration.title?.replace(/calendar/i, "")} - ${ selectedIntegration?.primary?.name })`, }); } }, [query.data?.connectedCalendars]); if (!query.data?.connectedCalendars.length) { return null; } const options = query.data.connectedCalendars.map((selectedCalendar) => ({ key: selectedCalendar.credentialId, label: `${selectedCalendar.integration.title?.replace(/calendar/i, "")} (${ selectedCalendar.primary?.name })`, options: (selectedCalendar.calendars ?? []) .filter((cal) => cal.readOnly === false) .map((cal) => ({ label: ` ${cal.name} `, subtitle: `(${selectedCalendar?.integration.title?.replace(/calendar/i, "")} - ${ selectedCalendar?.primary?.name })`, value: `${cal.integration}:${cal.externalId}`, })), })) ?? []; // Get primary calendar, which is shown in the placeholder since this is the calendar that will // be used when no destination calendar is selected. const primaryCalendar = query.data.connectedCalendars.filter((cal) => Boolean(cal.primary))[0]; return ( {t("default_calendar_selected")}{" "} {primaryCalendar?.primary?.externalId && `(${primaryCalendar?.primary?.externalId})`} ) } options={options} styles={{ placeholder: (styles) => ({ ...styles, ...content(hidePlaceholder) }), singleValue: (styles) => ({ ...styles, ...content(hidePlaceholder) }), option: (defaultStyles, state) => ({ ...defaultStyles, backgroundColor: state.isSelected ? state.isFocused ? "var(--brand-color)" : "var(--brand-color)" : state.isFocused ? "var(--brand-color-dark-mode)" : "var(--brand-text-color)", }), control: (defaultStyles) => { return { ...defaultStyles, "@media only screen and (min-width: 640px)": { ...(defaultStyles["@media only screen and (min-width: 640px)"] as object), maxWidth, }, }; }, }} isSearchable={false} className={classNames( "mt-1 mb-2 block w-full min-w-0 flex-1 rounded-none rounded-r-sm border-gray-300 text-sm" )} onChange={(newValue) => { setSelectedOption(newValue); if (!newValue) { return; } /* Split only the first `:`, since Apple uses the full URL as externalId */ const [integration, externalId] = newValue.value.split(/:(.+)/); onChange({ integration, externalId, }); }} isLoading={isLoading} value={selectedOption} components={{ SingleValue: SingleValueComponent, Option: OptionComponent }} isMulti={false} /> ); }; export default DestinationCalendarSelector;
{label}
{subtitle}