/* tslint:disable jsx-no-multiline-js */ import * as React from 'react'; import * as moment from 'moment'; import { polyfill } from 'react-lifecycles-compat'; import RangeCalendar from 'rc-calendar/lib/RangeCalendar'; import RcDatePicker from 'rc-calendar/lib/Picker'; import classNames from 'classnames'; import shallowequal from 'shallowequal'; import Icon from '../icon'; import Tag from '../tag'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; import warning from '../_util/warning'; import interopDefault from '../_util/interopDefault'; import { RangePickerValue, RangePickerPresetRange } from './interface'; export interface RangePickerState { value?: RangePickerValue; showDate?: RangePickerValue; open?: boolean; hoverValue?: RangePickerValue; } function getShowDateFromValue(value: RangePickerValue) { const [start, end] = value; // value could be an empty array, then we should not reset showDate if (!start && !end) { return; } const newEnd = end && end.isSame(start, 'month') ? end.clone().add(1, 'month') : end; return [start, newEnd] as RangePickerValue; } function formatValue(value: moment.Moment | undefined, format: string): string { return (value && value.format(format)) || ''; } function pickerValueAdapter( value?: moment.Moment | RangePickerValue, ): RangePickerValue | undefined { if (!value) { return; } if (Array.isArray(value)) { return value; } return [value, value.clone().add(1, 'month')]; } function isEmptyArray(arr: any) { if (Array.isArray(arr)) { return arr.length === 0 || arr.every(i => !i); } return false; } function fixLocale(value: RangePickerValue | undefined, localeCode: string) { if (!localeCode) { return; } if (!value || value.length === 0) { return; } const [start, end] = value; if (start) { start!.locale(localeCode); } if (end) { end!.locale(localeCode); } } class RangePicker extends React.Component { static defaultProps = { allowClear: true, showToday: false, }; static getDerivedStateFromProps(nextProps: any, prevState: any) { let state = null; if ('value' in nextProps) { const value = nextProps.value || []; state = { value, }; if (!shallowequal(nextProps.value, prevState.value)) { state = { ...state, showDate: getShowDateFromValue(value) || prevState.showDate, }; } } if ('open' in nextProps && prevState.open !== nextProps.open) { state = { ...state, open: nextProps.open, }; } return state; } private picker: HTMLSpanElement; private prefixCls?: string; private tagPrefixCls?: string; constructor(props: any) { super(props); const value = props.value || props.defaultValue || []; const [start, end] = value; if ( (start && !interopDefault(moment).isMoment(start)) || (end && !interopDefault(moment).isMoment(end)) ) { throw new Error( 'The value/defaultValue of RangePicker must be a moment object array after `antd@2.0`, ' + 'see: https://u.ant.design/date-picker-value', ); } const pickerValue = !value || isEmptyArray(value) ? props.defaultPickerValue : value; this.state = { value, showDate: pickerValueAdapter(pickerValue || interopDefault(moment)()), open: props.open, hoverValue: [], }; } clearSelection = (e: React.MouseEvent) => { e.preventDefault(); e.stopPropagation(); this.setState({ value: [] }); this.handleChange([]); }; clearHoverValue = () => this.setState({ hoverValue: [] }); handleChange = (value: RangePickerValue) => { const props = this.props; if (!('value' in props)) { this.setState(({ showDate }) => ({ value, showDate: getShowDateFromValue(value) || showDate, })); } const [start, end] = value; props.onChange(value, [formatValue(start, props.format), formatValue(end, props.format)]); }; handleOpenChange = (open: boolean) => { if (!('open' in this.props)) { this.setState({ open }); } if (open === false) { this.clearHoverValue(); } const { onOpenChange } = this.props; if (onOpenChange) { onOpenChange(open); } if (!open) { this.focus(); } }; handleShowDateChange = (showDate: RangePickerValue) => this.setState({ showDate }); handleHoverChange = (hoverValue: any) => this.setState({ hoverValue }); handleRangeMouseLeave = () => { if (this.state.open) { this.clearHoverValue(); } }; handleCalendarInputSelect = (value: RangePickerValue) => { const [start] = value; if (!start) { return; } this.setState(({ showDate }) => ({ value, showDate: getShowDateFromValue(value) || showDate, })); }; handleRangeClick = (value: RangePickerPresetRange) => { if (typeof value === 'function') { value = value(); } this.setValue(value, true); const { onOk, onOpenChange } = this.props; if (onOk) { onOk(value); } if (onOpenChange) { onOpenChange(false); } }; setValue(value: RangePickerValue, hidePanel?: boolean) { this.handleChange(value); if ((hidePanel || !this.props.showTime) && !('open' in this.props)) { this.setState({ open: false }); } } focus() { this.picker.focus(); } blur() { this.picker.blur(); } savePicker = (node: HTMLSpanElement) => { this.picker = node; }; renderFooter = (...args: any[]) => { const { ranges, renderExtraFooter } = this.props; const { prefixCls, tagPrefixCls } = this; if (!ranges && !renderExtraFooter) { return null; } const customFooter = renderExtraFooter ? (
{renderExtraFooter(...args)}
) : null; const operations = Object.keys(ranges || {}).map(range => { const value = ranges[range]; return ( this.handleRangeClick(value)} onMouseEnter={() => this.setState({ hoverValue: value })} onMouseLeave={this.handleRangeMouseLeave} > {range} ); }); const rangeNode = operations && operations.length > 0 ? (
{operations}
) : null; return [rangeNode, customFooter]; }; renderRangePicker = ({ getPrefixCls }: ConfigConsumerProps) => { const { state, props } = this; const { value, showDate, hoverValue, open } = state; const { prefixCls: customizePrefixCls, tagPrefixCls: customizeTagPrefixCls, popupStyle, style, disabledDate, disabledTime, showTime, showToday, ranges, onOk, locale, localeCode, format, dateRender, onCalendarChange, suffixIcon, } = props; const prefixCls = getPrefixCls('calendar', customizePrefixCls); const tagPrefixCls = getPrefixCls('tag', customizeTagPrefixCls); // To support old version react. // Have to add prefixCls on the instance. // https://github.com/facebook/react/issues/12397 this.prefixCls = prefixCls; this.tagPrefixCls = tagPrefixCls; fixLocale(value, localeCode); fixLocale(showDate, localeCode); warning(!('onOK' in props), 'It should be `RangePicker[onOk]`, instead of `onOK`!'); const calendarClassName = classNames({ [`${prefixCls}-time`]: showTime, [`${prefixCls}-range-with-ranges`]: ranges, }); // 需要选择时间时,点击 ok 时才触发 onChange const pickerChangeHandler = { onChange: this.handleChange, }; let calendarProps: any = { onOk: this.handleChange, }; if (props.timePicker) { pickerChangeHandler.onChange = changedValue => this.handleChange(changedValue); } else { calendarProps = {}; } if ('mode' in props) { calendarProps.mode = props.mode; } const startPlaceholder = 'placeholder' in props ? props.placeholder[0] : locale.lang.rangePlaceholder[0]; const endPlaceholder = 'placeholder' in props ? props.placeholder[1] : locale.lang.rangePlaceholder[1]; const calendar = ( ); // default width for showTime const pickerStyle = {} as any; if (props.showTime) { pickerStyle.width = (style && style.width) || 350; } const [startValue, endValue] = value as RangePickerValue; const clearIcon = !props.disabled && props.allowClear && value && (startValue || endValue) ? ( ) : null; const inputIcon = (suffixIcon && (React.isValidElement<{ className?: string }>(suffixIcon) ? ( React.cloneElement(suffixIcon, { className: classNames({ [suffixIcon.props.className!]: suffixIcon.props.className, [`${prefixCls}-picker-icon`]: true, }), }) ) : ( {suffixIcon} ))) || ; const input = ({ value: inputValue }: { value: any }) => { const [start, end] = inputValue; return ( ~ {clearIcon} {inputIcon} ); }; return ( {input} ); }; render() { return {this.renderRangePicker}; } } polyfill(RangePicker); export default RangePicker;