import { makeObservable, observable, computed, action, runInAction } from "mobx"; import groupBy from "lodash/groupBy"; import keyBy from "lodash/keyBy"; import set from "lodash/set"; // store import { RootStore } from "./root.store"; // types import { IState } from "types"; // services import { ProjectStateService } from "services/project"; export interface IStateStore { // observables stateMap: Record; // computed projectStates: IState[] | undefined; groupedProjectStates: Record | undefined; // computed actions getProjectStates: (projectId: string) => IState[]; // actions fetchProjectStates: (workspaceSlug: string, projectId: string) => Promise; createState: (workspaceSlug: string, projectId: string, data: Partial) => Promise; updateState: ( workspaceSlug: string, projectId: string, stateId: string, data: Partial ) => Promise; deleteState: (workspaceSlug: string, projectId: string, stateId: string) => Promise; markStateAsDefault: (workspaceSlug: string, projectId: string, stateId: string) => Promise; moveStatePosition: ( workspaceSlug: string, projectId: string, stateId: string, direction: "up" | "down", groupIndex: number ) => Promise; } export class StateStore implements IStateStore { stateMap: Record = {}; router; stateService; constructor(_rootStore: RootStore) { makeObservable(this, { // observables stateMap: observable, // computed projectStates: computed, groupedProjectStates: computed, // computed actions getProjectStates: action, // actions fetchProjectStates: action, createState: action, updateState: action, deleteState: action, markStateAsDefault: action, moveStatePosition: action, }); this.stateService = new ProjectStateService(); this.router = _rootStore.app.router; } /** * Returns the stateMap belongs to a specific project */ get projectStates() { if (!this.router.query?.projectId) return; return Object.values(this.stateMap).filter((state) => state.project === this.router.query.projectId); } /** * Returns the stateMap belongs to a specific project grouped by group */ get groupedProjectStates() { if (!this.router.query?.projectId) return; return groupBy(this.projectStates, "group") as Record; } /** * Returns the stateMap belongs to a project by projectId * @param projectId * @returns IState[] */ getProjectStates(projectId: string) { return Object.values(this.stateMap).filter((state) => state.project === projectId); } /** * fetches the stateMap of a project * @param workspaceSlug * @param projectId * @returns */ fetchProjectStates = async (workspaceSlug: string, projectId: string) => { const stateMap = await this.stateService.getStates(workspaceSlug, projectId); runInAction(() => { //todo add iteratively without modifying original reference this.stateMap = { ...this.stateMap, ...keyBy(stateMap, "id"), }; }); return stateMap; }; /** * creates a new state in a project and adds it to the store * @param workspaceSlug * @param projectId * @param data * @returns */ createState = async (workspaceSlug: string, projectId: string, data: Partial) => { const response = await this.stateService.createState(workspaceSlug, projectId, data); runInAction(() => { set(this.stateMap, [response?.id], response); }); return response; }; /** * Updates the state details in the store, in case of failure reverts back to original state * @param workspaceSlug * @param projectId * @param stateId * @param data * @returns */ updateState = async (workspaceSlug: string, projectId: string, stateId: string, data: Partial) => { const originalState = this.stateMap[stateId]; try { runInAction(() => { set(this.stateMap, [stateId], { ...this.stateMap?.[stateId], ...data }); }); const response = await this.stateService.patchState(workspaceSlug, projectId, stateId, data); return response; } catch (error) { runInAction(() => { this.stateMap = { ...this.stateMap, [stateId]: originalState, }; }); throw error; } }; /** * deletes the state from the store, incase of failure reverts back to original state * @param workspaceSlug * @param projectId * @param stateId */ deleteState = async (workspaceSlug: string, projectId: string, stateId: string) => { const originalStates = this.stateMap; try { if (!this.stateMap?.[stateId]) return; runInAction(() => { delete this.stateMap[stateId]; }); await this.stateService.deleteState(workspaceSlug, projectId, stateId); } catch (error) { runInAction(() => { this.stateMap = originalStates; }); throw error; } }; /** * marks a state as default in a project * @param workspaceSlug * @param projectId * @param stateId */ markStateAsDefault = async (workspaceSlug: string, projectId: string, stateId: string) => { const originalStates = this.stateMap; try { runInAction(() => { set(this.stateMap, [stateId, "default"], true); }); await this.stateService.markDefault(workspaceSlug, projectId, stateId); } catch (error) { runInAction(() => { this.stateMap = originalStates; }); throw error; } }; /** * updates the sort order of a state and updates the state information using API, in case of failure reverts back to original state * @param workspaceSlug * @param projectId * @param stateId * @param direction * @param groupIndex */ moveStatePosition = async ( workspaceSlug: string, projectId: string, stateId: string, direction: "up" | "down", groupIndex: number ) => { const SEQUENCE_GAP = 15000; const originalStates = this.stateMap; try { let newSequence = SEQUENCE_GAP; const stateMap = this.projectStates || []; const selectedState = stateMap?.find((state) => state.id === stateId); const groupStates = stateMap?.filter((state) => state.group === selectedState?.group); const groupLength = groupStates.length; if (direction === "up") { if (groupIndex === 1) newSequence = groupStates[0].sequence - SEQUENCE_GAP; else newSequence = (groupStates[groupIndex - 2].sequence + groupStates[groupIndex - 1].sequence) / 2; } else { if (groupIndex === groupLength - 2) newSequence = groupStates[groupLength - 1].sequence + SEQUENCE_GAP; else newSequence = (groupStates[groupIndex + 2].sequence + groupStates[groupIndex + 1].sequence) / 2; } runInAction(() => { set(this.stateMap, [stateId, "sequence"], newSequence); }); // updating using api await this.stateService.patchState(workspaceSlug, projectId, stateId, { sequence: newSequence }); } catch (err) { // reverting back to old state group if api fails runInAction(() => { this.stateMap = originalStates; }); } }; }