import CalendarOutlined from '@ant-design/icons-vue/CalendarOutlined'; import ClockCircleOutlined from '@ant-design/icons-vue/ClockCircleOutlined'; import CloseCircleFilled from '@ant-design/icons-vue/CloseCircleFilled'; import SwapRightOutlined from '@ant-design/icons-vue/SwapRightOutlined'; import { RangePicker as VCRangePicker } from '../../vc-picker'; import type { GenerateConfig } from '../../vc-picker/generate/index'; import enUS from '../locale/en_US'; import { useLocaleReceiver } from '../../locale-provider/LocaleReceiver'; import { getRangePlaceholder, transPlacement2DropdownAlign } from '../util'; import { getTimeProps, Components } from '.'; import { computed, defineComponent, ref } from 'vue'; import useConfigInject from '../../_util/hooks/useConfigInject'; import classNames from '../../_util/classNames'; import type { CommonProps, RangePickerProps } from './props'; import { commonProps, rangePickerProps } from './props'; import type { PanelMode, RangeValue } from '../../vc-picker/interface'; import type { RangePickerSharedProps } from '../../vc-picker/RangePicker'; import devWarning from '../../vc-util/devWarning'; import { FormItemInputContext, useInjectFormItemContext } from '../../form/FormItemContext'; import omit from '../../_util/omit'; import { getMergedStatus, getStatusClassNames } from '../../_util/statusUtils'; export default function generateRangePicker( generateConfig: GenerateConfig, extraProps: ExtraProps, ) { const RangePicker = defineComponent({ compatConfig: { MODE: 3 }, name: 'ARangePicker', inheritAttrs: false, props: { ...commonProps(), ...rangePickerProps(), ...extraProps, }, slots: [ 'suffixIcon', // 'clearIcon', 'prevIcon', 'nextIcon', 'superPrevIcon', 'superNextIcon', // 'panelRender', 'dateRender', 'renderExtraFooter', // 'separator', ], setup(_props, { expose, slots, attrs, emit }) { const props = _props as unknown as CommonProps & RangePickerProps; const formItemContext = useInjectFormItemContext(); const formItemInputContext = FormItemInputContext.useInject(); devWarning( !attrs.getCalendarContainer, 'DatePicker', '`getCalendarContainer` is deprecated. Please use `getPopupContainer"` instead.', ); const { prefixCls, direction, getPopupContainer, size, rootPrefixCls } = useConfigInject( 'picker', props, ); const pickerRef = ref(); expose({ focus: () => { pickerRef.value?.focus(); }, blur: () => { pickerRef.value?.blur(); }, }); const maybeToStrings = (dates: DateType[]) => { return props.valueFormat ? generateConfig.toString(dates, props.valueFormat) : dates; }; const onChange = (dates: [DateType, DateType], dateStrings: [string, string]) => { const values = maybeToStrings(dates); emit('update:value', values); emit('change', values, dateStrings); formItemContext.onFieldChange(); }; const onOpenChange = (open: boolean) => { emit('update:open', open); emit('openChange', open); }; const onFocus = (e: FocusEvent) => { emit('focus', e); }; const onBlur = (e: FocusEvent) => { emit('blur', e); formItemContext.onFieldBlur(); }; const onPanelChange = (dates: RangeValue, modes: [PanelMode, PanelMode]) => { const values = maybeToStrings(dates); emit('panelChange', values, modes); }; const onOk = (dates: DateType[]) => { const value = maybeToStrings(dates); emit('ok', value); }; const onCalendarChange: RangePickerSharedProps['onCalendarChange'] = ( dates: [DateType, DateType], dateStrings: [string, string], info, ) => { const values = maybeToStrings(dates); emit('calendarChange', values, dateStrings, info); }; const [contextLocale] = useLocaleReceiver('DatePicker', enUS); const value = computed(() => { if (props.value) { return props.valueFormat ? generateConfig.toDate(props.value, props.valueFormat) : props.value; } return props.value; }); const defaultValue = computed(() => { if (props.defaultValue) { return props.valueFormat ? generateConfig.toDate(props.defaultValue, props.valueFormat) : props.defaultValue; } return props.defaultValue; }); const defaultPickerValue = computed(() => { if (props.defaultPickerValue) { return props.valueFormat ? generateConfig.toDate(props.defaultPickerValue, props.valueFormat) : props.defaultPickerValue; } return props.defaultPickerValue; }); return () => { const locale = { ...contextLocale.value, ...props.locale }; const p = { ...props, ...attrs }; const { prefixCls: customizePrefixCls, bordered = true, placeholder, suffixIcon = slots.suffixIcon?.(), picker = 'date', transitionName, allowClear = true, dateRender = slots.dateRender, renderExtraFooter = slots.renderExtraFooter, separator = slots.separator?.(), clearIcon = slots.clearIcon?.(), id = formItemContext.id.value, ...restProps } = p; delete restProps['onUpdate:value']; delete restProps['onUpdate:open']; const { format, showTime } = p as any; let additionalOverrideProps: any = {}; additionalOverrideProps = { ...additionalOverrideProps, ...(showTime ? getTimeProps({ format, picker, ...showTime }) : {}), ...(picker === 'time' ? getTimeProps({ format, ...omit(restProps, ['disabledTime']), picker }) : {}), }; const pre = prefixCls.value; const suffixNode = ( <> {suffixIcon || (picker === 'time' ? : )} {formItemInputContext.hasFeedback && formItemInputContext.feedbackIcon} ); return ( ) } ref={pickerRef} dropdownAlign={transPlacement2DropdownAlign(direction.value, props.placement)} placeholder={getRangePlaceholder(picker, locale, placeholder as [string, string])} suffixIcon={suffixNode} clearIcon={clearIcon || } allowClear={allowClear} transitionName={transitionName || `${rootPrefixCls.value}-slide-up`} {...restProps} {...additionalOverrideProps} id={id} value={value.value} defaultValue={defaultValue.value} defaultPickerValue={defaultPickerValue.value} picker={picker} class={classNames( { [`${pre}-${size.value}`]: size.value, [`${pre}-borderless`]: !bordered, }, getStatusClassNames( pre, getMergedStatus(formItemInputContext.status, props.status), formItemInputContext.hasFeedback, ), attrs.class, )} locale={locale!.lang} prefixCls={pre} getPopupContainer={attrs.getCalendarContainer || getPopupContainer.value} generateConfig={generateConfig} prevIcon={slots.prevIcon?.() || } nextIcon={slots.nextIcon?.() || } superPrevIcon={slots.superPrevIcon?.() || } superNextIcon={slots.superNextIcon?.() || } components={Components} direction={direction.value} onChange={onChange} onOpenChange={onOpenChange} onFocus={onFocus} onBlur={onBlur} onPanelChange={onPanelChange} onOk={onOk} onCalendarChange={onCalendarChange} /> ); }; }, }); return RangePicker; }