import * as React from 'react'; import * as moment from 'moment'; import Select from '../select'; import { Group, Button, RadioChangeEvent } from '../radio'; import { ConfigConsumer, ConfigConsumerProps } from '../config-provider'; const { Option } = Select; export interface RenderHeader { value: moment.Moment; onChange?: (value: moment.Moment) => void; type: string; onTypeChange: (type: string) => void; } export interface HeaderProps { prefixCls?: string; locale?: any; fullscreen?: boolean; yearSelectOffset?: number; yearSelectTotal?: number; type?: string; onValueChange?: (value: moment.Moment) => void; onTypeChange?: (type: string) => void; value: moment.Moment; validRange?: [moment.Moment, moment.Moment]; headerRender: (header: RenderHeader) => React.ReactNode; } export default class Header extends React.Component { static defaultProps = { yearSelectOffset: 10, yearSelectTotal: 20, }; private calenderHeaderNode: HTMLDivElement; getYearSelectElement(prefixCls: string, year: number) { const { yearSelectOffset, yearSelectTotal, locale, fullscreen, validRange } = this.props; let start = year - (yearSelectOffset as number); let end = start + (yearSelectTotal as number); if (validRange) { start = validRange[0].get('year'); end = validRange[1].get('year') + 1; } const suffix = locale.year === '年' ? '年' : ''; const options: React.ReactElement[] = []; for (let index = start; index < end; index++) { options.push(); } return ( ); } getMonthsLocale(value: moment.Moment) { const current = value.clone(); const localeData = value.localeData(); const months: any[] = []; for (let i = 0; i < 12; i++) { current.month(i); months.push(localeData.monthsShort(current)); } return months; } getMonthSelectElement(prefixCls: string, month: number, months: number[]) { const { fullscreen, validRange, value } = this.props; const options: React.ReactElement[] = []; let start = 0; let end = 12; if (validRange) { const [rangeStart, rangeEnd] = validRange; const currentYear = value.get('year'); if (rangeEnd.get('year') === currentYear) { end = rangeEnd.get('month') + 1; } if (rangeStart.get('year') === currentYear) { start = rangeStart.get('month'); } } for (let index = start; index < end; index++) { options.push(); } return ( ); } onYearChange = (year: string) => { const { value, validRange } = this.props; const newValue = value.clone(); newValue.year(parseInt(year, 10)); // switch the month so that it remains within range when year changes if (validRange) { const [start, end] = validRange; const newYear = newValue.get('year'); const newMonth = newValue.get('month'); if (newYear === end.get('year') && newMonth > end.get('month')) { newValue.month(end.get('month')); } if (newYear === start.get('year') && newMonth < start.get('month')) { newValue.month(start.get('month')); } } const onValueChange = this.props.onValueChange; if (onValueChange) { onValueChange(newValue); } }; onMonthChange = (month: string) => { const newValue = this.props.value.clone(); newValue.month(parseInt(month, 10)); const onValueChange = this.props.onValueChange; if (onValueChange) { onValueChange(newValue); } }; onInternalTypeChange = (e: RadioChangeEvent) => { this.onTypeChange(e.target.value); }; onTypeChange = (type: string) => { const onTypeChange = this.props.onTypeChange; if (onTypeChange) { onTypeChange(type); } }; getCalenderHeaderNode = (node: HTMLDivElement) => { this.calenderHeaderNode = node; }; getMonthYearSelections = (getPrefixCls: ConfigConsumerProps['getPrefixCls']) => { const { prefixCls: customizePrefixCls, type, value } = this.props; const prefixCls = getPrefixCls('fullcalendar', customizePrefixCls); const yearReactNode = this.getYearSelectElement(prefixCls, value.year()); const monthReactNode = type === 'month' ? this.getMonthSelectElement(prefixCls, value.month(), this.getMonthsLocale(value)) : null; return { yearReactNode, monthReactNode, }; }; getTypeSwitch = () => { const { locale, type, fullscreen } = this.props; const size = fullscreen ? 'default' : 'small'; return ( ); }; headerRenderCustom = (): React.ReactNode => { const { headerRender, type, onValueChange, value } = this.props; return headerRender({ value, type: type || 'month', onChange: onValueChange, onTypeChange: this.onTypeChange, }); }; renderHeader = ({ getPrefixCls }: ConfigConsumerProps) => { const { prefixCls, headerRender } = this.props; const typeSwitch = this.getTypeSwitch(); const { yearReactNode, monthReactNode } = this.getMonthYearSelections(getPrefixCls); return headerRender ? ( this.headerRenderCustom() ) : (
{yearReactNode} {monthReactNode} {typeSwitch}
); }; render() { return {this.renderHeader}; } }