【无标题】

React 日历组件

使用React编写一个日历组件,能够选择月份,添加标记,如标记当天,节日等。

思路

  • 先把指定月份的日期全部渲染一遍,也就是先渲染基本的日期,不管当月有多少天,都要渲染出 7 * 5的单元格,如下图所示:
    ![在这里插入图片描述](https://img-blog.csdnimg.cn/0dd0f64a705b4c47b58b8d56b8a822a4.png在这里插入图片描述

  • 在遍历35天的时候,对于每一天看看有没有特殊的标记,比如对“今天”进行标记,另外,看看今天有没有deadline,如果有的话,把deadline对应的任务显示出来。

import moment from "moment";
import {useEffect, useState} from "react";

export const Calendar = (props) => {

  const {date, tasks, project} = props;
  const [days, setDays] = useState([]);
  useEffect(() => {
    if (date) {
      const newDays = [];
      let daysOfWeeks = [];
      let daysOfWeek = [];
      const daysOfMonth = date.daysInMonth();
      let startDateOfMonth = date.startOf('month');
      let startDateOfNextMonth = date.clone().add(1, 'months').startOf('month');

      let startDayOfMonth = date.startOf('month').day();


      if (startDayOfMonth === 0) {
        startDayOfMonth = 7;
      }


      for (let i = 0; i < startDayOfMonth - 1; i ++) {
        newDays.push({
          date: startDateOfMonth.clone().subtract(startDayOfMonth - 1 + i, 'days')
        });
      }


      for (let i = 0; i < daysOfMonth; i ++) {
        newDays.push({
          date: startDateOfMonth.clone().add(i, 'days')
        });
      }

      for (let i = 0, len = newDays.length; i < 42 - len; i ++) {
        newDays.push({
          date: startDateOfNextMonth.clone().add(i, 'days')
        });
      }
      for (let i = 0; i < newDays.length; i ++) {
        daysOfWeek.push(newDays[i]);
        if (daysOfWeek.length === 7) {
          daysOfWeeks.push(daysOfWeek);
          daysOfWeek = [];
        }
      }
      setDays(daysOfWeeks);
    }
  }, [date]);

  if (!date) {
    return <></>;
  }
  return (
    <div>
      <table className="table border-top">
        <thead>
          <tr>
            <th scope="col">MON</th>
            <th scope="col">TUE</th>
            <th scope="col">WED</th>
            <th scope="col">THU</th>
            <th scope="col">FRI</th>
            <th scope="col">SAT</th>
            <th scope="col">SUN</th>
          </tr>
        </thead>
        <tbody>
          {days.map(daysOfWeek => {
            return (
              <tr>
                {daysOfWeek.map(day => {
                  const tasks = tasks.filter(task => moment(task.ddl).isSame(day.date, 'day'));

                  return (
                    <td>
                      {day.date.date()}
                      <div>

                        {day.date.isSame(moment(), 'day') && (
                          <span className="badge text-bg-info text-white">
                           Today
                          </span>)}
                        {project && day.date.isSame(project.ddl, 'day') && (
                          <span className={'badge text-bg-danger text-white'}>ddl</span>
                        )}


                        {tasks.map(task => {
                          return (
                            <div key={task._id}
                                 className={'rounded-1 text-white mt-1'}>
                              <span
                                style={{borderLeft: '5px solid #2dc07c', paddingLeft: '5px', borderRadius: '1px'}}>
                                {task.homework}
                              </span>
                            </div>
                          )
                        })}
                      </div>
                    </td>
                  )
                })}
              </tr>
            )
          })}

        </tbody>
      </table>
    </div>
  )
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值