| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317 | /* eslint-disable max-len */import React from 'react';import PropTypes from 'prop-types';import YearAndMonthFoundation, { MonthScrollItem, YearAndMonthAdapter, YearAndMonthFoundationProps, YearAndMonthFoundationState, YearScrollItem } from '@douyinfe/semi-foundation/datePicker/yearAndMonthFoundation';import BaseComponent, { BaseProps } from '../_base/baseComponent';import ScrollList from '../scrollList/index';import ScrollItem from '../scrollList/scrollItem';import { getYears } from '@douyinfe/semi-foundation/datePicker/_utils/index';import IconButton from '../iconButton';import { IconChevronLeft } from '@douyinfe/semi-icons';import { BASE_CLASS_PREFIX } from '@douyinfe/semi-foundation/base/constants';import { noop, stubFalse, isEqual } from 'lodash';import { setYear, setMonth, set } from 'date-fns';import { Locale } from '../locale/interface';import { strings } from '@douyinfe/semi-foundation/datePicker/constants';import { PanelType } from '@douyinfe/semi-foundation/datePicker/monthsGridFoundation';const prefixCls = `${BASE_CLASS_PREFIX}-datepicker`;export interface YearAndMonthProps extends YearAndMonthFoundationProps, BaseProps {    locale?: Locale['DatePicker']}export type YearAndMonthState = YearAndMonthFoundationState;class YearAndMonth extends BaseComponent<YearAndMonthProps, YearAndMonthState> {    static propTypes = {        currentYear: PropTypes.object,        currentMonth: PropTypes.object,        onSelect: PropTypes.func,        locale: PropTypes.object,        localeCode: PropTypes.string,        monthCycled: PropTypes.bool,        yearCycled: PropTypes.bool,        noBackBtn: PropTypes.bool,        disabledDate: PropTypes.func,        density: PropTypes.string,        presetPosition: PropTypes.oneOf(strings.PRESET_POSITION_SET),        renderQuickControls: PropTypes.node,        renderDateInput: PropTypes.node,        type: PropTypes.oneOf(strings.TYPE_SET),    };    static defaultProps = {        disabledDate: stubFalse,        monthCycled: false,        yearCycled: false,        noBackBtn: false,        onSelect: noop,        type: 'month',    };    foundation: YearAndMonthFoundation;    yearRef: React.RefObject<ScrollItem<YearScrollItem>>;    monthRef: React.RefObject<ScrollItem<MonthScrollItem>>;    constructor(props: YearAndMonthProps) {        super(props);        const now = new Date();        let { currentYear, currentMonth } = props;        const currentLeftYear = currentYear.left || now.getFullYear();        const currentLeftMonth = currentMonth.left || now.getMonth() + 1;        currentYear = { left: currentLeftYear, right: currentLeftYear };        currentMonth = { left: currentLeftMonth, right: currentMonth.right || currentLeftMonth + 1 };        this.state = {            years: getYears().map(year => ({                value: year,                year,            })),            months: Array(12)                .fill(0)                .map((v, idx) => ({                    value: idx + 1,                    month: idx + 1,                })),            currentYear,            currentMonth,        };        this.yearRef = React.createRef();        this.monthRef = React.createRef();        this.foundation = new YearAndMonthFoundation(this.adapter);    }    get adapter(): YearAndMonthAdapter {        return {            ...super.adapter,            // updateYears: years => this.setState({ years }),            // updateMonths: months => this.setState({ months }),            setCurrentYear: (currentYear, cb) => this.setState({ currentYear }, cb),            setCurrentMonth: currentMonth => this.setState({ currentMonth }),            setCurrentYearAndMonth: (currentYear, currentMonth) => this.setState({ currentYear, currentMonth }),            notifySelectYear: (year) =>                this.props.onSelect({                    currentMonth: this.state.currentMonth,                    currentYear: year,                }),            notifySelectMonth: (month) =>                this.props.onSelect({                    currentYear: this.state.currentYear,                    currentMonth: month,                }),            notifySelectYearAndMonth: (year, month) =>                this.props.onSelect({                    currentYear: year,                    currentMonth: month,                }),            notifyBackToMain: () => this.props.onBackToMain(),        };    }    static getDerivedStateFromProps(props: YearAndMonthProps, state: YearAndMonthState) {        const willUpdateStates: Partial<YearAndMonthState> = {};        if (!isEqual(props.currentYear, state.currentYear) && props.currentYear.left !== 0) {            willUpdateStates.currentYear = props.currentYear;        }        if (!isEqual(props.currentMonth, state.currentMonth) && props.currentMonth.left !== 0) {            willUpdateStates.currentMonth = props.currentMonth;        }        return willUpdateStates;    }    renderColYear(panelType: PanelType) {        const { years, currentYear, currentMonth, months } = this.state;        const { disabledDate, localeCode, yearCycled, yearAndMonthOpts } = this.props;        const currentDate = setMonth(Date.now(), currentMonth[panelType] - 1);        const left = strings.PANEL_TYPE_LEFT;        const right = strings.PANEL_TYPE_RIGHT;        const needDisabled = (year) => {            if (panelType === right && currentYear[left]) {                if (currentMonth[left] <= currentMonth[right]) {                    return currentYear[left] > year;                } else {                    return currentYear[left] >= year;                }            }            return false;        };        const list: any[] = years.map(({ value, year }) => {            const isAllMonthDisabled = months.every(({ month }) => {                return disabledDate(set(currentDate, { year, month: month - 1 }));            });            const isRightPanelDisabled = needDisabled(year);            return ({                year,                value, // Actual rendered text                disabled: isAllMonthDisabled || isRightPanelDisabled,            });        });        let transform = (val: string) => val;        if (localeCode === 'zh-CN' || localeCode === 'zh-TW') {            // Only Chinese needs to add [year] after the selected year            transform = val => `${val}年`;        }        return (            <ScrollItem                ref={this.yearRef}                cycled={yearCycled}                list={list}                transform={transform}                selectedIndex={years.findIndex(item => item.value === currentYear[panelType])}                type="year"                onSelect={item => this.selectYear(item as YearScrollItem, panelType)}                mode="normal"                {...yearAndMonthOpts}            />        );    }    selectYear = (item: YearScrollItem, panelType?: PanelType) => {        this.foundation.selectYear(item, panelType);    };    selectMonth = (item: MonthScrollItem, panelType?: PanelType) => {        this.foundation.selectMonth(item, panelType);    };    reselect = () => {        const refKeys = ['yearRef', 'monthRef'];        refKeys.forEach(key => {            const ref = this[key];            if (ref && ref.current && ref.current.scrollToIndex) {                ref.current.scrollToIndex();            }        });    };    renderColMonth(panelType: PanelType) {        const { months, currentMonth, currentYear } = this.state;        const { locale, localeCode, monthCycled, disabledDate, yearAndMonthOpts } = this.props;        let transform = (val: string) => val;        const currentDate = setYear(Date.now(), currentYear[panelType]);        const left = strings.PANEL_TYPE_LEFT;        const right = strings.PANEL_TYPE_RIGHT;        if (localeCode === 'zh-CN' || localeCode === 'zh-TW') {            // Only Chinese needs to add [month] after the selected month            transform = val => `${val}月`;        }        // i18n        const list: MonthScrollItem[] = months.map(({ value, month }) => {            const isRightPanelDisabled = panelType === right && currentMonth[left] && currentYear[left] === currentYear[right] && currentMonth[left] > month;            return ({                month,                disabled: disabledDate(setMonth(currentDate, month - 1)) || isRightPanelDisabled,                value: locale.fullMonths[value], // Actual rendered text            });        });        const selectedIndex = list.findIndex(item => item.month === currentMonth[panelType]);        return (            <ScrollItem                ref={this.monthRef}                cycled={monthCycled}                list={list}                transform={transform}                selectedIndex={selectedIndex}                type="month"                onSelect={item => this.selectMonth(item as MonthScrollItem, panelType)}                mode='normal'                {...yearAndMonthOpts}            />        );    }    backToMain: React.MouseEventHandler<HTMLButtonElement> = e => {        e.nativeEvent.stopImmediatePropagation();        this.foundation.backToMain();    };    renderPanel(panelType: PanelType) {        return (            <>                <ScrollList>                    {this.renderColYear(panelType)}                    {this.renderColMonth(panelType)}                </ScrollList>            </>        );    }    render() {        const { locale, noBackBtn, density, presetPosition, renderQuickControls, renderDateInput, type } = this.props;        const prefix = `${prefixCls}-yearmonth-header`;        const bodyCls = `${prefixCls}-yearmonth-body`;        // i18n        const selectDateText = locale.selectDate;        const iconSize = density === 'compact' ? 'default' : 'large';        const buttonSize = density === 'compact' ? 'small' : 'default';        const panelTypeLeft = strings.PANEL_TYPE_LEFT;        const panelTypeRight = strings.PANEL_TYPE_RIGHT;        let content = null;        if (type === 'month') {            content = this.renderPanel(panelTypeLeft);        } else {            content = (                <div className={bodyCls}>                    {this.renderPanel(panelTypeLeft)}                    {this.renderPanel(panelTypeRight)}                </div>            );        }        return (            <React.Fragment>                {noBackBtn ? null : (                    <div className={prefix}>                        <IconButton                            noHorizontalPadding={false}                            icon={<IconChevronLeft aria-hidden size={iconSize} />}                            size={buttonSize}                            onClick={this.backToMain}                        >                            <span>{selectDateText}</span>                        </IconButton>                    </div>                )}                {                    presetPosition ? (                        <div style={{ display: 'flex' }}>                            {/* todo: monthRange does not support presetPosition temporarily */}                            {presetPosition === "left" && type !== 'monthRange' && renderQuickControls}                            <div>                                {renderDateInput}                                {content}                             </div>                            {/* todo: monthRange does not support presetPosition temporarily */}                            {presetPosition === "right" && type !== 'monthRange' && renderQuickControls}                        </div>                    ) :                        <>                            {renderDateInput}                            {content}                         </>                }            </React.Fragment>        );    }}export default YearAndMonth;
 |