import { CSSProperties, DefineComponent, defineComponent, inject, nextTick } from 'vue'; import moment from 'moment'; import omit from 'lodash-es/omit'; import MonthCalendar from '../vc-calendar/src/MonthCalendar'; import VcDatePicker from '../vc-calendar/src/Picker'; import classNames from '../_util/classNames'; import CloseCircleFilled from '@ant-design/icons-vue/CloseCircleFilled'; import CalendarOutlined from '@ant-design/icons-vue/CalendarOutlined'; import { defaultConfigProvider } from '../config-provider'; import interopDefault from '../_util/interopDefault'; import BaseMixin from '../_util/BaseMixin'; import PropTypes from '../_util/vue-types'; import { hasProp, getOptionProps, getComponent, isValidElement } from '../_util/props-util'; import { cloneElement } from '../_util/vnode'; import { formatDate } from './utils'; import { getDataAndAriaProps } from '../_util/util'; export interface PickerProps { value?: moment.Moment; open?: boolean; prefixCls?: string; } export interface PickerState { sOpen?: boolean; sValue?: moment.Moment | null; showDate?: moment.Moment | null; } export default function createPicker

( TheCalendar: DefineComponent

, props: any, name: string, ): any { return defineComponent({ name, mixins: [BaseMixin], inheritAttrs: false, props: { ...props, allowClear: PropTypes.looseBool.def(true), showToday: PropTypes.looseBool.def(true), }, setup() { return { configProvider: inject('configProvider', defaultConfigProvider), input: undefined, sPrefixCls: undefined, }; }, data(): PickerState { const value = this.value || this.defaultValue; return { sValue: value, showDate: value, sOpen: !!this.open, }; }, watch: { open(val) { const props: PickerProps = getOptionProps(this); const state: PickerState = {}; state.sOpen = val; if ('value' in props && !val && props.value !== this.showDate) { state.showDate = props.value; } this.setState(state); }, value(val) { const state: PickerState = {}; state.sValue = val; if (val !== this.sValue) { state.showDate = val; } this.setState(state); }, sOpen(val, oldVal) { nextTick(() => { if (!hasProp(this, 'open') && oldVal && !val) { this.focus(); } }); }, }, methods: { saveInput(node: any) { this.input = node; }, clearSelection(e: MouseEvent) { e.preventDefault(); e.stopPropagation(); this.handleChange(null); }, handleChange(value: moment.Moment | null) { if (!hasProp(this, 'value')) { this.setState({ sValue: value, showDate: value, }); } this.$emit('change', value, formatDate(value, this.format)); }, handleCalendarChange(value: moment.Moment) { this.setState({ showDate: value }); }, handleOpenChange(open: boolean) { const props = getOptionProps(this); if (!('open' in props)) { this.setState({ sOpen: open }); } this.$emit('openChange', open); }, focus() { this.input?.focus(); }, blur() { this.input?.blur(); }, renderFooter(...args: any[]) { const { $slots, sPrefixCls: prefixCls } = this; const renderExtraFooter: Function = this.renderExtraFooter || $slots.renderExtraFooter; return renderExtraFooter ? (

) : null; }, onMouseEnter(e: MouseEvent) { this.$emit('mouseenter', e); }, onMouseLeave(e: MouseEvent) { this.$emit('mouseleave', e); }, }, render() { const { $slots } = this; const { sValue: value, showDate, sOpen: open } = this.$data; let suffixIcon = getComponent(this, 'suffixIcon'); suffixIcon = Array.isArray(suffixIcon) ? suffixIcon[0] : suffixIcon; const props: any = omit({ ...getOptionProps(this), ...this.$attrs }, ['onChange']); const { prefixCls: customizePrefixCls, locale, localeCode, inputReadOnly } = props; const getPrefixCls = this.configProvider.getPrefixCls; const prefixCls = getPrefixCls('calendar', customizePrefixCls); this.sPrefixCls = prefixCls; const dateRender = props.dateRender || $slots.dateRender; const monthCellContentRender = props.monthCellContentRender || $slots.monthCellContentRender; const placeholder = 'placeholder' in props ? props.placeholder : locale.lang.placeholder; const disabledTime = props.showTime ? props.disabledTime : null; const calendarClassName = classNames({ [`${prefixCls}-time`]: props.showTime, [`${prefixCls}-month`]: (MonthCalendar as any) === TheCalendar, }); if (value && localeCode) { value.locale(localeCode); } const pickerProps: any = {}; const calendarProps: any = {}; const pickerStyle: CSSProperties = {}; if (props.showTime) { // fix https://github.com/ant-design/ant-design/issues/1902 calendarProps.onSelect = this.handleChange; pickerStyle.minWidth = '195px'; } else { pickerProps.onChange = this.handleChange; } if ('mode' in props) { calendarProps.mode = props.mode; } const theCalendarProps = { ...calendarProps, disabledDate: props.disabledDate, disabledTime, locale: locale.lang, timePicker: props.timePicker, defaultValue: props.defaultPickerValue || interopDefault(moment)(), dateInputPlaceholder: placeholder, prefixCls, dateRender, format: props.format, showToday: props.showToday, monthCellContentRender, renderFooter: this.renderFooter, value: showDate, inputReadOnly, onOk: props.onOk, onPanelChange: props.onPanelChange, onChange: this.handleCalendarChange, class: calendarClassName, }; const calendar = ; const clearIcon = !props.disabled && props.allowClear && value ? ( ) : null; const inputIcon = (suffixIcon && (isValidElement(suffixIcon) ? ( cloneElement(suffixIcon, { class: `${prefixCls}-picker-icon`, }) ) : ( {suffixIcon} ))) || ; const input = ({ value: inputValue }) => (
{clearIcon} {inputIcon}
); const vcDatePickerProps = { ...props, ...pickerProps, calendar, value, prefixCls: `${prefixCls}-picker-container`, open, onOpenChange: this.handleOpenChange, style: props.popupStyle, }; return ( ); }, }); }