import useMergedState from '../_util/hooks/useMergedState'; import padStart from 'lodash-es/padStart'; import { PickerPanel } from '../vc-picker'; import type { Locale } from '../vc-picker/interface'; import type { GenerateConfig } from '../vc-picker/generate'; import type { PickerPanelBaseProps as RCPickerPanelBaseProps, PickerPanelDateProps as RCPickerPanelDateProps, PickerPanelTimeProps as RCPickerPanelTimeProps, } from '../vc-picker/PickerPanel'; import { useLocaleReceiver } from '../locale-provider/LocaleReceiver'; import enUS from './locale/en_US'; import CalendarHeader from './Header'; import type { VueNode } from '../_util/type'; import type { App } from 'vue'; import { computed, defineComponent, toRef } from 'vue'; import useConfigInject from '../_util/hooks/useConfigInject'; import classNames from '../_util/classNames'; type InjectDefaultProps = Omit< Props, 'locale' | 'generateConfig' | 'prevIcon' | 'nextIcon' | 'superPrevIcon' | 'superNextIcon' > & { locale?: typeof enUS; size?: 'large' | 'default' | 'small'; }; // Picker Props export type PickerPanelBaseProps = InjectDefaultProps>; export type PickerPanelDateProps = InjectDefaultProps>; export type PickerPanelTimeProps = InjectDefaultProps>; export type PickerProps = | PickerPanelBaseProps | PickerPanelDateProps | PickerPanelTimeProps; export type CalendarMode = 'year' | 'month'; export type HeaderRender = (config: { value: DateType; type: CalendarMode; onChange: (date: DateType) => void; onTypeChange: (type: CalendarMode) => void; }) => VueNode; type CustomRenderType = (config: { current: DateType }) => VueNode; export interface CalendarProps { prefixCls?: string; locale?: typeof enUS; validRange?: [DateType, DateType]; disabledDate?: (date: DateType) => boolean; dateFullCellRender?: CustomRenderType; dateCellRender?: CustomRenderType; monthFullCellRender?: CustomRenderType; monthCellRender?: CustomRenderType; headerRender?: HeaderRender; value?: DateType | string; defaultValue?: DateType | string; mode?: CalendarMode; fullscreen?: boolean; onChange?: (date: DateType | string) => void; 'onUpdate:value'?: (date: DateType | string) => void; onPanelChange?: (date: DateType | string, mode: CalendarMode) => void; onSelect?: (date: DateType | string) => void; valueFormat?: string; } function generateCalendar< DateType, Props extends CalendarProps = CalendarProps, >(generateConfig: GenerateConfig) { function isSameYear(date1: DateType, date2: DateType) { return date1 && date2 && generateConfig.getYear(date1) === generateConfig.getYear(date2); } function isSameMonth(date1: DateType, date2: DateType) { return ( isSameYear(date1, date2) && generateConfig.getMonth(date1) === generateConfig.getMonth(date2) ); } function isSameDate(date1: DateType, date2: DateType) { return ( isSameMonth(date1, date2) && generateConfig.getDate(date1) === generateConfig.getDate(date2) ); } const Calendar = defineComponent({ name: 'ACalendar', inheritAttrs: false, props: [ 'prefixCls', 'locale', 'validRange', 'disabledDate', 'dateFullCellRender', 'dateCellRender', 'monthFullCellRender', 'monthCellRender', 'headerRender', 'value', 'defaultValue', 'mode', 'fullscreen', 'onChange', 'onPanelChange', 'onSelect', 'valueFormat', ] as any, slots: [ 'dateFullCellRender', 'dateCellRender', 'monthFullCellRender', 'monthCellRender', 'headerRender', ], setup(props, { emit, slots, attrs }) { const { prefixCls, direction } = useConfigInject('picker', props); const calendarPrefixCls = computed(() => `${prefixCls.value}-calendar`); const maybeToString = (date: DateType) => { return props.valueFormat ? generateConfig.toString(date, props.valueFormat) : date; }; const value = computed(() => { if (props.value) { return props.valueFormat ? (generateConfig.toDate(props.value, props.valueFormat) as DateType) : (props.value as DateType); } return (props.value === '' ? undefined : props.value) as DateType; }); const defaultValue = computed(() => { if (props.defaultValue) { return props.valueFormat ? (generateConfig.toDate(props.defaultValue, props.valueFormat) as DateType) : (props.defaultValue as DateType); } return (props.defaultValue === '' ? undefined : props.defaultValue) as DateType; }); // Value const [mergedValue, setMergedValue] = useMergedState( () => value.value || generateConfig.getNow(), { defaultValue: defaultValue.value, value, }, ); // Mode const [mergedMode, setMergedMode] = useMergedState('month', { value: toRef(props, 'mode'), }); const panelMode = computed(() => (mergedMode.value === 'year' ? 'month' : 'date')); const mergedDisabledDate = computed(() => { return (date: DateType) => { const notInRange = props.validRange ? generateConfig.isAfter(props.validRange[0], date) || generateConfig.isAfter(date, props.validRange[1]) : false; return notInRange || !!props.disabledDate?.(date); }; }); // ====================== Events ====================== const triggerPanelChange = (date: DateType, newMode: CalendarMode) => { emit('panelChange', maybeToString(date), newMode); }; const triggerChange = (date: DateType) => { setMergedValue(date); if (!isSameDate(date, mergedValue.value)) { // Trigger when month panel switch month if ( (panelMode.value === 'date' && !isSameMonth(date, mergedValue.value)) || (panelMode.value === 'month' && !isSameYear(date, mergedValue.value)) ) { triggerPanelChange(date, mergedMode.value); } const val = maybeToString(date); emit('update:value', val); emit('change', val); } }; const triggerModeChange = (newMode: CalendarMode) => { setMergedMode(newMode); triggerPanelChange(mergedValue.value, newMode); }; const onInternalSelect = (date: DateType) => { triggerChange(date); emit('select', maybeToString(date)); }; // ====================== Locale ====================== const defaultLocale = computed(() => { const { locale } = props; const result = { ...enUS, ...locale, }; result.lang = { ...result.lang, ...(locale || {}).lang, }; return result; }); const [mergedLocale] = useLocaleReceiver('Calendar', defaultLocale) as [typeof defaultLocale]; return () => { const today = generateConfig.getNow(); const { dateFullCellRender = slots?.dateFullCellRender, dateCellRender = slots?.dateCellRender, monthFullCellRender = slots?.monthFullCellRender, monthCellRender = slots?.monthCellRender, headerRender = slots?.headerRender, fullscreen = true, validRange, } = props; // ====================== Render ====================== const dateRender = ({ current: date }) => { if (dateFullCellRender) { return dateFullCellRender({ current: date }); } return (
{padStart(String(generateConfig.getDate(date)), 2, '0')}
{dateCellRender && dateCellRender({ current: date })}
); }; const monthRender = ({ current: date }, locale: Locale) => { if (monthFullCellRender) { return monthFullCellRender({ current: date }); } const months = locale.shortMonths || generateConfig.locale.getShortMonths!(locale.locale); return (
{months[generateConfig.getMonth(date)]}
{monthCellRender && monthCellRender({ current: date })}
); }; return (
{headerRender ? ( headerRender({ value: mergedValue.value, type: mergedMode.value, onChange: onInternalSelect, onTypeChange: triggerModeChange, }) ) : ( )} monthRender(obj, mergedLocale.value.lang)} onSelect={onInternalSelect} mode={panelMode.value} picker={panelMode.value} disabledDate={mergedDisabledDate.value} hideHeader />
); }; }, }); Calendar.install = function (app: App) { app.component(Calendar.name, Calendar); return app; }; return Calendar; } export default generateCalendar;