| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312 | /* eslint-disable max-len */import { cloneDeep, isObject, set, get } from 'lodash';import { format as formatFn } from 'date-fns';import BaseFoundation, { DefaultAdapter } from '../base/foundation';import { BaseValueType, ValidateStatus, ValueType } from './foundation';import { formatDateValues } from './_utils/formatter';import { getDefaultFormatTokenByType } from './_utils/getDefaultFormatToken';import getInsetInputFormatToken from './_utils/getInsetInputFormatToken';import getInsetInputValueFromInsetInputStr from './_utils/getInsetInputValueFromInsetInputStr';import { strings } from './constants';import getDefaultPickerDate from './_utils/getDefaultPickerDate';import { compatibleParse } from './_utils/parser';import { isValidDate } from './_utils';const KEY_CODE_ENTER = 'Enter';const KEY_CODE_TAB = 'Tab';export type Type = 'date' | 'dateRange' | 'year' | 'month' | 'dateTime' | 'dateTimeRange';export type RangeType = 'rangeStart' | 'rangeEnd';export type PanelType = 'left' | 'right';export interface DateInputEventHandlerProps {    onClick?: (e: any) => void;    onChange?: (value: string, e: any) => void;    onEnterPress?: (e: any) => void;    onBlur?: (e: any) => void;    onFocus?: (e: any, rangeType: RangeType) => void;    onClear?: (e: any) => void;    onRangeInputClear?: (e: any) => void;    onRangeEndTabPress?: (e: any) => void;    onInsetInputChange?: (options: InsetInputChangeProps) => void}export interface DateInputElementProps {    insetLabel?: any;    prefix?: any}export interface DateInputFoundationProps extends DateInputElementProps, DateInputEventHandlerProps {    [x: string]: any;    value?: BaseValueType[];    disabled?: boolean;    type?: Type;    showClear?: boolean;    format?: string;    inputStyle?: Record<string, any>;    inputReadOnly?: boolean;    validateStatus?: ValidateStatus;    prefixCls?: string;    rangeSeparator?: string;    panelType?: PanelType;    insetInput?: boolean;    insetInputValue?: InsetInputValue;    density?: typeof strings.DENSITY_SET[number];    defaultPickerValue?: ValueType}export interface InsetInputValue {    monthLeft: {        dateInput: string;        timeInput: string    };    monthRight: {        dateInput: string;        timeInput: string    }}export interface InsetInputChangeFoundationProps {    value: string;    insetInputValue: InsetInputValue;    event: any;    valuePath: string}export interface InsetInputChangeProps {     insetInputStr: string;    format: string;    insetInputValue: InsetInputValue}export interface DateInputAdapter extends DefaultAdapter {    updateIsFocusing: (isFocusing: boolean) => void;    notifyClick: DateInputFoundationProps['onClick'];    notifyChange: DateInputFoundationProps['onChange'];    notifyInsetInputChange: DateInputFoundationProps['onInsetInputChange'];    notifyEnter: DateInputFoundationProps['onEnterPress'];    notifyBlur: DateInputFoundationProps['onBlur'];    notifyClear: DateInputFoundationProps['onClear'];    notifyFocus: DateInputFoundationProps['onFocus'];    notifyRangeInputClear: DateInputFoundationProps['onRangeInputClear'];    notifyRangeInputFocus: DateInputFoundationProps['onFocus'];    notifyTabPress: DateInputFoundationProps['onRangeEndTabPress']}export default class InputFoundation extends BaseFoundation<DateInputAdapter> {    constructor(adapter: DateInputAdapter) {        super({ ...adapter });    }    // eslint-disable-next-line @typescript-eslint/no-empty-function    init() {}    // eslint-disable-next-line @typescript-eslint/no-empty-function    destroy() {}    handleClick(e: any) {        this._adapter.notifyClick(e);    }    handleChange(value: string, e: any) {        this._adapter.notifyChange(value, e);    }    handleInputComplete(e: any) {        /**         * onKeyPress, e.key Code gets a value of 0 instead of 13         * Here key is used to judge the button         */        if (e.key === KEY_CODE_ENTER) {            this._adapter.notifyEnter(e.target.value);        }    }    handleInputClear(e: any) {        this._adapter.notifyClear(e);    }    handleRangeInputClear(e: any) {        // prevent trigger click outside        this.stopPropagation(e);        this._adapter.notifyRangeInputClear(e);    }    handleRangeInputEnterPress(e: any, rangeInputValue: string) {        if (e.key === KEY_CODE_ENTER) {            this._adapter.notifyEnter(rangeInputValue);        }    }    handleRangeInputEndKeyPress(e: any) {        if (e.key === KEY_CODE_TAB) {            this._adapter.notifyTabPress(e);        }    }    handleRangeInputFocus(e: any, rangeType: RangeType) {        this._adapter.notifyRangeInputFocus(e, rangeType);    }    formatShowText(value: BaseValueType[], customFormat?: string) {        const { type, dateFnsLocale, format, rangeSeparator } = this._adapter.getProps();        const formatToken = customFormat || format || getDefaultFormatTokenByType(type);        let text = '';        switch (type) {            case 'date':                text = formatDateValues(value, formatToken, undefined, dateFnsLocale);                break;            case 'dateRange':                text = formatDateValues(value, formatToken, { groupSize: 2, groupInnerSeparator: rangeSeparator }, dateFnsLocale);                break;            case 'dateTime':                text = formatDateValues(value, formatToken, undefined, dateFnsLocale);                break;            case 'dateTimeRange':                text = formatDateValues(value, formatToken, { groupSize: 2, groupInnerSeparator: rangeSeparator }, dateFnsLocale);                break;            case 'month':                text = formatDateValues(value, formatToken, undefined, dateFnsLocale);                break;            default:                break;        }        return text;    }    handleInsetInputChange(options: InsetInputChangeFoundationProps) {        const { value, valuePath, insetInputValue } = options;        const { format, type } = this._adapter.getProps();        const insetFormatToken = getInsetInputFormatToken({ type, format });        let newInsetInputValue = set(cloneDeep(insetInputValue), valuePath, value);        newInsetInputValue = this._autoFillTimeToInsetInputValue({ insetInputValue: newInsetInputValue, valuePath, format: insetFormatToken });        const newInputValue = this.concatInsetInputValue({ insetInputValue: newInsetInputValue });        this._adapter.notifyInsetInputChange({ insetInputValue: newInsetInputValue, format: insetFormatToken, insetInputStr: newInputValue });    }    _autoFillTimeToInsetInputValue(options: { insetInputValue: InsetInputValue; format: string; valuePath: string}) {        const { valuePath, insetInputValue, format } = options;        const { type, defaultPickerValue, dateFnsLocale } = this._adapter.getProps();        const insetInputValueWithTime = cloneDeep(insetInputValue);        const { nowDate, nextDate } = getDefaultPickerDate({ defaultPickerValue, format, dateFnsLocale });        if (type.includes('Time')) {            let timeStr = '';            const dateFormatToken = get(format.split(' '), '0', strings.FORMAT_FULL_DATE);            const timeFormatToken = get(format.split(' '), '1', strings.FORMAT_TIME_PICKER);                        switch (valuePath) {                case 'monthLeft.dateInput':                    const dateLeftStr = insetInputValueWithTime.monthLeft.dateInput;                    if (!insetInputValueWithTime.monthLeft.timeInput && dateLeftStr.length === dateFormatToken.length) {                        const dateLeftParsed = compatibleParse(insetInputValueWithTime.monthLeft.dateInput, dateFormatToken);                        if (isValidDate(dateLeftParsed)) {                            timeStr = formatFn(nowDate, timeFormatToken);                            insetInputValueWithTime.monthLeft.timeInput = timeStr;                        }                    }                    break;                case 'monthRight.dateInput':                    const dateRightStr = insetInputValueWithTime.monthRight.dateInput;                    if (!insetInputValueWithTime.monthRight.timeInput && dateRightStr.length === dateFormatToken.length) {                        const dateRightParsed = compatibleParse(dateRightStr, dateFormatToken);                        if (isValidDate(dateRightParsed)) {                            timeStr = formatFn(nextDate, timeFormatToken);                            insetInputValueWithTime.monthRight.timeInput = timeStr;                        }                    }                    break;                default:                    break;            }        }        return insetInputValueWithTime;    }    /**     * 只有传入的 format 符合 formatReg 时,才会使用用户传入的 format     * 否则会使用默认的 format 作为 placeholder     *      * The format passed in by the user will be used only if the incoming format conforms to formatReg     * Otherwise the default format will be used as placeholder     */    getInsetInputPlaceholder() {        const { type, format } = this._adapter.getProps();        const insetInputFormat = getInsetInputFormatToken({ type, format });        let datePlaceholder, timePlaceholder;        switch (type) {            case 'date':            case 'month':            case 'dateRange':                datePlaceholder = insetInputFormat;                break;            case 'dateTime':            case 'dateTimeRange':                [datePlaceholder, timePlaceholder] = insetInputFormat.split(' ');                break;        }        return ({            datePlaceholder,            timePlaceholder,        });    }    /**     * 从当前日期值或 inputValue 中解析出 insetInputValue     *      * Parse out insetInputValue from current date value or inputValue     */    getInsetInputValue({ value, insetInputValue } : { value: BaseValueType[]; insetInputValue: InsetInputValue }) {        const { type, rangeSeparator, format } = this._adapter.getProps();        let inputValueStr = '';        if (isObject(insetInputValue)) {            inputValueStr = this.concatInsetInputValue({ insetInputValue });        } else {            const insetInputFormat = getInsetInputFormatToken({ format, type });            inputValueStr = this.formatShowText(value, insetInputFormat);        }        const newInsetInputValue = getInsetInputValueFromInsetInputStr({ inputValue: inputValueStr, type, rangeSeparator });        return newInsetInputValue;    }    concatInsetDateAndTime({ date, time }) {        return `${date} ${time}`;    }    concatInsetDateRange({ rangeStart, rangeEnd }) {        const { rangeSeparator } = this._adapter.getProps();        return `${rangeStart}${rangeSeparator}${rangeEnd}`;    }    concatInsetInputValue({ insetInputValue }: { insetInputValue: InsetInputValue }) {        const { type } = this._adapter.getProps();        let inputValue = '';        switch (type) {            case 'date':            case 'month':                inputValue = insetInputValue.monthLeft.dateInput;                break;            case 'dateRange':                inputValue = this.concatInsetDateRange({ rangeStart: insetInputValue.monthLeft.dateInput, rangeEnd: insetInputValue.monthRight.dateInput });                break;            case 'dateTime':                inputValue = this.concatInsetDateAndTime({ date: insetInputValue.monthLeft.dateInput, time: insetInputValue.monthLeft.timeInput });                break;            case 'dateTimeRange':                const rangeStart = this.concatInsetDateAndTime({ date: insetInputValue.monthLeft.dateInput, time: insetInputValue.monthLeft.timeInput });                const rangeEnd = this.concatInsetDateAndTime({ date: insetInputValue.monthRight.dateInput, time: insetInputValue.monthRight.timeInput });                inputValue = this.concatInsetDateRange({ rangeStart, rangeEnd });                break;        }        return inputValue;    }}
 |