Calendar.js 2.09 KB
import React, { useEffect, useReducer, useState } from "react";
import { useNavigate, Route, Routes } from "react-router-dom";

import Month from "../components/Month";
import Header from "../components/Header";
import Side from "../components/Side";
import localforage from "localforage";
import axios from "axios";

export const CalendarStateContext = React.createContext();

const render = (subsObj, args) => {
  switch (args.type) {
    case "CHECKED":
      const sub = subsObj[args.subjectID];
      sub.status = !sub.status;
      axios.put("http://localhost:3001/db/user-subject/check", {
        userID: sub.userID,
        subjectID: args.subjectID,
        status: +sub.status,
      });
      return { ...subsObj, [args.subjectID]: sub };
    case "INIT":
      return args.subsObj;
    default:
      return subsObj;
  }
};

const Calendar = () => {
  console.log("visit Calendar");

  const [state, setState] = useState({
    scope: "month",
    date: new Date(),
  });
  const [subsObj, dispatch] = useReducer(render, {});

  const navigate = useNavigate();
  useEffect(() => {
    async function onMount() {
      if (!(await localforage.getItem("session"))) return navigate("/login");
      // get user's subjects
      const userID = await localforage.getItem("userID");
      const { data: subjects } = await axios.get(
        "http://localhost:3001/db/user-subject",
        { params: { userID } }
      );
      let tsubsObj = {};
      for (const sub of subjects) {
        tsubsObj[sub.subjectID] = sub;
      }
      dispatch({ type: "INIT", subsObj: tsubsObj });
    }
    onMount();
  }, [navigate]);

  return (
    <CalendarStateContext.Provider
      value={{ state, setState, subsObj, dispatch }}
    >
      <div className="Calendar">
        <Header />
        <div className="content">
          <Side />
          <Routes>
            <Route path="/month/*" element={<Month />} />
            <Route path="/week/*" element={<></>} />
            <Route path="/day/*" element={<></>} />
          </Routes>
        </div>
      </div>
    </CalendarStateContext.Provider>
  );
};

export default Calendar;