import * as React from 'react'; import * as PropTypes from 'prop-types'; import * as moment from 'moment'; import FullCalendar from 'rc-calendar/lib/FullCalendar'; import Header from './Header'; import enUS from './locale/en_US'; import LocaleReceiver from '../locale-provider/LocaleReceiver'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; import interopDefault from '../_util/interopDefault'; export { HeaderProps } from './Header'; function noop() { return null; } function zerofixed(v: number) { if (v < 10) { return `0${v}`; } return `${v}`; } export type CalendarMode = 'month' | 'year'; export interface CalendarProps { prefixCls?: string; className?: string; value?: moment.Moment; defaultValue?: moment.Moment; mode?: CalendarMode; fullscreen?: boolean; dateCellRender?: (date: moment.Moment) => React.ReactNode; monthCellRender?: (date: moment.Moment) => React.ReactNode; dateFullCellRender?: (date: moment.Moment) => React.ReactNode; monthFullCellRender?: (date: moment.Moment) => React.ReactNode; locale?: any; style?: React.CSSProperties; onPanelChange?: (date?: moment.Moment, mode?: CalendarMode) => void; onSelect?: (date?: moment.Moment) => void; onChange?: (date?: moment.Moment) => void; disabledDate?: (current: moment.Moment) => boolean; validRange?: [moment.Moment, moment.Moment]; } export interface CalendarState { value: moment.Moment; mode?: CalendarMode; } export default class Calendar extends React.Component { static defaultProps = { locale: {}, fullscreen: true, mode: 'month' as CalendarMode, onSelect: noop, onPanelChange: noop, onChange: noop, }; static propTypes = { monthCellRender: PropTypes.func, dateCellRender: PropTypes.func, monthFullCellRender: PropTypes.func, dateFullCellRender: PropTypes.func, fullscreen: PropTypes.bool, locale: PropTypes.object, prefixCls: PropTypes.string, className: PropTypes.string, style: PropTypes.object, onPanelChange: PropTypes.func, value: PropTypes.object as PropTypes.Requireable, onSelect: PropTypes.func, onChange: PropTypes.func, }; prefixCls?: string; constructor(props: CalendarProps) { super(props); const value = props.value || props.defaultValue || interopDefault(moment)(); if (!interopDefault(moment).isMoment(value)) { throw new Error( 'The value/defaultValue of Calendar must be a moment object after `antd@2.0`, ' + 'see: https://u.ant.design/calendar-value', ); } this.state = { value, mode: props.mode, }; } componentWillReceiveProps(nextProps: CalendarProps) { if ('value' in nextProps) { this.setState({ value: nextProps.value!, }); } if ('mode' in nextProps && nextProps.mode !== this.props.mode) { this.setState({ mode: nextProps.mode!, }); } } monthCellRender = (value: moment.Moment) => { const { monthCellRender = noop as Function } = this.props; const { prefixCls } = this; return (
{value.localeData().monthsShort(value)}
{monthCellRender(value)}
); }; dateCellRender = (value: moment.Moment) => { const { dateCellRender = noop as Function } = this.props; const { prefixCls } = this; return (
{zerofixed(value.date())}
{dateCellRender(value)}
); }; setValue = (value: moment.Moment, way: 'select' | 'changePanel') => { const prevValue = this.props.value || this.state.value; const { mode } = this.state; if (!('value' in this.props)) { this.setState({ value }); } if (way === 'select') { if (prevValue && prevValue.month() !== value.month()) { this.onPanelChange(value, mode); } if (this.props.onSelect) { this.props.onSelect(value); } } else if (way === 'changePanel') { this.onPanelChange(value, mode); } }; setType = (type: string) => { const mode = type === 'date' ? 'month' : 'year'; if (this.state.mode !== mode) { this.setState({ mode }); this.onPanelChange(this.state.value, mode); } }; onHeaderValueChange = (value: moment.Moment) => { this.setValue(value, 'changePanel'); }; onHeaderTypeChange = (type: string) => { this.setType(type); }; onPanelChange(value: moment.Moment, mode: CalendarMode | undefined) { const { onPanelChange, onChange } = this.props; if (onPanelChange) { onPanelChange(value, mode); } if (onChange && value !== this.state.value) { onChange(value); } } onSelect = (value: moment.Moment) => { this.setValue(value, 'select'); }; getDateRange = ( validRange: [moment.Moment, moment.Moment], disabledDate?: (current: moment.Moment) => boolean, ) => (current: moment.Moment) => { if (!current) { return false; } const [startDate, endDate] = validRange; const inRange = !current.isBetween(startDate, endDate, 'days', '[]'); if (disabledDate) { return disabledDate(current) || inRange; } return inRange; }; getDefaultLocale = () => { const result = { ...enUS, ...this.props.locale, }; result.lang = { ...result.lang, ...(this.props.locale || {}).lang, }; return result; }; renderCalendar = (locale: any, localeCode: string) => { const { state, props } = this; const { value, mode } = state; if (value && localeCode) { value.locale(localeCode); } const { prefixCls: customizePrefixCls, style, className, fullscreen, dateFullCellRender, monthFullCellRender, } = props; const type = mode === 'year' ? 'month' : 'date'; const monthCellRender = monthFullCellRender || this.monthCellRender; const dateCellRender = dateFullCellRender || this.dateCellRender; let disabledDate = props.disabledDate; if (props.validRange) { disabledDate = this.getDateRange(props.validRange, disabledDate); } return ( {({ getPrefixCls }: ConfigConsumerProps) => { const prefixCls = getPrefixCls('fullcalendar', customizePrefixCls); // To support old version react. // Have to add prefixCls on the instance. // https://github.com/facebook/react/issues/12397 this.prefixCls = prefixCls; let cls = className || ''; if (fullscreen) { cls += ` ${prefixCls}-fullscreen`; } return (
); }}
); }; render() { return ( {this.renderCalendar} ); } }