import dayjs, { type Dayjs } from 'dayjs'; import React from 'react'; import lunisolar from 'lunisolar'; import zhCn from 'lunisolar/locale/zh-cn'; import { createStyles } from 'antd-style'; import classNames from 'classnames'; import { Calendar, Col, Radio, Row, Select } from 'antd'; import type { CalendarProps } from 'antd'; lunisolar.locale(zhCn); const useStyle = createStyles(({ token, css, cx }) => { const lunar = css` color: ${token.colorTextTertiary}; font-size: ${token.fontSizeSM}px; `; return { wrapper: css` width: 400px; border: 1px solid ${token.colorBorderSecondary}; border-radius: ${token.borderRadiusOuter}; padding: 5px; `, dateCell: css` position: relative; &:before { content: ''; position: absolute; left: 0; right: 0; top: 0; bottom: 0; margin: auto; max-width: 40px; max-height: 40px; background: transparent; transition: background 300ms; border-radius: ${token.borderRadiusOuter}px; border: 1px solid transparent; box-sizing: border-box; } &:hover:before { background: rgba(0, 0, 0, 0.04); } `, today: css` &:before { border: 1px solid ${token.colorPrimary}; } `, text: css` position: relative; z-index: 1; `, lunar, current: css` color: ${token.colorTextLightSolid}; &:before { background: ${token.colorPrimary}; } &:hover:before { background: ${token.colorPrimary}; opacity: 0.8; } .${cx(lunar)} { color: ${token.colorTextLightSolid}; opacity: 0.9; } `, monthCell: css` width: 120px; color: ${token.colorTextBase}; border-radius: ${token.borderRadiusOuter}px; padding: 5px 0; &:hover { background: rgba(0, 0, 0, 0.04); } `, monthCellCurrent: css` color: ${token.colorTextLightSolid}; background: ${token.colorPrimary}; &:hover { background: ${token.colorPrimary}; opacity: 0.8; } `, }; }); const App: React.FC = () => { const { styles } = useStyle({ test: true }); const [selectDate, setSelectDate] = React.useState(dayjs()); const onPanelChange = (value: Dayjs, mode: CalendarProps['mode']) => { console.log(value.format('YYYY-MM-DD'), mode); setSelectDate(value); }; const onDateChange = (value: Dayjs) => { setSelectDate(value); }; const cellRender: CalendarProps['fullCellRender'] = (date, info) => { const d = lunisolar(date.toDate()); const lunar = d.lunar.getDayName(); const solarTerm = d.solarTerm?.name; if (info.type === 'date') { return React.cloneElement(info.originNode, { ...info.originNode.props, className: classNames(styles.dateCell, { [styles.current]: selectDate.isSame(date, 'date'), [styles.today]: date.isSame(dayjs(), 'date'), }), children: (
{date.get('date')} {info.type === 'date' &&
{solarTerm || lunar}
}
), }); } if (info.type === 'month') { // Due to the fact that a solar month is part of the lunar month X and part of the lunar month X+1, // when rendering a month, always take X as the lunar month of the month const d2 = lunisolar(new Date(date.get('year'), date.get('month'))); const month = d2.lunar.getMonthName(); return (
{date.get('month') + 1}月({month})
); } }; const getYearLabel = (year: number) => { const d = lunisolar(new Date(year + 1, 0)); return `${year}年(${d.format('cYcZ年')})`; }; const getMonthLabel = (month: number, value: Dayjs) => { const d = lunisolar(new Date(value.year(), month)); const lunar = d.lunar.getMonthName(); return `${month + 1}月(${lunar})`; }; return (
{ const start = 0; const end = 12; const monthOptions = []; let current = value.clone(); const localeData = value.localeData(); const months = []; for (let i = 0; i < 12; i++) { current = current.month(i); months.push(localeData.monthsShort(current)); } for (let i = start; i < end; i++) { monthOptions.push({ label: getMonthLabel(i, value), value: i, }); } const year = value.year(); const month = value.month(); const options = []; for (let i = year - 10; i < year + 10; i += 1) { options.push({ label: getYearLabel(i), value: i, }); } return ( { const now = value.clone().month(newMonth); onChange(now); }} /> onTypeChange(e.target.value)} value={type} > ); }} />
); }; export default App;