import { isFuture, isPast, isToday } from "date-fns"; import set from "lodash/set"; import sortBy from "lodash/sortBy"; import { action, computed, observable, makeObservable, runInAction } from "mobx"; import { computedFn } from "mobx-utils"; // types import { ICycle, CycleDateCheckData, TCyclePlotType } from "@plane/types"; // helpers import { orderCycles, shouldFilterCycle } from "@/helpers/cycle.helper"; import { getDate } from "@/helpers/date-time.helper"; // plane web store import { RootStore } from "@/plane-web/store/root.store"; // services import { CycleService } from "@/services/cycle.service"; import { CycleArchiveService } from "@/services/cycle_archive.service"; import { IssueService } from "@/services/issue"; import { ProjectService } from "@/services/project"; export interface ICycleStore { // loaders loader: boolean; // observables fetchedMap: Record; cycleMap: Record; plotType: Record; activeCycleIdMap: Record; // computed currentProjectCycleIds: string[] | null; currentProjectCompletedCycleIds: string[] | null; currentProjectUpcomingCycleIds: string[] | null; currentProjectIncompleteCycleIds: string[] | null; currentProjectDraftCycleIds: string[] | null; currentProjectActiveCycleId: string | null; currentProjectArchivedCycleIds: string[] | null; // computed actions getFilteredCycleIds: (projectId: string, sortByManual: boolean) => string[] | null; getFilteredCompletedCycleIds: (projectId: string) => string[] | null; getFilteredArchivedCycleIds: (projectId: string) => string[] | null; getCycleById: (cycleId: string) => ICycle | null; getCycleNameById: (cycleId: string) => string | undefined; getActiveCycleById: (cycleId: string) => ICycle | null; getProjectCycleIds: (projectId: string) => string[] | null; getPlotTypeByCycleId: (cycleId: string) => TCyclePlotType; // actions validateDate: (workspaceSlug: string, projectId: string, payload: CycleDateCheckData) => Promise; setPlotType: (cycleId: string, plotType: TCyclePlotType) => void; // fetch fetchWorkspaceCycles: (workspaceSlug: string) => Promise; fetchAllCycles: (workspaceSlug: string, projectId: string) => Promise; fetchActiveCycle: (workspaceSlug: string, projectId: string) => Promise; fetchArchivedCycles: (workspaceSlug: string, projectId: string) => Promise; fetchArchivedCycleDetails: (workspaceSlug: string, projectId: string, cycleId: string) => Promise; fetchCycleDetails: (workspaceSlug: string, projectId: string, cycleId: string) => Promise; // crud createCycle: (workspaceSlug: string, projectId: string, data: Partial) => Promise; updateCycleDetails: ( workspaceSlug: string, projectId: string, cycleId: string, data: Partial ) => Promise; deleteCycle: (workspaceSlug: string, projectId: string, cycleId: string) => Promise; // favorites addCycleToFavorites: (workspaceSlug: string, projectId: string, cycleId: string) => Promise; removeCycleFromFavorites: (workspaceSlug: string, projectId: string, cycleId: string) => Promise; // archive archiveCycle: (workspaceSlug: string, projectId: string, cycleId: string) => Promise; restoreCycle: (workspaceSlug: string, projectId: string, cycleId: string) => Promise; } export class CycleStore implements ICycleStore { // observables loader: boolean = false; cycleMap: Record = {}; plotType: Record = {}; activeCycleIdMap: Record = {}; //loaders fetchedMap: Record = {}; // root store rootStore; // services projectService; issueService; cycleService; cycleArchiveService; constructor(_rootStore: RootStore) { makeObservable(this, { // observables loader: observable.ref, cycleMap: observable, plotType: observable.ref, activeCycleIdMap: observable, fetchedMap: observable, // computed currentProjectCycleIds: computed, currentProjectCompletedCycleIds: computed, currentProjectUpcomingCycleIds: computed, currentProjectIncompleteCycleIds: computed, currentProjectDraftCycleIds: computed, currentProjectActiveCycleId: computed, currentProjectArchivedCycleIds: computed, // actions setPlotType: action, fetchWorkspaceCycles: action, fetchAllCycles: action, fetchActiveCycle: action, fetchArchivedCycles: action, fetchArchivedCycleDetails: action, fetchCycleDetails: action, createCycle: action, updateCycleDetails: action, deleteCycle: action, addCycleToFavorites: action, removeCycleFromFavorites: action, archiveCycle: action, restoreCycle: action, }); this.rootStore = _rootStore; // services this.projectService = new ProjectService(); this.issueService = new IssueService(); this.cycleService = new CycleService(); this.cycleArchiveService = new CycleArchiveService(); } // computed /** * returns all cycle ids for a project */ get currentProjectCycleIds() { const projectId = this.rootStore.router.projectId; if (!projectId || !this.fetchedMap[projectId]) return null; let allCycles = Object.values(this.cycleMap ?? {}).filter((c) => c?.project_id === projectId && !c?.archived_at); allCycles = sortBy(allCycles, [(c) => c.sort_order]); const allCycleIds = allCycles.map((c) => c.id); return allCycleIds; } /** * returns all completed cycle ids for a project */ get currentProjectCompletedCycleIds() { const projectId = this.rootStore.router.projectId; if (!projectId || !this.fetchedMap[projectId]) return null; let completedCycles = Object.values(this.cycleMap ?? {}).filter((c) => { const endDate = getDate(c.end_date); const hasEndDatePassed = endDate && isPast(endDate); const isEndDateToday = endDate && isToday(endDate); return c.project_id === projectId && hasEndDatePassed && !isEndDateToday && !c?.archived_at; }); completedCycles = sortBy(completedCycles, [(c) => c.sort_order]); const completedCycleIds = completedCycles.map((c) => c.id); return completedCycleIds; } /** * returns all upcoming cycle ids for a project */ get currentProjectUpcomingCycleIds() { const projectId = this.rootStore.router.projectId; if (!projectId || !this.fetchedMap[projectId]) return null; let upcomingCycles = Object.values(this.cycleMap ?? {}).filter((c) => { const startDate = getDate(c.start_date); const isStartDateUpcoming = startDate && isFuture(startDate); return c.project_id === projectId && isStartDateUpcoming && !c?.archived_at; }); upcomingCycles = sortBy(upcomingCycles, [(c) => c.sort_order]); const upcomingCycleIds = upcomingCycles.map((c) => c.id); return upcomingCycleIds; } /** * returns all incomplete cycle ids for a project */ get currentProjectIncompleteCycleIds() { const projectId = this.rootStore.router.projectId; if (!projectId || !this.fetchedMap[projectId]) return null; let incompleteCycles = Object.values(this.cycleMap ?? {}).filter((c) => { const endDate = getDate(c.end_date); const hasEndDatePassed = endDate && isPast(endDate); return c.project_id === projectId && !hasEndDatePassed && !c?.archived_at; }); incompleteCycles = sortBy(incompleteCycles, [(c) => c.sort_order]); const incompleteCycleIds = incompleteCycles.map((c) => c.id); return incompleteCycleIds; } /** * returns all draft cycle ids for a project */ get currentProjectDraftCycleIds() { const projectId = this.rootStore.router.projectId; if (!projectId || !this.fetchedMap[projectId]) return null; let draftCycles = Object.values(this.cycleMap ?? {}).filter( (c) => c.project_id === projectId && !c.start_date && !c.end_date && !c?.archived_at ); draftCycles = sortBy(draftCycles, [(c) => c.sort_order]); const draftCycleIds = draftCycles.map((c) => c.id); return draftCycleIds; } /** * returns active cycle id for a project */ get currentProjectActiveCycleId() { const projectId = this.rootStore.router.projectId; if (!projectId) return null; const activeCycle = Object.keys(this.activeCycleIdMap ?? {}).find( (cycleId) => this.cycleMap?.[cycleId]?.project_id === projectId ); return activeCycle || null; } /** * returns all archived cycle ids for a project */ get currentProjectArchivedCycleIds() { const projectId = this.rootStore.router.projectId; if (!projectId || !this.fetchedMap[projectId]) return null; let archivedCycles = Object.values(this.cycleMap ?? {}).filter( (c) => c.project_id === projectId && !!c.archived_at ); archivedCycles = sortBy(archivedCycles, [(c) => c.sort_order]); const archivedCycleIds = archivedCycles.map((c) => c.id); return archivedCycleIds; } /** * @description returns filtered cycle ids based on display filters and filters * @param {TCycleDisplayFilters} displayFilters * @param {TCycleFilters} filters * @returns {string[] | null} */ getFilteredCycleIds = computedFn((projectId: string, sortByManual: boolean) => { const filters = this.rootStore.cycleFilter.getFiltersByProjectId(projectId); const searchQuery = this.rootStore.cycleFilter.searchQuery; if (!this.fetchedMap[projectId]) return null; let cycles = Object.values(this.cycleMap ?? {}).filter( (c) => c.project_id === projectId && !c.archived_at && c.name.toLowerCase().includes(searchQuery.toLowerCase()) && shouldFilterCycle(c, filters ?? {}) ); cycles = orderCycles(cycles, sortByManual); const cycleIds = cycles.map((c) => c.id); return cycleIds; }); /** * @description returns filtered cycle ids based on display filters and filters * @param {TCycleDisplayFilters} displayFilters * @param {TCycleFilters} filters * @returns {string[] | null} */ getFilteredCompletedCycleIds = computedFn((projectId: string) => { const filters = this.rootStore.cycleFilter.getFiltersByProjectId(projectId); const searchQuery = this.rootStore.cycleFilter.searchQuery; if (!this.fetchedMap[projectId]) return null; let cycles = Object.values(this.cycleMap ?? {}).filter( (c) => c.project_id === projectId && !c.archived_at && c.status?.toLowerCase() === "completed" && c.name.toLowerCase().includes(searchQuery.toLowerCase()) && shouldFilterCycle(c, filters ?? {}) ); cycles = sortBy(cycles, [(c) => !c.start_date]); const cycleIds = cycles.map((c) => c.id); return cycleIds; }); /** * @description returns filtered archived cycle ids based on display filters and filters * @param {string} projectId * @returns {string[] | null} */ getFilteredArchivedCycleIds = computedFn((projectId: string) => { const filters = this.rootStore.cycleFilter.getArchivedFiltersByProjectId(projectId); const searchQuery = this.rootStore.cycleFilter.archivedCyclesSearchQuery; if (!this.fetchedMap[projectId]) return null; let cycles = Object.values(this.cycleMap ?? {}).filter( (c) => c.project_id === projectId && !!c.archived_at && c.name.toLowerCase().includes(searchQuery.toLowerCase()) && shouldFilterCycle(c, filters ?? {}) ); cycles = sortBy(cycles, [(c) => !c.start_date]); const cycleIds = cycles.map((c) => c.id); return cycleIds; }); /** * @description returns cycle details by cycle id * @param cycleId * @returns */ getCycleById = computedFn((cycleId: string): ICycle | null => this.cycleMap?.[cycleId] ?? null); /** * @description returns cycle name by cycle id * @param cycleId * @returns */ getCycleNameById = computedFn((cycleId: string): string => this.cycleMap?.[cycleId]?.name); /** * @description returns active cycle details by cycle id * @param cycleId * @returns */ getActiveCycleById = computedFn((cycleId: string): ICycle | null => this.activeCycleIdMap?.[cycleId] && this.cycleMap?.[cycleId] ? this.cycleMap?.[cycleId] : null ); /** * @description returns list of cycle ids of the project id passed as argument * @param projectId */ getProjectCycleIds = computedFn((projectId: string): string[] | null => { if (!this.fetchedMap[projectId]) return null; let cycles = Object.values(this.cycleMap ?? {}).filter((c) => c.project_id === projectId && !c?.archived_at); cycles = sortBy(cycles, [(c) => c.sort_order]); const cycleIds = cycles.map((c) => c.id); return cycleIds || null; }); /** * @description validates cycle dates * @param workspaceSlug * @param projectId * @param payload * @returns */ validateDate = async (workspaceSlug: string, projectId: string, payload: CycleDateCheckData) => await this.cycleService.cycleDateCheck(workspaceSlug, projectId, payload); /** * @description gets the plot type for the module store * @param {TCyclePlotType} plotType */ getPlotTypeByCycleId = (cycleId: string) => { const { projectId } = this.rootStore.router; return projectId && this.rootStore.projectEstimate.areEstimateEnabledByProjectId(projectId) ? this.plotType[cycleId] || "burndown" : "burndown"; }; /** * @description updates the plot type for the module store * @param {TCyclePlotType} plotType */ setPlotType = (cycleId: string, plotType: TCyclePlotType) => { set(this.plotType, [cycleId], plotType); }; /** * @description fetch all cycles * @param workspaceSlug * @returns ICycle[] */ fetchWorkspaceCycles = async (workspaceSlug: string) => await this.cycleService.getWorkspaceCycles(workspaceSlug).then((response) => { runInAction(() => { response.forEach((cycle) => { set(this.cycleMap, [cycle.id], { ...this.cycleMap[cycle.id], ...cycle }); set(this.fetchedMap, cycle.project_id, true); }); }); return response; }); /** * @description fetches all cycles for a project * @param workspaceSlug * @param projectId * @returns */ fetchAllCycles = async (workspaceSlug: string, projectId: string) => { try { this.loader = true; await this.cycleService.getCyclesWithParams(workspaceSlug, projectId).then((response) => { runInAction(() => { response.forEach((cycle) => { set(this.cycleMap, [cycle.id], cycle); }); set(this.fetchedMap, projectId, true); this.loader = false; }); return response; }); } catch (error) { this.loader = false; return undefined; } }; /** * @description fetches archived cycles for a project * @param workspaceSlug * @param projectId * @returns */ fetchArchivedCycles = async (workspaceSlug: string, projectId: string) => { this.loader = true; return await this.cycleArchiveService .getArchivedCycles(workspaceSlug, projectId) .then((response) => { runInAction(() => { response.forEach((cycle) => { set(this.cycleMap, [cycle.id], cycle); }); this.loader = false; }); return response; }) .catch(() => { this.loader = false; return undefined; }); }; /** * @description fetches active cycle for a project * @param workspaceSlug * @param projectId * @returns */ fetchActiveCycle = async (workspaceSlug: string, projectId: string) => await this.cycleService.getCyclesWithParams(workspaceSlug, projectId, "current").then((response) => { runInAction(() => { response.forEach((cycle) => { set(this.activeCycleIdMap, [cycle.id], true); set(this.cycleMap, [cycle.id], cycle); }); }); return response; }); /** * @description fetches cycle details * @param workspaceSlug * @param projectId * @param cycleId * @returns */ fetchArchivedCycleDetails = async (workspaceSlug: string, projectId: string, cycleId: string) => await this.cycleArchiveService.getArchivedCycleDetails(workspaceSlug, projectId, cycleId).then((response) => { runInAction(() => { set(this.cycleMap, [response.id], { ...this.cycleMap?.[response.id], ...response }); }); return response; }); /** * @description fetches cycle details * @param workspaceSlug * @param projectId * @param cycleId * @returns */ fetchCycleDetails = async (workspaceSlug: string, projectId: string, cycleId: string) => await this.cycleService.getCycleDetails(workspaceSlug, projectId, cycleId).then((response) => { runInAction(() => { set(this.cycleMap, [response.id], { ...this.cycleMap?.[response.id], ...response }); }); return response; }); /** * @description creates a new cycle * @param workspaceSlug * @param projectId * @param data * @returns */ createCycle = async (workspaceSlug: string, projectId: string, data: Partial) => await this.cycleService.createCycle(workspaceSlug, projectId, data).then((response) => { runInAction(() => { set(this.cycleMap, [response.id], response); }); return response; }); /** * @description updates cycle details * @param workspaceSlug * @param projectId * @param cycleId * @param data * @returns */ updateCycleDetails = async (workspaceSlug: string, projectId: string, cycleId: string, data: Partial) => { try { runInAction(() => { set(this.cycleMap, [cycleId], { ...this.cycleMap?.[cycleId], ...data }); }); const response = await this.cycleService.patchCycle(workspaceSlug, projectId, cycleId, data); this.fetchCycleDetails(workspaceSlug, projectId, cycleId); return response; } catch (error) { console.log("Failed to patch cycle from cycle store"); this.fetchAllCycles(workspaceSlug, projectId); this.fetchActiveCycle(workspaceSlug, projectId); throw error; } }; /** * @description deletes a cycle * @param workspaceSlug * @param projectId * @param cycleId */ deleteCycle = async (workspaceSlug: string, projectId: string, cycleId: string) => await this.cycleService.deleteCycle(workspaceSlug, projectId, cycleId).then(() => { runInAction(() => { delete this.cycleMap[cycleId]; delete this.activeCycleIdMap[cycleId]; }); }); /** * @description adds a cycle to favorites * @param workspaceSlug * @param projectId * @param cycleId * @returns */ addCycleToFavorites = async (workspaceSlug: string, projectId: string, cycleId: string) => { const currentCycle = this.getCycleById(cycleId); try { runInAction(() => { if (currentCycle) set(this.cycleMap, [cycleId, "is_favorite"], true); }); // updating through api. const response = await this.cycleService.addCycleToFavorites(workspaceSlug, projectId, { cycle: cycleId }); return response; } catch (error) { runInAction(() => { if (currentCycle) set(this.cycleMap, [cycleId, "is_favorite"], false); }); throw error; } }; /** * @description removes a cycle from favorites * @param workspaceSlug * @param projectId * @param cycleId * @returns */ removeCycleFromFavorites = async (workspaceSlug: string, projectId: string, cycleId: string) => { const currentCycle = this.getCycleById(cycleId); try { runInAction(() => { if (currentCycle) set(this.cycleMap, [cycleId, "is_favorite"], false); }); const response = await this.cycleService.removeCycleFromFavorites(workspaceSlug, projectId, cycleId); return response; } catch (error) { runInAction(() => { if (currentCycle) set(this.cycleMap, [cycleId, "is_favorite"], true); }); throw error; } }; /** * @description archives a cycle * @param workspaceSlug * @param projectId * @param cycleId * @returns */ archiveCycle = async (workspaceSlug: string, projectId: string, cycleId: string) => { const cycleDetails = this.getCycleById(cycleId); if (cycleDetails?.archived_at) return; await this.cycleArchiveService .archiveCycle(workspaceSlug, projectId, cycleId) .then((response) => { runInAction(() => { set(this.cycleMap, [cycleId, "archived_at"], response.archived_at); }); }) .catch((error) => { console.error("Failed to archive cycle in cycle store", error); }); }; /** * @description restores a cycle * @param workspaceSlug * @param projectId * @param cycleId * @returns */ restoreCycle = async (workspaceSlug: string, projectId: string, cycleId: string) => { const cycleDetails = this.getCycleById(cycleId); if (!cycleDetails?.archived_at) return; await this.cycleArchiveService .restoreCycle(workspaceSlug, projectId, cycleId) .then(() => { runInAction(() => { set(this.cycleMap, [cycleId, "archived_at"], null); }); }) .catch((error) => { console.error("Failed to restore cycle in cycle store", error); }); }; }