1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411 |
- /* eslint-disable no-nested-ternary */
- /* eslint-disable max-len, max-depth, */
- import { format, isValid, isSameSecond, isEqual as isDateEqual, isDate } from 'date-fns';
- import { get, isObject, isString, isEqual, isFunction } from 'lodash';
- import BaseFoundation, { DefaultAdapter } from '../base/foundation';
- import { isValidDate, isTimestamp } from './_utils/index';
- import isNullOrUndefined from '../utils/isNullOrUndefined';
- import { utcToZonedTime, zonedTimeToUtc } from '../utils/date-fns-extra';
- import { compatibleParse } from './_utils/parser';
- import { getDefaultFormatTokenByType } from './_utils/getDefaultFormatToken';
- import { strings } from './constants';
- import { strings as inputStrings } from '../input/constants';
- import getInsetInputFormatToken from './_utils/getInsetInputFormatToken';
- import getInsetInputValueFromInsetInputStr from './_utils/getInsetInputValueFromInsetInputStr';
- import type { ArrayElement, Motion } from '../utils/type';
- import type { Type, DateInputFoundationProps, InsetInputValue } from './inputFoundation';
- import type { MonthsGridFoundationProps } from './monthsGridFoundation';
- import type { WeekStartNumber } from './_utils/getMonthTable';
- import isValidTimeZone from './_utils/isValidTimeZone';
- export type ValidateStatus = ArrayElement<typeof strings.STATUS>;
- export type InputSize = ArrayElement<typeof strings.SIZE_SET>;
- export type Position = ArrayElement<typeof strings.POSITION_SET>;
- export type PresetPosition = ArrayElement<typeof strings.PRESET_POSITION_SET>;
- export type BaseValueType = string | number | Date;
- export type DayStatusType = {
- isToday?: boolean; // Current day
- isSelected?: boolean; // Selected
- isDisabled?: boolean; // Disabled
- isSelectedStart?: boolean; // Select Start
- isSelectedEnd?: boolean; // End of selection
- isInRange?: boolean; // Range within the selected date
- isHover?: boolean; // Date between selection and hover date
- isOffsetRangeStart?: boolean; // Week selection start
- isOffsetRangeEnd?: boolean; // End of week selection
- isHoverInOffsetRange?: boolean // Hover in the week selection
- };
- export type DisabledDateOptions = {
- rangeStart?: string;
- rangeEnd?: string;
- /**
- * current select of range type
- */
- rangeInputFocus?: 'rangeStart' | 'rangeEnd' | false
- };
- export type PresetType = {
- start?: string | Date | number;
- end?: string | Date | number;
- text?: string
- };
- export type TriggerRenderProps = {
- [x: string]: any;
- value?: ValueType;
- inputValue?: string;
- placeholder?: string | string[];
- autoFocus?: boolean;
- size?: InputSize;
- disabled?: boolean;
- inputReadOnly?: boolean;
- componentProps?: DatePickerFoundationProps
- };
- export type DateOffsetType = (selectedDate?: Date) => Date;
- export type DensityType = 'default' | 'compact';
- export type DisabledDateType = (date?: Date, options?: DisabledDateOptions) => boolean;
- export type DisabledTimeType = (date?: Date | Date[], panelType?: string) => ({
- disabledHours?: () => number[];
- disabledMinutes?: (hour: number) => number[];
- disabledSeconds?: (hour: number, minute: number) => number[]
- });
- export type OnCancelType = (date: Date | Date[], dateStr: string | string[]) => void;
- export type OnPanelChangeType = (date: Date | Date[], dateStr: string | string[]) => void;
- export type OnChangeType = (date?: Date | Date[] | string | string[], dateStr?: string | string[] | Date | Date[]) => void;
- export type OnConfirmType = (date: Date | Date[], dateStr: string | string[]) => void;
- // type OnPresetClickType = (item: PresetType, e: React.MouseEvent<HTMLDivElement>) => void;
- export type OnPresetClickType = (item: PresetType, e: any) => void;
- export type PresetsType = Array<PresetType | (() => PresetType)>;
- // type RenderDateType = (dayNumber?: number, fullDate?: string) => React.ReactNode;
- export type RenderDateType = (dayNumber?: number, fullDate?: string) => any;
- // type RenderFullDateType = (dayNumber?: number, fullDate?: string, dayStatus?: DayStatusType) => React.ReactNode;
- export type RenderFullDateType = (dayNumber?: number, fullDate?: string, dayStatus?: DayStatusType) => any;
- // type TriggerRenderType = (props: TriggerRenderProps) => React.ReactNode;
- export type TriggerRenderType = (props: TriggerRenderProps) => any;
- export type ValueType = BaseValueType | BaseValueType[];
- export interface ElementProps {
- bottomSlot?: any;
- insetLabel?: any;
- prefix?: any;
- topSlot?: any
- }
- export interface RenderProps {
- renderDate?: RenderDateType;
- renderFullDate?: RenderFullDateType;
- triggerRender?: TriggerRenderType
- }
- export type RangeType = 'rangeStart' | 'rangeEnd' | false;
- export interface EventHandlerProps {
- onCancel?: OnCancelType;
- onChange?: OnChangeType;
- onOpenChange?: (status: boolean) => void;
- onPanelChange?: OnPanelChangeType;
- onConfirm?: OnConfirmType;
- // properties below need overwrite
- onBlur?: (e: any) => void;
- onClear?: (e: any) => void;
- onFocus?: (e: any, rangType: RangeType) => void;
- onPresetClick?: OnPresetClickType;
- onClickOutSide?: () => void
- }
- export interface DatePickerFoundationProps extends ElementProps, RenderProps, EventHandlerProps {
- autoAdjustOverflow?: boolean;
- autoFocus?: boolean;
- autoSwitchDate?: boolean;
- borderless?: boolean;
- className?: string;
- defaultOpen?: boolean;
- defaultPickerValue?: ValueType;
- defaultValue?: ValueType;
- density?: DensityType;
- disabled?: boolean;
- disabledDate?: DisabledDateType;
- disabledTime?: DisabledTimeType;
- dropdownClassName?: string;
- dropdownStyle?: Record<string, any>;
- endDateOffset?: DateOffsetType;
- format?: string;
- getPopupContainer?: () => HTMLElement;
- inputReadOnly?: boolean;
- inputStyle?: Record<string, any>;
- max?: number;
- motion?: boolean;
- multiple?: boolean;
- needConfirm?: boolean;
- onChangeWithDateFirst?: boolean;
- open?: boolean;
- placeholder?: string | string[];
- position?: Position;
- prefixCls?: string;
- presets?: PresetsType;
- presetPosition?: PresetPosition;
- showClear?: boolean;
- size?: InputSize;
- spacing?: number;
- startDateOffset?: DateOffsetType;
- stopPropagation?: boolean | string;
- style?: Record<string, any>;
- timePickerOpts?: any; // TODO import timePicker props
- timeZone?: string | number;
- type?: Type;
- validateStatus?: ValidateStatus;
- value?: ValueType;
- weekStartsOn?: WeekStartNumber;
- zIndex?: number;
- syncSwitchMonth?: boolean;
- hideDisabledOptions?: MonthsGridFoundationProps['hideDisabledOptions'];
- disabledTimePicker?: MonthsGridFoundationProps['disabledTimePicker'];
- locale?: any;
- dateFnsLocale?: any;
- localeCode?: string;
- rangeSeparator?: string;
- insetInput?: DateInputFoundationProps['insetInput'];
- preventScroll?: boolean
- }
- export interface DatePickerFoundationState {
- panelShow: boolean;
- isRange: boolean;
- /** value of trigger input */
- inputValue: string;
- value: Date[];
- cachedSelectedValue: Date[];
- prevTimeZone: string | number;
- rangeInputFocus: RangeType;
- autofocus: boolean;
- /** value of inset input */
- insetInputValue: InsetInputValue;
- triggerDisabled: boolean
- }
- export { Type, DateInputFoundationProps };
- export interface DatePickerAdapter extends DefaultAdapter<DatePickerFoundationProps, DatePickerFoundationState> {
- togglePanel: (panelShow: boolean, cb?: () => void) => void;
- registerClickOutSide: () => void;
- unregisterClickOutSide: () => void;
- notifyBlur: DatePickerFoundationProps['onBlur'];
- notifyFocus: DatePickerFoundationProps['onFocus'];
- notifyClear: DatePickerFoundationProps['onClear'];
- notifyChange: DatePickerFoundationProps['onChange'];
- notifyCancel: DatePickerFoundationProps['onCancel'];
- notifyConfirm: DatePickerFoundationProps['onConfirm'];
- notifyOpenChange: DatePickerFoundationProps['onOpenChange'];
- notifyPresetsClick: DatePickerFoundationProps['onPresetClick'];
- updateValue: (value: Date[]) => void;
- updatePrevTimezone: (prevTimeZone: string | number) => void;
- updateCachedSelectedValue: (cachedSelectedValue: Date[]) => void;
- updateInputValue: (inputValue: string) => void;
- needConfirm: () => boolean;
- typeIsYearOrMonth: () => boolean;
- setRangeInputFocus: (rangeInputFocus: DatePickerFoundationState['rangeInputFocus']) => void;
- couldPanelClosed: () => boolean;
- isEventTarget: (e: any) => boolean;
- updateInsetInputValue: (insetInputValue: InsetInputValue) => void;
- setInsetInputFocus: () => void;
- setTriggerDisabled: (disabled: boolean) => void;
- setInputFocus: () => void;
- setInputBlur: () => void;
- setRangeInputBlur: () => void
- }
-
- /**
- * The datePicker foundation.js is responsible for maintaining the date value and the input box value, as well as the callback of both
- * task 1. Accept the selected date change, update the date value, and update the input box value according to the date = > Notify the change
- * task 2. When the input box changes, update the date value = > Notify the change
- */
- export default class DatePickerFoundation extends BaseFoundation<DatePickerAdapter> {
- clickConfirmButton: boolean;
- constructor(adapter: DatePickerAdapter) {
- super({ ...adapter });
- }
- init() {
- const timeZone = this.getProp('timeZone');
- if (this._isControlledComponent()) {
- this.initFromProps({ timeZone, value: this.getProp('value') });
- } else if (this._isInProps('defaultValue')) {
- this.initFromProps({ timeZone, value: this.getProp('defaultValue') });
- }
- this.initPanelOpenStatus(this.getProp('defaultOpen'));
- }
- initFromProps({ value, timeZone, prevTimeZone }: Pick<DatePickerFoundationProps, 'value' | 'timeZone'> & { prevTimeZone?: string | number }) {
- const _value = (Array.isArray(value) ? [...value] : (value || value === 0) && [value]) || [];
- const result = this.parseWithTimezone(_value, timeZone, prevTimeZone);
- this._adapter.updatePrevTimezone(prevTimeZone);
- // reset input value when value update
- this.clearInputValue();
- this._adapter.updateValue(result);
- this.resetCachedSelectedValue(result);
- this.initRangeInputFocus(result);
- if (this._adapter.needConfirm()) {
- this._adapter.updateCachedSelectedValue(result);
- }
- }
- /**
- * 如果用户传了一个空的 value,需要把 range input focus 设置为 rangeStart,这样用户可以清除完之后继续从开始选择
- *
- * If the user passes an empty value, you need to set the range input focus to rangeStart, so that the user can continue to select from the beginning after clearing
- */
- initRangeInputFocus(result: Date[]) {
- const { triggerRender } = this.getProps();
- if (this._isRangeType() && isFunction(triggerRender) && result.length === 0) {
- this._adapter.setRangeInputFocus('rangeStart');
- }
- }
- /**
- * value 可能是 UTC value 也可能是 zoned value
- *
- * UTC value -> 受控传入的 value
- *
- * zoned value -> statue.value,保存的是当前计算机时区下选择的日期
- *
- * 如果是时区变化,则需要将旧 zoned value 转为新时区下的 zoned value
- *
- * 如果是 value 变化,则不需要传入之前的时区,将 UTC value 转为 zoned value 即可
- *
- */
- parseWithTimezone(value: ValueType, timeZone: string | number, prevTimeZone: string | number) {
- const result: Date[] = [];
- if (Array.isArray(value) && value.length) {
- for (const v of value) {
- let parsedV = (v || v === 0) && this._parseValue(v);
- if (parsedV) {
- if (isValidTimeZone(prevTimeZone)) {
- parsedV = zonedTimeToUtc(parsedV, prevTimeZone);
- }
- result.push(isValidTimeZone(timeZone) ? utcToZonedTime(parsedV, timeZone) : parsedV);
- }
- }
- }
- return result;
- }
- _isMultiple() {
- return Boolean(this.getProp('multiple'));
- }
- /**
- *
- * Verify and parse the following three format inputs
- *
- 1. Date object
- 2. ISO 9601-compliant string
- 3. ts timestamp
- Unified here to format the incoming value and output it as a Date object
- *
- */
- _parseValue(value: BaseValueType): Date {
- const dateFnsLocale = this._adapter.getProp('dateFnsLocale');
- let dateObj: Date;
- if (!value && value !== 0) {
- return new Date();
- }
- if (isValidDate(value)) {
- dateObj = value as Date;
- } else if (isString(value)) {
- dateObj = compatibleParse(value as string, this.getProp('format'), undefined, dateFnsLocale);
- } else if (isTimestamp(value)) {
- dateObj = new Date(value);
- } else {
- throw new TypeError('defaultValue should be valid Date object/timestamp or string');
- }
- return dateObj;
- }
- destroy() {
- // Ensure that event listeners will be uninstalled and users may not trigger closePanel
- this._adapter.togglePanel(false);
- this._adapter.unregisterClickOutSide();
- }
- initPanelOpenStatus(defaultOpen?: boolean) {
- if ((this.getProp('open') || defaultOpen) && !this.getProp('disabled')) {
- this._adapter.togglePanel(true);
- this._adapter.registerClickOutSide();
- } else {
- this._adapter.togglePanel(false);
- this._adapter.unregisterClickOutSide();
- }
- }
- openPanel() {
- if (!this.getProp('disabled')) {
- if (!this._isControlledComponent('open')) {
- this.open();
- }
- this._adapter.notifyOpenChange(true);
- }
- }
- /**
- * @deprecated
- * do these side effects when type is dateRange or dateTimeRange
- * 1. trigger input blur, if input value is invalid, set input value and state value to previous status
- * 2. set cachedSelectedValue using given dates(in needConfirm mode)
- * - directly closePanel without click confirm will set cachedSelectedValue to state value
- * - select one date(which means that the selection value is incomplete) and click confirm also set cachedSelectedValue to state value
- */
- // rangeTypeSideEffectsWhenClosePanel(inputValue: string, willUpdateDates: Date[]) {
- // if (this._isRangeType()) {
- // this._adapter.setRangeInputFocus(false);
- // /**
- // * inputValue is string when it is not disabled or can't parsed
- // * when inputValue is null, picker value will back to last selected value
- // */
- // this.handleInputBlur(inputValue);
- // this.resetCachedSelectedValue(willUpdateDates);
- // }
- // }
- /**
- * @deprecated
- * clear input value when selected date is not confirmed
- */
- // needConfirmSideEffectsWhenClosePanel(willUpdateDates: Date[] | null | undefined) {
- // if (this._adapter.needConfirm() && !this._isRangeType()) {
- // /**
- // * if `null` input element will show `cachedSelectedValue` formatted value(format in DateInput render)
- // * if `` input element will show `` directly
- // */
- // this._adapter.updateInputValue(null);
- // this.resetCachedSelectedValue(willUpdateDates);
- // }
- // }
- /**
- * clear inset input value when close panel
- */
- clearInsetInputValue() {
- const { insetInput } = this._adapter.getProps();
- if (insetInput) {
- this._adapter.updateInsetInputValue(null);
- }
- }
- /**
- * call it when change state value or input value
- */
- resetCachedSelectedValue(willUpdateDates?: Date[]) {
- const { value, cachedSelectedValue } = this._adapter.getStates();
- const newCachedSelectedValue = Array.isArray(willUpdateDates) ? willUpdateDates : value;
- if (!isEqual(newCachedSelectedValue, cachedSelectedValue)) {
- this._adapter.updateCachedSelectedValue(newCachedSelectedValue);
- }
- }
- /**
- * timing to call closePanel
- * 1. click confirm button
- * 2. click cancel button
- * 3. select date, time, year, month
- * - date type and not multiple, close panel after select date
- * - dateRange type, close panel after select rangeStart and rangeEnd
- * 4. click outside
- * @param {Event} e
- * @param {String} inputValue
- * @param {Date[]} dates
- */
- closePanel(e?: any, inputValue: string = null, dates?: Date[]) {
- const { value } = this._adapter.getStates();
- const willUpdateDates = isNullOrUndefined(dates) ? value : dates;
- if (!this._isControlledComponent('open')) {
- this.close();
- } else {
- this.resetInnerSelectedStates(willUpdateDates);
- }
- this._adapter.notifyOpenChange(false);
- }
- open() {
- this._adapter.togglePanel(true);
- this._adapter.registerClickOutSide();
- }
- close() {
- this._adapter.togglePanel(false, () => this.resetInnerSelectedStates());
- this._adapter.unregisterClickOutSide();
- }
- focus(focusType?: Exclude<RangeType, false>) {
- if (this._isRangeType()) {
- const rangeInputFocus = focusType ?? 'rangeStart';
- this._adapter.setRangeInputFocus(rangeInputFocus);
- } else {
- this._adapter.setInputFocus();
- }
- }
- blur() {
- if (this._isRangeType()) {
- this._adapter.setRangeInputBlur();
- } else {
- this._adapter.setInputBlur();
- }
- }
- /**
- * reset cachedSelectedValue, inputValue when close panel
- */
- resetInnerSelectedStates(willUpdateDates?: Date[]) {
- const { value } = this._adapter.getStates();
- const needResetCachedSelectedValue = !this.isCachedSelectedValueValid(willUpdateDates) || this._adapter.needConfirm() && !this.clickConfirmButton;
- if (needResetCachedSelectedValue) {
- this.resetCachedSelectedValue(value);
- }
- this.resetFocus();
- this.clearInputValue();
- this.clickConfirmButton = false;
- }
- resetFocus(e?: any) {
- this._adapter.setRangeInputFocus(false);
- this._adapter.notifyBlur(e);
- }
- /**
- * cachedSelectedValue can be `(Date|null)[]` or `null`
- */
- isCachedSelectedValueValid(dates: Date[]) {
- const cachedSelectedValue = dates || this._adapter.getState('cachedSelectedValue');
- const { type } = this._adapter.getProps();
- let isValid = true;
- switch (true) {
- case type === 'dateRange':
- case type === 'dateTimeRange':
- if (!this._isRangeValueComplete(cachedSelectedValue)) {
- isValid = false;
- }
- break;
- default:
- const value = cachedSelectedValue?.filter(item => item);
- if (!(Array.isArray(value) && value.length)) {
- isValid = false;
- }
- break;
- }
- return isValid;
- }
- /**
- * 将输入框内容置空
- */
- clearInputValue() {
- this._adapter.updateInputValue(null);
- this._adapter.updateInsetInputValue(null);
- }
- /**
- * clear range input focus when open is controlled
- * fixed github 1375
- */
- clearRangeInputFocus = () => {
- const { type } = this._adapter.getProps();
- const { rangeInputFocus } = this._adapter.getStates();
- if (type === 'dateTimeRange' && rangeInputFocus) {
- this._adapter.setRangeInputFocus(false);
- }
- }
- /**
- * Callback when the content of the input box changes
- * Update the date panel if the changed value is a legal date, otherwise only update the input box
- * @param {String} input The value of the input box after the change
- * @param {Event} e
- */
- handleInputChange(input: string, e: any) {
- const result = this._isMultiple() ? this.parseMultipleInput(input) : this.parseInput(input);
- const { value: stateValue } = this.getStates();
- this._updateCachedSelectedValueFromInput(input);
- // Enter a valid date or empty
- if ((result && result.length) || input === '') {
- // If you click the clear button
- if (get(e, inputStrings.CLEARBTN_CLICKED_EVENT_FLAG) && this._isControlledComponent('value')) {
- this._notifyChange(result);
- return;
- }
- this._updateValueAndInput(result, input === '', input);
- // Updates the selected value when entering a valid date
- const changedDates = this._getChangedDates(result);
- if (!this._someDateDisabled(changedDates, result)) {
- if (!isEqual(result, stateValue)) {
- this._notifyChange(result);
- }
- }
- } else {
- this._adapter.updateInputValue(input);
- }
- }
- /**
- * inset input 变化时需要更新以下 state 状态
- * - insetInputValue(总是)
- * - inputValue(可以解析为合法日期时)
- * - value(可以解析为合法日期时)
- */
- handleInsetInputChange(options: { insetInputStr: string; format: string; insetInputValue: InsetInputValue }) {
- const { insetInputStr, format, insetInputValue } = options;
- const _isMultiple = this._isMultiple();
- const result = _isMultiple ? this.parseMultipleInput(insetInputStr, format) : this.parseInput(insetInputStr, format);
- const { value: stateValue } = this.getStates();
- this._updateCachedSelectedValueFromInput(insetInputStr);
- if ((result && result.length)) {
- const changedDates = this._getChangedDates(result);
- if (!this._someDateDisabled(changedDates, result)) {
- if (!isEqual(result, stateValue)) {
- if (!this._isControlledComponent() && !this._adapter.needConfirm()) {
- this._adapter.updateValue(result);
- }
- this._notifyChange(result);
- }
- const triggerInput = _isMultiple ? this.formatMultipleDates(result) : this.formatDates(result);
- this._adapter.updateInputValue(triggerInput);
- }
- }
- this._adapter.updateInsetInputValue(insetInputValue);
- }
- /**
- * when input change we reset cached selected value
- */
- _updateCachedSelectedValueFromInput(input: string) {
- const looseResult = this.getLooseDateFromInput(input);
- const changedLooseResult = this._getChangedDates(looseResult);
- if (!this._someDateDisabled(changedLooseResult, looseResult)) {
- this.resetCachedSelectedValue(looseResult);
- }
- }
- /**
- * Input box blur
- * @param {String} input
- * @param {Event} e
- */
- // eslint-disable-next-line @typescript-eslint/no-empty-function
- handleInputBlur(input = '', e?: any) {
- }
- /**
- * called when range type rangeEnd input tab press
- * @param {Event} e
- */
- handleRangeEndTabPress(e: any) {
- this._adapter.setRangeInputFocus(false);
- }
- /**
- * called when the input box is focused
- * @param {Event} e input focus event
- * @param {String} range 'rangeStart' or 'rangeEnd', use when type is range
- */
- handleInputFocus(e: any, range: 'rangeStart' | 'rangeEnd') {
- const rangeInputFocus = this._adapter.getState('rangeInputFocus');
- range && this._adapter.setRangeInputFocus(range);
- /**
- * rangeType: only notify when range is false
- * not rangeType: notify when focus
- */
- if (!range || !['rangeStart', 'rangeEnd'].includes(rangeInputFocus)) {
- this._adapter.notifyFocus(e, range);
- }
- }
- handleSetRangeFocus(rangeInputFocus: RangeType) {
- this._adapter.setRangeInputFocus(rangeInputFocus);
- }
- handleInputClear(e: any) {
- this._adapter.notifyClear(e);
- }
- /**
- * 范围选择清除按钮回调
- * 因为清除按钮没有集成在Input内,因此需要手动清除 value、inputValue、cachedValue
- *
- * callback of range input clear button
- * Since the clear button is not integrated in Input, you need to manually clear value, inputValue, cachedValue
- */
- handleRangeInputClear(e: any) {
- const value: Date[] = [];
- const inputValue = '';
- if (!this._isControlledComponent('value')) {
- this._updateValueAndInput(value, true, inputValue);
- this.resetCachedSelectedValue(value);
- }
- this._notifyChange(value);
- this._adapter.notifyClear(e);
- }
- // eslint-disable-next-line @typescript-eslint/no-empty-function
- handleRangeInputBlur(value: any, e: any) {
- }
- // Parses input only after user returns
- handleInputComplete(input: any = '') {
- // console.log(input);
- let parsedResult = input ?
- this._isMultiple() ?
- this.parseMultipleInput(input, ',', true) :
- this.parseInput(input) :
- [];
- parsedResult = parsedResult && parsedResult.length ? parsedResult : this.getState('value');
- // Use the current date as the value when the current input is empty and the last input is also empty
- if (!parsedResult || !parsedResult.length) {
- const nowDate = new Date();
- if (this._isRangeType()) {
- parsedResult = [nowDate, nowDate];
- } else {
- parsedResult = [nowDate];
- }
- }
- this._updateValueAndInput(parsedResult);
- const { value: stateValue } = this.getStates();
- const changedDates = this._getChangedDates(parsedResult);
- if (!this._someDateDisabled(changedDates, parsedResult) && !isEqual(parsedResult, stateValue)) {
- this._notifyChange(parsedResult);
- }
- }
- /**
- * Parse the input, return the time object if it is valid,
- * otherwise return "
- *
- * @param {string} input
- * @returns {Date [] | '}
- */
- parseInput(input = '', format?: string) {
- let result: Date[] = [];
- // console.log(input);
- const { dateFnsLocale, rangeSeparator } = this.getProps();
- if (input && input.length) {
- const type = this.getProp('type');
- const formatToken = format || this.getProp('format') || getDefaultFormatTokenByType(type);
- let parsedResult,
- formatedInput;
- const nowDate = new Date();
- switch (type) {
- case 'date':
- case 'dateTime':
- case 'month':
- parsedResult = input ? compatibleParse(input, formatToken, nowDate, dateFnsLocale) : '';
- formatedInput = parsedResult && isValid(parsedResult) && this.localeFormat(parsedResult as Date, formatToken);
- if (parsedResult && formatedInput === input) {
- result = [parsedResult as Date];
- }
- break;
- case 'dateRange':
- case 'dateTimeRange':
- case 'monthRange':
- const separator = rangeSeparator;
- const values = input.split(separator);
- parsedResult =
- values &&
- values.reduce((arr, cur) => {
- const parsedVal = cur && compatibleParse(cur, formatToken, nowDate, dateFnsLocale);
- parsedVal && arr.push(parsedVal);
- return arr;
- }, []);
- formatedInput =
- parsedResult &&
- parsedResult.map(v => v && isValid(v) && this.localeFormat(v, formatToken)).join(separator);
- if (parsedResult && formatedInput === input) {
- parsedResult.sort((d1, d2) => d1.getTime() - d2.getTime());
- result = parsedResult;
- }
- break;
- default:
- break;
- }
- }
- return result;
- }
- /**
- * get date which may include null from input
- */
- getLooseDateFromInput(input: string): Array<Date | null> {
- const value = this._isMultiple() ? this.parseMultipleInputLoose(input) : this.parseInputLoose(input);
- return value;
- }
- /**
- * parse input into `Array<Date|null>`, loose means return value includes `null`
- *
- * @example
- * ```javascript
- * parseInputLoose('2022-03-15 ~ '); // [Date, null]
- * parseInputLoose(' ~ 2022-03-15 '); // [null, Date]
- * parseInputLoose(''); // []
- * parseInputLoose('2022-03- ~ 2022-0'); // [null, null]
- * ```
- */
- parseInputLoose(input = ''): Array<Date | null> {
- let result: Array<Date | null> = [];
- const { dateFnsLocale, rangeSeparator, type, format } = this.getProps();
- if (input && input.length) {
- const formatToken = format || getDefaultFormatTokenByType(type);
- let parsedResult, formatedInput;
- const nowDate = new Date();
- switch (type) {
- case 'date':
- case 'dateTime':
- case 'month':
- const _parsedResult = compatibleParse(input, formatToken, nowDate, dateFnsLocale);
- if (isValidDate(_parsedResult)) {
- formatedInput = this.localeFormat(_parsedResult as Date, formatToken);
- if (formatedInput === input) {
- parsedResult = _parsedResult;
- }
- } else {
- parsedResult = null;
- }
- result = [parsedResult];
- break;
- case 'dateRange':
- case 'dateTimeRange':
- const separator = rangeSeparator;
- const values = input.split(separator);
- parsedResult =
- values &&
- values.reduce((arr, cur) => {
- let parsedVal = null;
- const _parsedResult = compatibleParse(cur, formatToken, nowDate, dateFnsLocale);
- if (isValidDate(_parsedResult)) {
- formatedInput = this.localeFormat(_parsedResult as Date, formatToken);
- if (formatedInput === cur) {
- parsedVal = _parsedResult;
- }
- }
- arr.push(parsedVal);
- return arr;
- }, []);
- if (Array.isArray(parsedResult) && parsedResult.every(item => isValid(item))) {
- parsedResult.sort((d1, d2) => d1.getTime() - d2.getTime());
- }
- result = parsedResult;
- break;
- default:
- break;
- }
- }
- return result;
- }
- /**
- * parse multiple into `Array<Date|null>`, loose means return value includes `null`
- *
- * @example
- * ```javascript
- * parseMultipleInputLoose('2021-01-01,2021-10-15'); // [Date, Date];
- * parseMultipleInputLoose('2021-01-01,2021-10-'); // [Date, null];
- * parseMultipleInputLoose(''); // [];
- * ```
- */
- parseMultipleInputLoose(input = '', separator: string = strings.DEFAULT_SEPARATOR_MULTIPLE, needDedupe = false) {
- const max = this.getProp('max');
- const inputArr = input.split(separator);
- const result: Date[] = [];
- for (const curInput of inputArr) {
- let tmpParsed = curInput && this.parseInputLoose(curInput);
- tmpParsed = Array.isArray(tmpParsed) ? tmpParsed : tmpParsed && [tmpParsed];
- if (tmpParsed && tmpParsed.length) {
- if (needDedupe) {
- !result.filter(r => Boolean(tmpParsed.find(tp => isSameSecond(r, tp)))) && result.push(...tmpParsed);
- } else {
- result.push(...tmpParsed);
- }
- } else {
- return [];
- }
- if (max && max > 0 && result.length > max) {
- return [];
- }
- }
- return result;
- }
- /**
- * Parses the input when multiple is true, if valid,
- * returns a list of time objects, otherwise returns an array
- *
- * @param {string} [input='']
- * @param {string} [separator=',']
- * @param {boolean} [needDedupe=false]
- * @returns {Date[]}
- */
- parseMultipleInput(input = '', separator: string = strings.DEFAULT_SEPARATOR_MULTIPLE, needDedupe = false) {
- const max = this.getProp('max');
- const inputArr = input.split(separator);
- const result: Date[] = [];
- for (const curInput of inputArr) {
- let tmpParsed = curInput && this.parseInput(curInput);
- tmpParsed = Array.isArray(tmpParsed) ? tmpParsed : tmpParsed && [tmpParsed];
- if (tmpParsed && tmpParsed.length) {
- if (needDedupe) {
- // 20190519 TODO: needs to determine the case where multiple is true and range
- !result.filter(r => Boolean(tmpParsed.find(tp => isSameSecond(r, tp)))) && result.push(...tmpParsed);
- } else {
- result.push(...tmpParsed);
- }
- } else {
- return [];
- }
- if (max && max > 0 && result.length > max) {
- return [];
- }
- }
- return result;
- }
- /**
- * dates[] => string
- *
- * @param {Date[]} dates
- * @returns {string}
- */
- formatDates(dates: Date[] = [], customFormat?: string) {
- let str = '';
- const rangeSeparator = this.getProp('rangeSeparator');
- if (Array.isArray(dates) && dates.length) {
- const type = this.getProp('type');
- const formatToken = customFormat || this.getProp('format') || getDefaultFormatTokenByType(type);
- switch (type) {
- case 'date':
- case 'dateTime':
- case 'month':
- str = this.localeFormat(dates[0], formatToken);
- break;
- case 'dateRange':
- case 'dateTimeRange':
- case 'monthRange':
- const startIsTruthy = !isNullOrUndefined(dates[0]);
- const endIsTruthy = !isNullOrUndefined(dates[1]);
- if (startIsTruthy && endIsTruthy) {
- str = `${this.localeFormat(dates[0], formatToken)}${rangeSeparator}${this.localeFormat(dates[1], formatToken)}`;
- } else {
- if (startIsTruthy) {
- str = `${this.localeFormat(dates[0], formatToken)}${rangeSeparator}`;
- } else if (endIsTruthy) {
- str = `${rangeSeparator}${this.localeFormat(dates[1], formatToken)}`;
- }
- }
- break;
- default:
- break;
- }
- }
- return str;
- }
- /**
- * dates[] => string
- *
- * @param {Date[]} dates
- * @returns {string}
- */
- formatMultipleDates(dates: Date[] = [], separator: string = strings.DEFAULT_SEPARATOR_MULTIPLE, customFormat?: string) {
- const strs = [];
- if (Array.isArray(dates) && dates.length) {
- const type = this.getProp('type');
- switch (type) {
- case 'date':
- case 'dateTime':
- case 'month':
- dates.forEach(date => strs.push(this.formatDates([date], customFormat)));
- break;
- case 'dateRange':
- case 'dateTimeRange':
- case 'monthRange':
- for (let i = 0; i < dates.length; i += 2) {
- strs.push(this.formatDates(dates.slice(i, i + 2), customFormat));
- }
- break;
- default:
- break;
- }
- }
- return strs.join(separator);
- }
- /**
- * Update date value and the value of the input box
- * 1. Select Update
- * 2. Input Update
- * @param {Date|''} value
- * @param {Boolean} forceUpdateValue
- * @param {String} input
- */
- _updateValueAndInput(value: Date | Array<Date>, forceUpdateValue?: boolean, input?: string) {
- let _value: Array<Date>;
- if (forceUpdateValue || value) {
- if (!Array.isArray(value)) {
- _value = value ? [value] : [];
- } else {
- _value = value;
- }
- const changedDates = this._getChangedDates(_value);
- // You cannot update the value directly when needConfirm, you can only change the value through handleConfirm
- if (!this._isControlledComponent() && !this._someDateDisabled(changedDates, _value) && !this._adapter.needConfirm()) {
- this._adapter.updateValue(_value);
- }
- }
- this._adapter.updateInputValue(input);
- }
- /**
- * when changing the selected value through the date panel
- * @param {*} value
- * @param {*} options
- */
- handleSelectedChange(value: Date[], options?: { fromPreset?: boolean; needCheckFocusRecord?: boolean }) {
- const { type, format, rangeSeparator, insetInput } = this._adapter.getProps();
- const { value: stateValue } = this.getStates();
- const controlled = this._isControlledComponent();
- const fromPreset = isObject(options) ? options.fromPreset : options;
- const closePanel = get(options, 'closePanel', true);
- /**
- * It is used to determine whether the panel can be stowed. In a Range type component, it is necessary to select both starting Time and endTime before stowing.
- * To determine whether both starting Time and endTime have been selected, it is used to judge whether the two inputs have been Focused.
- * This variable is used to indicate whether such a judgment is required. In the scene with shortcut operations, it is not required.
- */
- const needCheckFocusRecord = get(options, 'needCheckFocusRecord', true);
- const dates = Array.isArray(value) ? [...value] : value ? [value] : [];
- const changedDates = this._getChangedDates(dates);
- let inputValue, insetInputValue;
- if (!this._someDateDisabled(changedDates, dates)) {
- this.resetCachedSelectedValue(dates);
- inputValue = this._isMultiple() ? this.formatMultipleDates(dates) : this.formatDates(dates);
- if (insetInput) {
- const insetInputFormatToken = getInsetInputFormatToken({ format, type });
- const insetInputStr = this._isMultiple() ? this.formatMultipleDates(dates, undefined, insetInputFormatToken) : this.formatDates(dates, insetInputFormatToken);
- insetInputValue = getInsetInputValueFromInsetInputStr({ inputValue: insetInputStr, type, rangeSeparator });
- }
- const isRangeTypeAndInputIncomplete = this._isRangeType() && !this._isRangeValueComplete(dates);
- /**
- * If the input is incomplete when under control, the notifyChange is not triggered because
- * You need to update the value of the input box, otherwise there will be a problem that a date is selected but the input box does not show the date #1357
- *
- * 受控时如果输入不完整,由于没有触发 notifyChange
- * 需要组件内更新一下输入框的值,否则会出现选了一个日期但是输入框没有回显日期的问题 #1357
- */
- if (!this._adapter.needConfirm() || fromPreset) {
- if (isRangeTypeAndInputIncomplete) {
- // do not change value when selected value is incomplete
- this._adapter.updateInputValue(inputValue);
- this._adapter.updateInsetInputValue(insetInputValue);
- return;
- } else {
- if (!controlled || fromPreset) {
- this._updateValueAndInput(dates, true, inputValue);
- this._adapter.updateInsetInputValue(insetInputValue);
- }
- }
- }
- if (!controlled && this._adapter.needConfirm()) {
- // select date only change inputValue when needConfirm is true
- this._adapter.updateInputValue(inputValue);
- this._adapter.updateInsetInputValue(insetInputValue);
- // if inputValue is not complete, don't notifyChange
- if (isRangeTypeAndInputIncomplete) {
- return;
- }
- }
- if (!isEqual(value, stateValue)) {
- this._notifyChange(value);
- }
- }
- const focusRecordChecked = !needCheckFocusRecord || (needCheckFocusRecord && this._adapter.couldPanelClosed());
- if ((type === 'date' && !this._isMultiple() && closePanel) || (type === 'dateRange' && this._isRangeValueComplete(dates) && closePanel && focusRecordChecked)) {
- this.closePanel(undefined, inputValue, dates);
- }
- }
- /**
- * when changing the year and month through the panel when the type is year or month or monthRange
- * @param {*} item
- */
- handleYMSelectedChange(item: { currentMonth?: { left: number; right: number }; currentYear?: { left: number; right: number } } = {}) {
- // console.log(item);
- const { currentMonth, currentYear } = item;
- const { type } = this.getProps();
- if (type === 'month') {
- const date = new Date(currentYear['left'], currentMonth['left'] - 1);
- this.handleSelectedChange([date]);
- } else {
- const dateLeft = new Date(currentYear['left'], currentMonth['left'] - 1);
- const dateRight = new Date(currentYear['right'], currentMonth['right'] - 1);
- this.handleSelectedChange([dateLeft, dateRight]);
- }
- }
- handleConfirm() {
- this.clickConfirmButton = true;
- const { cachedSelectedValue, value } = this._adapter.getStates();
- const isRangeValueComplete = this._isRangeValueComplete(cachedSelectedValue);
- const newValue = isRangeValueComplete ? cachedSelectedValue : value;
- if (this._adapter.needConfirm() && !this._isControlledComponent()) {
- this._adapter.updateValue(newValue);
- }
- // If the input is incomplete, the legal date of the last input is used
- this.closePanel(undefined, undefined, newValue);
- if (isRangeValueComplete) {
- const { notifyValue, notifyDate } = this.disposeCallbackArgs(cachedSelectedValue);
- this._adapter.notifyConfirm(notifyDate, notifyValue);
- }
- }
- handleCancel() {
- this.closePanel();
- const value = this.getState('value');
- const { notifyValue, notifyDate } = this.disposeCallbackArgs(value);
- this._adapter.notifyCancel(notifyDate, notifyValue);
- }
- handlePresetClick(item: PresetType, e: any) {
- const { type, timeZone } = this.getProps();
- const prevTimeZone = this.getState('prevTimezone');
- let value;
- switch (type) {
- case 'month':
- case 'dateTime':
- case 'date':
- value = this.parseWithTimezone([item.start], timeZone, prevTimeZone);
- this.handleSelectedChange(value);
- break;
- case 'dateTimeRange':
- case 'dateRange':
- value = this.parseWithTimezone([item.start, item.end], timeZone, prevTimeZone);
- this.handleSelectedChange(value, { needCheckFocusRecord: false });
- break;
- default:
- break;
- }
- this._adapter.notifyPresetsClick(item, e);
- }
- /**
- * 根据 type 处理 onChange 返回的参数
- *
- * - 返回的日期需要把用户时间转换为设置的时区时间
- * - 用户时间:用户计算机系统时间
- * - 时区时间:通过 ConfigProvider 设置的 timeZone
- * - 例子:用户设置时区为+9,计算机所在时区为+8区,然后用户选择了22:00
- * - DatePicker 内部保存日期 state 为 +8 的 22:00 => a = new Date("2021-05-25 22:00:00")
- * - 传出去时,需要把 +8 的 22:00 => +9 的 22:00 => b = zonedTimeToUtc(a, "+09:00");
- *
- * According to the type processing onChange returned parameters
- *
- * - the returned date needs to convert the user time to the set time zone time
- * - user time: user computer system time
- * - time zone time: timeZone set by ConfigProvider
- * - example: the user sets the time zone to + 9, the computer's time zone is + 8 zone, and then the user selects 22:00
- * - DatePicker internal save date state is + 8 22:00 = > a = new Date ("2021-05-25 22:00:00")
- * - when passed out, you need to + 8 22:00 = > + 9 22:00 = > b = zonedTimeToUtc (a, "+ 09:00");
- *
- * e.g.
- * let a = new Date ("2021-05-25 22:00:00");
- * = > Tue May 25 2021 22:00:00 GMT + 0800 (China Standard Time)
- * let b = zonedTimeToUtc (a, "+ 09:00");
- * = > Tue May 25 2021 21:00:00 GMT + 0800 (China Standard Time)
- *
- * @param {Date|Date[]} value
- * @return {{ notifyDate: Date|Date[], notifyValue: string|string[]}}
- */
- disposeCallbackArgs(value: Date | Date[]) {
- let _value = Array.isArray(value) ? value : (value && [value]) || [];
- const timeZone = this.getProp('timeZone');
- if (isValidTimeZone(timeZone)) {
- _value = _value.map(date => zonedTimeToUtc(date, timeZone));
- }
- const type = this.getProp('type');
- const formatToken = this.getProp('format') || getDefaultFormatTokenByType(type);
- let notifyValue,
- notifyDate;
- switch (type) {
- case 'date':
- case 'dateTime':
- case 'month':
- if (!this._isMultiple()) {
- notifyValue = _value[0] && this.localeFormat(_value[0], formatToken);
- [notifyDate] = _value;
- } else {
- notifyValue = _value.map(v => v && this.localeFormat(v, formatToken));
- notifyDate = [..._value];
- }
- break;
- case 'dateRange':
- case 'dateTimeRange':
- case 'monthRange':
- notifyValue = _value.map(v => v && this.localeFormat(v, formatToken));
- notifyDate = [..._value];
- break;
- default:
- break;
- }
- return {
- notifyValue,
- notifyDate,
- };
- }
- /**
- * Notice: Check whether the date is the same as the state value before calling
- * @param {Date[]} value
- */
- _notifyChange(value: Date[]) {
- if (this._isRangeType() && !this._isRangeValueComplete(value)) {
- return;
- }
- const { onChangeWithDateFirst } = this.getProps();
- const { notifyValue, notifyDate } = this.disposeCallbackArgs(value);
- if (onChangeWithDateFirst) {
- this._adapter.notifyChange(notifyDate, notifyValue);
- } else {
- this._adapter.notifyChange(notifyValue, notifyDate);
- }
- }
- /**
- * Get the date changed through the date panel or enter
- * @param {Date[]} dates
- * @returns {Date[]}
- */
- _getChangedDates(dates: Date[]) {
- const type = this._adapter.getProp('type');
- const stateValue: Date[] = this._adapter.getState('value');
- const changedDates = [];
- switch (type) {
- case 'dateRange':
- case 'dateTimeRange':
- const [stateStart, stateEnd] = stateValue;
- const [start, end] = dates;
- if (!isDateEqual(start, stateStart)) {
- changedDates.push(start);
- }
- if (!isDateEqual(end, stateEnd)) {
- changedDates.push(end);
- }
- break;
- default:
- const stateValueSet = new Set<number>();
- stateValue.forEach(value => stateValueSet.add(isDate(value) && value.valueOf()));
- for (const date of dates) {
- if (!stateValueSet.has(isDate(date) && date.valueOf())) {
- changedDates.push(date);
- }
- }
- }
- return changedDates;
- }
- /**
- * Whether a date is disabled
- * @param value The date that needs to be judged whether to disable
- * @param selectedValue Selected date, when selecting a range, pass this date to the second parameter of `disabledDate`
- */
- _someDateDisabled(value: Date[], selectedValue: Date[]) {
- const { rangeInputFocus } = this.getStates();
- const disabledOptions = { rangeStart: '', rangeEnd: '', rangeInputFocus };
- // DisabledDate needs to pass the second parameter
- if (this._isRangeType() && Array.isArray(selectedValue)) {
- if (isValid(selectedValue[0])) {
- const rangeStart = format(selectedValue[0], 'yyyy-MM-dd');
- disabledOptions.rangeStart = rangeStart;
- }
- if (isValid(selectedValue[1])) {
- const rangeEnd = format(selectedValue[1], 'yyyy-MM-dd');
- disabledOptions.rangeEnd = rangeEnd;
- }
- }
- let isSomeDateDisabled = false;
- for (const date of value) {
- // skip check if date is null
- if (!isNullOrUndefined(date) && this.disabledDisposeDate(date, disabledOptions)) {
- isSomeDateDisabled = true;
- break;
- }
- }
- return isSomeDateDisabled;
- }
- /**
- * Format locale date
- * locale get from LocaleProvider
- * @param {Date} date
- * @param {String} token
- */
- localeFormat(date: Date, token: string) {
- const dateFnsLocale = this._adapter.getProp('dateFnsLocale');
- return format(date, token, { locale: dateFnsLocale });
- }
- _isRangeType = () => {
- const type = this._adapter.getProp('type');
- return /range/i.test(type);
- };
- _isRangeValueComplete = (value: Date[] | Date) => {
- let result = false;
- if (Array.isArray(value)) {
- result = !value.some(date => isNullOrUndefined(date));
- }
- return result;
- };
- /**
- * Convert computer date to UTC date
- * Before passing the date to the user, you need to convert the date to UTC time
- * dispose date from computer date to utc date
- * When given timeZone prop, you should convert computer date to utc date before passing to user
- * @param {(date: Date) => Boolean} fn
- * @param {Date|Date[]} date
- * @returns {Boolean}
- */
- disposeDateFn(fn: (date: Date, ...rest: any) => boolean, date: Date | Date[], ...rest: any[]) {
- const { notifyDate } = this.disposeCallbackArgs(date);
- const dateIsArray = Array.isArray(date);
- const notifyDateIsArray = Array.isArray(notifyDate);
- let disposeDate;
- if (dateIsArray === notifyDateIsArray) {
- disposeDate = notifyDate;
- } else {
- disposeDate = dateIsArray ? [notifyDate] : notifyDate[0];
- }
- return fn(disposeDate, ...rest);
- }
- /**
- * Determine whether the date is disabled
- * Whether the date is disabled
- * @param {Date} date
- * @returns {Boolean}
- */
- disabledDisposeDate(date: Date, ...rest: any[]) {
- const { disabledDate } = this.getProps();
- return this.disposeDateFn(disabledDate, date, ...rest);
- }
- /**
- * Determine whether the date is disabled
- * Whether the date time is disabled
- * @param {Date|Date[]} date
- * @returns {Object}
- */
- disabledDisposeTime(date: Date | Date[], ...rest: any[]) {
- const { disabledTime } = this.getProps();
- return this.disposeDateFn(disabledTime, date, ...rest);
- }
- /**
- * Trigger wrapper needs to do two things:
- * 1. Open Panel when clicking trigger;
- * 2. When clicking on a child but the child does not listen to the focus event, manually trigger focus
- *
- * @param {Event} e
- * @returns
- */
- handleTriggerWrapperClick(e: any) {
- const { disabled, triggerRender } = this._adapter.getProps();
- const { rangeInputFocus } = this._adapter.getStates();
- if (disabled) {
- return;
- }
- /**
- * - 非范围选择时,trigger 为原生输入框,已在组件内处理了 focus 逻辑
- * - isEventTarget 函数用于判断触发事件的是否为 input wrapper。如果是冒泡上来的不用处理,因为在子级已经处理了 focus 逻辑。
- *
- * - When type is not range type, Input component will automatically focus in the same case
- * - isEventTarget is used to judge whether the event is a bubbling event
- */
- if (this._isRangeType() && !rangeInputFocus) {
- if (this._adapter.isEventTarget(e)) {
- setTimeout(() => {
- // using setTimeout get correct state value 'rangeInputFocus'
- this.handleInputFocus(e, 'rangeStart');
- }, 0);
- } else if (isFunction(triggerRender)) {
- // 如果是 triggerRender 场景,因为没有 input,因此打开面板时默认 focus 在 rangeStart
- // If it is a triggerRender scene, because there is no input, the default focus is rangeStart when the panel is opened
- this._adapter.setRangeInputFocus('rangeStart');
- }
- this.openPanel();
- } else {
- this.openPanel();
- }
- }
- handlePanelVisibleChange(visible: boolean) {
- if (visible) {
- this._adapter.setInsetInputFocus();
- /**
- * After the panel is closed, the trigger input is disabled
- * 面板关闭后,trigger input 禁用
- */
- setTimeout(() => {
- this._adapter.setTriggerDisabled(true);
- }, 0);
- } else {
- this._adapter.setTriggerDisabled(false);
- }
- }
- }
|