import { observable, action, makeObservable, runInAction, computed } from "mobx"; // helpers import { generateCalendarData } from "helpers/calendar.helper"; import { getWeekNumberOfDate } from "helpers/date-time.helper"; // types import { RootStore } from "./root"; import { ICalendarPayload, ICalendarWeek } from "components/issues"; export interface ICalendarStore { calendarFilters: { activeMonthDate: Date; activeWeekDate: Date; }; calendarPayload: ICalendarPayload | null; // action updateCalendarFilters: (filters: Partial<{ activeMonthDate: Date; activeWeekDate: Date }>) => void; updateCalendarPayload: (date: Date) => void; // computed allWeeksOfActiveMonth: | { [weekNumber: string]: ICalendarWeek; } | undefined; activeWeekNumber: number; allDaysOfActiveWeek: ICalendarWeek | undefined; } class CalendarStore implements ICalendarStore { loader: boolean = false; error: any | null = null; // observables calendarFilters: { activeMonthDate: Date; activeWeekDate: Date } = { activeMonthDate: new Date(), activeWeekDate: new Date(), }; calendarPayload: ICalendarPayload | null = null; // root store rootStore; constructor(_rootStore: RootStore) { makeObservable(this, { loader: observable.ref, error: observable.ref, // observables calendarFilters: observable.ref, calendarPayload: observable.ref, // actions updateCalendarFilters: action, updateCalendarPayload: action, //computed allWeeksOfActiveMonth: computed, activeWeekNumber: computed, allDaysOfActiveWeek: computed, }); this.rootStore = _rootStore; this.initCalendar(); } get allWeeksOfActiveMonth() { if (!this.calendarPayload) return undefined; const { activeMonthDate } = this.calendarFilters; return this.calendarPayload[`y-${activeMonthDate.getFullYear()}`][`m-${activeMonthDate.getMonth()}`]; } get activeWeekNumber() { return getWeekNumberOfDate(this.calendarFilters.activeWeekDate); } get allDaysOfActiveWeek() { if (!this.calendarPayload) return undefined; const { activeWeekDate } = this.calendarFilters; return this.calendarPayload[`y-${activeWeekDate.getFullYear()}`][`m-${activeWeekDate.getMonth()}`][ `w-${this.activeWeekNumber}` ]; } updateCalendarFilters = (filters: Partial<{ activeMonthDate: Date; activeWeekDate: Date }>) => { this.updateCalendarPayload(filters.activeMonthDate || filters.activeWeekDate || new Date()); runInAction(() => { this.calendarFilters = { ...this.calendarFilters, ...filters, }; }); }; updateCalendarPayload = (date: Date) => { if (!this.calendarPayload) return null; const nextDate = new Date(date); runInAction(() => { this.calendarPayload = generateCalendarData(this.calendarPayload, nextDate); }); }; initCalendar = () => { const newCalendarPayload = generateCalendarData(null, new Date()); runInAction(() => { this.calendarPayload = newCalendarPayload; }); }; } export default CalendarStore;