2021-09-25 08:51:32 +00:00
|
|
|
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 RCPicker from '../../vc-picker';
|
|
|
|
import type { PanelMode, PickerMode } from '../../vc-picker/interface';
|
|
|
|
import type { GenerateConfig } from '../../vc-picker/generate/index';
|
|
|
|
import enUS from '../locale/en_US';
|
2022-05-15 06:48:46 +00:00
|
|
|
import { getPlaceholder, transPlacement2DropdownAlign } from '../util';
|
2021-09-25 08:51:32 +00:00
|
|
|
import { useLocaleReceiver } from '../../locale-provider/LocaleReceiver';
|
|
|
|
import { getTimeProps, Components } from '.';
|
2022-07-06 11:10:14 +00:00
|
|
|
import { computed, defineComponent, ref } from 'vue';
|
2021-09-25 08:51:32 +00:00
|
|
|
import useConfigInject from '../../_util/hooks/useConfigInject';
|
|
|
|
import classNames from '../../_util/classNames';
|
2022-01-21 13:58:10 +00:00
|
|
|
import type { CommonProps, DatePickerProps } from './props';
|
2021-09-25 08:51:32 +00:00
|
|
|
import { commonProps, datePickerProps } from './props';
|
|
|
|
|
|
|
|
import devWarning from '../../vc-util/devWarning';
|
2022-05-15 06:48:46 +00:00
|
|
|
import { FormItemInputContext, useInjectFormItemContext } from '../../form/FormItemContext';
|
|
|
|
import { getMergedStatus, getStatusClassNames } from '../../_util/statusUtils';
|
2021-09-25 08:51:32 +00:00
|
|
|
|
|
|
|
export default function generateSinglePicker<DateType, ExtraProps = {}>(
|
|
|
|
generateConfig: GenerateConfig<DateType>,
|
|
|
|
extraProps: ExtraProps,
|
|
|
|
) {
|
|
|
|
function getPicker(picker?: PickerMode, displayName?: string) {
|
2022-01-21 13:58:10 +00:00
|
|
|
const comProps = {
|
|
|
|
...commonProps<DateType>(),
|
|
|
|
...datePickerProps<DateType>(),
|
|
|
|
...extraProps,
|
|
|
|
};
|
2021-09-25 08:51:32 +00:00
|
|
|
return defineComponent({
|
2022-09-26 13:33:41 +00:00
|
|
|
compatConfig: { MODE: 3 },
|
2021-09-25 08:51:32 +00:00
|
|
|
name: displayName,
|
|
|
|
inheritAttrs: false,
|
2022-01-21 13:58:10 +00:00
|
|
|
props: comProps,
|
2021-09-25 08:51:32 +00:00
|
|
|
slots: [
|
|
|
|
'suffixIcon',
|
|
|
|
// 'clearIcon',
|
2022-03-12 01:56:32 +00:00
|
|
|
'prevIcon',
|
|
|
|
'nextIcon',
|
|
|
|
'superPrevIcon',
|
|
|
|
'superNextIcon',
|
2021-09-25 08:51:32 +00:00
|
|
|
// 'panelRender',
|
|
|
|
'dateRender',
|
|
|
|
'renderExtraFooter',
|
|
|
|
'monthCellRender',
|
|
|
|
],
|
2022-01-21 13:58:10 +00:00
|
|
|
setup(_props, { slots, expose, attrs, emit }) {
|
|
|
|
// 兼容 vue 3.2.7
|
|
|
|
const props = _props as unknown as CommonProps<DateType> &
|
|
|
|
DatePickerProps<DateType> &
|
|
|
|
ExtraProps;
|
2021-09-25 08:51:32 +00:00
|
|
|
const formItemContext = useInjectFormItemContext();
|
2022-05-15 06:48:46 +00:00
|
|
|
const formItemInputContext = FormItemInputContext.useInject();
|
2021-09-25 08:51:32 +00:00
|
|
|
devWarning(
|
|
|
|
!(props.monthCellContentRender || slots.monthCellContentRender),
|
|
|
|
'DatePicker',
|
|
|
|
'`monthCellContentRender` is deprecated. Please use `monthCellRender"` instead.',
|
|
|
|
);
|
|
|
|
|
|
|
|
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 maybeToString = (date: DateType) => {
|
|
|
|
return props.valueFormat ? generateConfig.toString(date, props.valueFormat) : date;
|
|
|
|
};
|
|
|
|
const onChange = (date: DateType, dateString: string) => {
|
|
|
|
const value = maybeToString(date);
|
|
|
|
emit('update:value', value);
|
|
|
|
emit('change', value, dateString);
|
|
|
|
formItemContext.onFieldChange();
|
|
|
|
};
|
|
|
|
const onOpenChange = (open: boolean) => {
|
|
|
|
emit('update:open', open);
|
|
|
|
emit('openChange', open);
|
|
|
|
};
|
2022-04-01 13:31:20 +00:00
|
|
|
const onFocus = (e: FocusEvent) => {
|
|
|
|
emit('focus', e);
|
2021-09-25 08:51:32 +00:00
|
|
|
};
|
2022-04-01 13:31:20 +00:00
|
|
|
const onBlur = (e: FocusEvent) => {
|
|
|
|
emit('blur', e);
|
2021-09-25 08:51:32 +00:00
|
|
|
formItemContext.onFieldBlur();
|
|
|
|
};
|
|
|
|
const onPanelChange = (date: DateType, mode: PanelMode | null) => {
|
|
|
|
const value = maybeToString(date);
|
|
|
|
emit('panelChange', value, mode);
|
|
|
|
};
|
|
|
|
const onOk = (date: DateType) => {
|
|
|
|
const value = maybeToString(date);
|
|
|
|
emit('ok', value);
|
|
|
|
};
|
|
|
|
|
|
|
|
const [contextLocale] = useLocaleReceiver('DatePicker', enUS);
|
|
|
|
|
|
|
|
const value = computed(() => {
|
|
|
|
if (props.value) {
|
|
|
|
return props.valueFormat
|
|
|
|
? generateConfig.toDate(props.value as string | DateType, props.valueFormat)
|
|
|
|
: props.value;
|
|
|
|
}
|
2022-03-12 08:28:36 +00:00
|
|
|
return (props.value === '' ? undefined : props.value) as DateType;
|
2021-09-25 08:51:32 +00:00
|
|
|
});
|
|
|
|
const defaultValue = computed(() => {
|
|
|
|
if (props.defaultValue) {
|
|
|
|
return props.valueFormat
|
|
|
|
? generateConfig.toDate(props.defaultValue as string | DateType, props.valueFormat)
|
|
|
|
: props.defaultValue;
|
|
|
|
}
|
2022-03-12 08:28:36 +00:00
|
|
|
return (props.defaultValue === '' ? undefined : props.defaultValue) as DateType;
|
2021-09-25 08:51:32 +00:00
|
|
|
});
|
|
|
|
const defaultPickerValue = computed(() => {
|
|
|
|
if (props.defaultPickerValue) {
|
|
|
|
return props.valueFormat
|
|
|
|
? generateConfig.toDate(
|
|
|
|
props.defaultPickerValue as string | DateType,
|
|
|
|
props.valueFormat,
|
|
|
|
)
|
|
|
|
: props.defaultPickerValue;
|
|
|
|
}
|
2022-03-12 08:28:36 +00:00
|
|
|
return (
|
|
|
|
props.defaultPickerValue === '' ? undefined : props.defaultPickerValue
|
|
|
|
) as DateType;
|
2021-09-25 08:51:32 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
const locale = { ...contextLocale.value, ...props.locale };
|
|
|
|
const p = { ...props, ...attrs };
|
|
|
|
const {
|
|
|
|
bordered = true,
|
|
|
|
placeholder,
|
|
|
|
suffixIcon = slots.suffixIcon?.(),
|
|
|
|
showToday = true,
|
|
|
|
transitionName,
|
|
|
|
allowClear = true,
|
|
|
|
dateRender = slots.dateRender,
|
|
|
|
renderExtraFooter = slots.renderExtraFooter,
|
|
|
|
monthCellRender = slots.monthCellRender ||
|
|
|
|
(props as any).monthCellContentRender ||
|
|
|
|
slots.monthCellContentRender,
|
|
|
|
clearIcon = slots.clearIcon?.(),
|
|
|
|
id = formItemContext.id.value,
|
|
|
|
...restProps
|
|
|
|
} = p;
|
|
|
|
const showTime = p.showTime === '' ? true : p.showTime;
|
|
|
|
const { format } = p as any;
|
|
|
|
|
|
|
|
let additionalOverrideProps: any = {};
|
|
|
|
if (picker) {
|
|
|
|
additionalOverrideProps.picker = picker;
|
|
|
|
}
|
|
|
|
const mergedPicker = picker || p.picker || 'date';
|
|
|
|
|
|
|
|
additionalOverrideProps = {
|
|
|
|
...additionalOverrideProps,
|
|
|
|
...(showTime
|
|
|
|
? getTimeProps({
|
|
|
|
format,
|
|
|
|
picker: mergedPicker,
|
|
|
|
...(typeof showTime === 'object' ? showTime : {}),
|
|
|
|
})
|
|
|
|
: {}),
|
|
|
|
...(mergedPicker === 'time'
|
|
|
|
? getTimeProps({ format, ...restProps, picker: mergedPicker })
|
|
|
|
: {}),
|
|
|
|
};
|
|
|
|
const pre = prefixCls.value;
|
2022-05-15 06:48:46 +00:00
|
|
|
const suffixNode = (
|
|
|
|
<>
|
|
|
|
{suffixIcon || (picker === 'time' ? <ClockCircleOutlined /> : <CalendarOutlined />)}
|
|
|
|
{formItemInputContext.hasFeedback && formItemInputContext.feedbackIcon}
|
|
|
|
</>
|
|
|
|
);
|
2021-09-25 08:51:32 +00:00
|
|
|
return (
|
|
|
|
<RCPicker
|
|
|
|
monthCellRender={monthCellRender}
|
|
|
|
dateRender={dateRender}
|
|
|
|
renderExtraFooter={renderExtraFooter}
|
|
|
|
ref={pickerRef}
|
|
|
|
placeholder={getPlaceholder(mergedPicker, locale, placeholder)}
|
2022-05-15 06:48:46 +00:00
|
|
|
suffixIcon={suffixNode}
|
|
|
|
dropdownAlign={transPlacement2DropdownAlign(direction.value, props.placement)}
|
2021-09-25 08:51:32 +00:00
|
|
|
clearIcon={clearIcon || <CloseCircleFilled />}
|
|
|
|
allowClear={allowClear}
|
|
|
|
transitionName={transitionName || `${rootPrefixCls.value}-slide-up`}
|
|
|
|
{...restProps}
|
|
|
|
{...additionalOverrideProps}
|
|
|
|
id={id}
|
|
|
|
picker={mergedPicker}
|
|
|
|
value={value.value}
|
|
|
|
defaultValue={defaultValue.value}
|
|
|
|
defaultPickerValue={defaultPickerValue.value}
|
|
|
|
showToday={showToday}
|
|
|
|
locale={locale!.lang}
|
|
|
|
class={classNames(
|
|
|
|
{
|
|
|
|
[`${pre}-${size.value}`]: size.value,
|
|
|
|
[`${pre}-borderless`]: !bordered,
|
|
|
|
},
|
2022-05-15 06:48:46 +00:00
|
|
|
getStatusClassNames(
|
|
|
|
pre,
|
|
|
|
getMergedStatus(formItemInputContext.status, props.status),
|
|
|
|
formItemInputContext.hasFeedback,
|
|
|
|
),
|
2021-09-25 08:51:32 +00:00
|
|
|
attrs.class,
|
|
|
|
)}
|
|
|
|
prefixCls={pre}
|
|
|
|
getPopupContainer={attrs.getCalendarContainer || getPopupContainer.value}
|
|
|
|
generateConfig={generateConfig}
|
2022-03-12 01:56:32 +00:00
|
|
|
prevIcon={slots.prevIcon?.() || <span class={`${pre}-prev-icon`} />}
|
|
|
|
nextIcon={slots.nextIcon?.() || <span class={`${pre}-next-icon`} />}
|
|
|
|
superPrevIcon={slots.superPrevIcon?.() || <span class={`${pre}-super-prev-icon`} />}
|
|
|
|
superNextIcon={slots.superNextIcon?.() || <span class={`${pre}-super-next-icon`} />}
|
2021-09-25 08:51:32 +00:00
|
|
|
components={Components}
|
|
|
|
direction={direction.value}
|
|
|
|
onChange={onChange}
|
|
|
|
onOpenChange={onOpenChange}
|
|
|
|
onFocus={onFocus}
|
|
|
|
onBlur={onBlur}
|
|
|
|
onPanelChange={onPanelChange}
|
|
|
|
onOk={onOk}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
const DatePicker = getPicker(undefined, 'ADatePicker');
|
|
|
|
const WeekPicker = getPicker('week', 'AWeekPicker');
|
|
|
|
const MonthPicker = getPicker('month', 'AMonthPicker');
|
|
|
|
const YearPicker = getPicker('year', 'AYearPicker');
|
|
|
|
const TimePicker = getPicker('time', 'TimePicker'); // 给独立组件 TimePicker 使用,此处名称不用更改
|
|
|
|
const QuarterPicker = getPicker('quarter', 'AQuarterPicker');
|
|
|
|
|
|
|
|
return {
|
|
|
|
DatePicker,
|
|
|
|
WeekPicker,
|
|
|
|
MonthPicker,
|
|
|
|
YearPicker,
|
|
|
|
TimePicker,
|
|
|
|
QuarterPicker,
|
|
|
|
};
|
|
|
|
}
|