2016-09-09 13:55:21 +08:00
|
|
|
import React from 'react';
|
2017-05-09 13:40:33 +08:00
|
|
|
import PropTypes from 'prop-types';
|
2016-09-09 13:55:21 +08:00
|
|
|
import moment from 'moment';
|
2015-11-11 00:15:12 +08:00
|
|
|
import FullCalendar from 'rc-calendar/lib/FullCalendar';
|
2017-10-20 14:54:38 +08:00
|
|
|
import LocaleReceiver from '../locale-provider/LocaleReceiver';
|
2016-01-05 14:42:06 +08:00
|
|
|
import { PREFIX_CLS } from './Constants';
|
2015-11-12 21:24:53 +08:00
|
|
|
import Header from './Header';
|
2016-10-01 00:17:31 +08:00
|
|
|
declare const require: Function;
|
2015-11-11 00:15:12 +08:00
|
|
|
|
2017-09-25 22:14:49 +08:00
|
|
|
export { HeaderProps } from './Header';
|
|
|
|
|
2016-01-14 16:46:38 +08:00
|
|
|
function noop() { return null; }
|
2015-11-11 00:15:12 +08:00
|
|
|
|
2016-01-14 16:46:38 +08:00
|
|
|
function zerofixed(v) {
|
2016-07-13 11:14:24 +08:00
|
|
|
if (v < 10) {
|
|
|
|
return `0${v}`;
|
|
|
|
}
|
2016-02-17 18:04:42 +08:00
|
|
|
return `${v}`;
|
2015-11-11 00:15:12 +08:00
|
|
|
}
|
2015-11-13 19:56:34 +08:00
|
|
|
|
2017-01-09 16:25:58 +08:00
|
|
|
export type CalendarMode = 'month' | 'year';
|
|
|
|
|
2016-08-19 17:11:06 +08:00
|
|
|
export interface CalendarProps {
|
|
|
|
prefixCls?: string;
|
|
|
|
className?: string;
|
2016-09-09 13:55:21 +08:00
|
|
|
value?: moment.Moment;
|
|
|
|
defaultValue?: moment.Moment;
|
2017-01-09 16:25:58 +08:00
|
|
|
mode?: CalendarMode;
|
2016-08-19 17:11:06 +08:00
|
|
|
fullscreen?: boolean;
|
2016-09-09 13:55:21 +08:00
|
|
|
dateCellRender?: (date: moment.Moment) => React.ReactNode;
|
|
|
|
monthCellRender?: (date: moment.Moment) => React.ReactNode;
|
2017-03-02 13:35:07 +08:00
|
|
|
dateFullCellRender?: (date: moment.Moment) => React.ReactNode;
|
|
|
|
monthFullCellRender?: (date: moment.Moment) => React.ReactNode;
|
2016-08-19 17:11:06 +08:00
|
|
|
locale?: any;
|
|
|
|
style?: React.CSSProperties;
|
2017-01-09 16:25:58 +08:00
|
|
|
onPanelChange?: (date?: moment.Moment, mode?: CalendarMode) => void;
|
2017-03-03 15:26:23 +08:00
|
|
|
onSelect?: (date?: moment.Moment) => void;
|
2017-05-23 17:07:17 +08:00
|
|
|
disabledDate?: (current: moment.Moment) => boolean;
|
2017-01-09 16:25:58 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface CalendarState {
|
|
|
|
value?: moment.Moment;
|
|
|
|
mode?: CalendarMode;
|
2016-08-19 17:11:06 +08:00
|
|
|
}
|
|
|
|
|
2017-01-09 16:25:58 +08:00
|
|
|
export default class Calendar extends React.Component<CalendarProps, CalendarState> {
|
2016-03-29 14:01:10 +08:00
|
|
|
static defaultProps = {
|
2016-03-31 17:37:24 +08:00
|
|
|
locale: {},
|
2016-03-29 14:01:10 +08:00
|
|
|
fullscreen: true,
|
|
|
|
prefixCls: PREFIX_CLS,
|
|
|
|
mode: 'month',
|
2017-03-03 15:26:23 +08:00
|
|
|
onSelect: noop,
|
|
|
|
onPanelChange: noop,
|
2016-07-13 11:14:24 +08:00
|
|
|
};
|
2016-03-29 14:01:10 +08:00
|
|
|
|
|
|
|
static propTypes = {
|
|
|
|
monthCellRender: PropTypes.func,
|
|
|
|
dateCellRender: PropTypes.func,
|
2017-03-02 13:35:07 +08:00
|
|
|
monthFullCellRender: PropTypes.func,
|
|
|
|
dateFullCellRender: PropTypes.func,
|
2016-03-29 14:01:10 +08:00
|
|
|
fullscreen: PropTypes.bool,
|
|
|
|
locale: PropTypes.object,
|
|
|
|
prefixCls: PropTypes.string,
|
|
|
|
className: PropTypes.string,
|
|
|
|
style: PropTypes.object,
|
|
|
|
onPanelChange: PropTypes.func,
|
2016-09-09 13:55:21 +08:00
|
|
|
value: PropTypes.object,
|
2017-03-03 15:26:23 +08:00
|
|
|
onSelect: PropTypes.func,
|
2016-07-13 11:14:24 +08:00
|
|
|
};
|
2016-03-29 14:01:10 +08:00
|
|
|
|
2017-01-09 16:25:58 +08:00
|
|
|
constructor(props, context) {
|
|
|
|
super(props, context);
|
|
|
|
|
2016-12-02 15:07:33 +08:00
|
|
|
const value = props.value || props.defaultValue || moment();
|
2016-12-06 14:21:17 +08:00
|
|
|
if (!moment.isMoment(value)) {
|
2016-12-02 15:07:33 +08:00
|
|
|
throw new Error(
|
|
|
|
'The value/defaultValue of Calendar must be a moment object after `antd@2.0`, ' +
|
2017-07-19 13:59:53 +08:00
|
|
|
'see: https://u.ant.design/calendar-value',
|
2016-12-02 15:07:33 +08:00
|
|
|
);
|
|
|
|
}
|
2015-11-12 21:24:53 +08:00
|
|
|
this.state = {
|
2016-12-02 15:07:33 +08:00
|
|
|
value,
|
2015-11-13 20:32:54 +08:00
|
|
|
mode: props.mode,
|
2015-11-12 21:24:53 +08:00
|
|
|
};
|
|
|
|
}
|
2016-03-29 14:01:10 +08:00
|
|
|
|
2017-01-09 16:25:58 +08:00
|
|
|
componentWillReceiveProps(nextProps: CalendarProps) {
|
2015-11-15 15:46:05 +08:00
|
|
|
if ('value' in nextProps) {
|
|
|
|
this.setState({
|
2016-09-09 13:55:21 +08:00
|
|
|
value: nextProps.value,
|
2015-11-15 15:46:05 +08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2016-03-29 14:01:10 +08:00
|
|
|
|
2016-09-09 13:55:21 +08:00
|
|
|
monthCellRender = (value) => {
|
2016-10-24 12:04:26 +08:00
|
|
|
const { prefixCls, monthCellRender = noop as Function } = this.props;
|
2016-01-07 14:21:29 +08:00
|
|
|
return (
|
|
|
|
<div className={`${prefixCls}-month`}>
|
|
|
|
<div className={`${prefixCls}-value`}>
|
2016-09-09 13:55:21 +08:00
|
|
|
{value.localeData().monthsShort(value)}
|
2016-01-07 14:21:29 +08:00
|
|
|
</div>
|
|
|
|
<div className={`${prefixCls}-content`}>
|
2016-10-24 12:04:26 +08:00
|
|
|
{monthCellRender(value)}
|
2016-01-07 14:21:29 +08:00
|
|
|
</div>
|
2015-11-13 22:34:49 +08:00
|
|
|
</div>
|
2016-01-07 14:21:29 +08:00
|
|
|
);
|
2015-11-11 00:15:12 +08:00
|
|
|
}
|
2016-03-29 14:01:10 +08:00
|
|
|
|
2016-03-23 19:50:44 +08:00
|
|
|
dateCellRender = (value) => {
|
2016-10-24 12:04:26 +08:00
|
|
|
const { prefixCls, dateCellRender = noop as Function } = this.props;
|
2016-01-07 14:21:29 +08:00
|
|
|
return (
|
|
|
|
<div className={`${prefixCls}-date`}>
|
|
|
|
<div className={`${prefixCls}-value`}>
|
2016-09-09 13:55:21 +08:00
|
|
|
{zerofixed(value.date())}
|
2016-01-07 14:21:29 +08:00
|
|
|
</div>
|
|
|
|
<div className={`${prefixCls}-content`}>
|
2016-10-24 12:04:26 +08:00
|
|
|
{dateCellRender(value)}
|
2016-01-07 14:21:29 +08:00
|
|
|
</div>
|
2015-11-13 22:34:49 +08:00
|
|
|
</div>
|
2016-01-07 14:21:29 +08:00
|
|
|
);
|
2015-11-11 00:15:12 +08:00
|
|
|
}
|
2016-03-29 14:01:10 +08:00
|
|
|
|
2017-10-24 14:08:39 +08:00
|
|
|
getDefaultLocale() {
|
|
|
|
const locale = require('./locale/en_US');
|
|
|
|
return locale.default || locale;
|
|
|
|
}
|
|
|
|
|
2017-03-03 15:26:23 +08:00
|
|
|
setValue = (value, way: 'select' | 'changePanel') => {
|
|
|
|
if (!('value' in this.props)) {
|
2015-11-12 23:51:38 +08:00
|
|
|
this.setState({ value });
|
|
|
|
}
|
2017-03-03 15:26:23 +08:00
|
|
|
if (way === 'select') {
|
|
|
|
if (this.props.onSelect) {
|
|
|
|
this.props.onSelect(value);
|
|
|
|
}
|
|
|
|
} else if (way === 'changePanel') {
|
|
|
|
this.onPanelChange(value, this.state.mode);
|
2016-10-24 12:04:26 +08:00
|
|
|
}
|
2015-11-12 21:24:53 +08:00
|
|
|
}
|
2016-03-29 14:01:10 +08:00
|
|
|
|
2016-03-23 19:50:44 +08:00
|
|
|
setType = (type) => {
|
2015-11-13 20:32:54 +08:00
|
|
|
const mode = (type === 'date') ? 'month' : 'year';
|
|
|
|
if (this.state.mode !== mode) {
|
|
|
|
this.setState({ mode });
|
2017-03-03 15:26:23 +08:00
|
|
|
this.onPanelChange(this.state.value, mode);
|
2015-11-13 20:32:54 +08:00
|
|
|
}
|
2015-11-12 21:24:53 +08:00
|
|
|
}
|
2016-03-29 14:01:10 +08:00
|
|
|
|
2017-03-03 15:26:23 +08:00
|
|
|
onHeaderValueChange = (value) => {
|
|
|
|
this.setValue(value, 'changePanel');
|
|
|
|
}
|
|
|
|
|
|
|
|
onHeaderTypeChange = (type) => {
|
|
|
|
this.setType(type);
|
|
|
|
}
|
|
|
|
|
|
|
|
onPanelChange(value, mode) {
|
|
|
|
const { onPanelChange } = this.props;
|
|
|
|
if (onPanelChange) {
|
|
|
|
onPanelChange(value, mode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
onSelect = (value) => {
|
|
|
|
this.setValue(value, 'select');
|
|
|
|
}
|
|
|
|
|
2017-10-20 14:54:38 +08:00
|
|
|
renderCalendar = (locale, localeCode) => {
|
|
|
|
const { state, props } = this;
|
2017-01-09 16:25:58 +08:00
|
|
|
const { value, mode } = state;
|
|
|
|
if (value && localeCode) {
|
|
|
|
value.locale(localeCode);
|
|
|
|
}
|
2017-03-02 13:35:07 +08:00
|
|
|
const { prefixCls, style, className, fullscreen, dateFullCellRender, monthFullCellRender } = props;
|
2015-11-13 20:32:54 +08:00
|
|
|
const type = (mode === 'year') ? 'month' : 'date';
|
2015-11-11 00:15:12 +08:00
|
|
|
|
2015-11-15 21:40:01 +08:00
|
|
|
let cls = className || '';
|
|
|
|
if (fullscreen) {
|
2016-02-17 18:04:42 +08:00
|
|
|
cls += (` ${prefixCls}-fullscreen`);
|
2015-11-15 21:40:01 +08:00
|
|
|
}
|
|
|
|
|
2017-03-02 13:35:07 +08:00
|
|
|
const monthCellRender = monthFullCellRender || this.monthCellRender;
|
|
|
|
const dateCellRender = dateFullCellRender || this.dateCellRender;
|
|
|
|
|
2015-11-12 21:24:53 +08:00
|
|
|
return (
|
2015-11-15 21:40:01 +08:00
|
|
|
<div className={cls} style={style}>
|
2015-11-12 21:24:53 +08:00
|
|
|
<Header
|
2015-11-12 23:51:38 +08:00
|
|
|
fullscreen={fullscreen}
|
2015-11-12 21:24:53 +08:00
|
|
|
type={type}
|
|
|
|
value={value}
|
2015-11-16 20:10:29 +08:00
|
|
|
locale={locale.lang}
|
2015-11-13 21:24:25 +08:00
|
|
|
prefixCls={prefixCls}
|
2017-03-03 15:26:23 +08:00
|
|
|
onTypeChange={this.onHeaderTypeChange}
|
|
|
|
onValueChange={this.onHeaderValueChange}
|
2016-06-06 13:54:10 +08:00
|
|
|
/>
|
2015-11-12 21:24:53 +08:00
|
|
|
<FullCalendar
|
|
|
|
{...props}
|
2015-12-27 23:39:33 +08:00
|
|
|
Select={noop}
|
2015-11-16 20:10:29 +08:00
|
|
|
locale={locale.lang}
|
2015-11-12 21:24:53 +08:00
|
|
|
type={type}
|
2015-11-13 21:24:25 +08:00
|
|
|
prefixCls={prefixCls}
|
2015-11-12 21:24:53 +08:00
|
|
|
showHeader={false}
|
|
|
|
value={value}
|
2017-03-02 13:35:07 +08:00
|
|
|
monthCellRender={monthCellRender}
|
|
|
|
dateCellRender={dateCellRender}
|
2017-03-03 15:26:23 +08:00
|
|
|
onSelect={this.onSelect}
|
2016-06-06 13:54:10 +08:00
|
|
|
/>
|
2015-11-12 21:24:53 +08:00
|
|
|
</div>
|
|
|
|
);
|
2015-11-11 00:15:12 +08:00
|
|
|
}
|
2017-10-20 14:54:38 +08:00
|
|
|
|
|
|
|
render() {
|
|
|
|
return (
|
|
|
|
<LocaleReceiver
|
|
|
|
componentName="Calendar"
|
2017-10-24 14:08:39 +08:00
|
|
|
defaultLocale={this.getDefaultLocale}
|
2017-10-20 14:54:38 +08:00
|
|
|
>
|
|
|
|
{this.renderCalendar}
|
|
|
|
</LocaleReceiver>
|
|
|
|
);
|
|
|
|
}
|
2015-11-11 00:15:12 +08:00
|
|
|
}
|