import { makeObservable } from "mobx"; import { computedFn } from "mobx-utils"; // types import { TIssue } from "@plane/types"; // services import { IssueArchiveService, IssueDraftService, IssueService } from "@/services/issue"; // types import { IIssueDetail } from "./root.store"; export interface IIssueStoreActions { // actions fetchIssue: ( workspaceSlug: string, projectId: string, issueId: string, issueType?: "DEFAULT" | "DRAFT" | "ARCHIVED" ) => Promise; updateIssue: (workspaceSlug: string, projectId: string, issueId: string, data: Partial) => Promise; removeIssue: (workspaceSlug: string, projectId: string, issueId: string) => Promise; archiveIssue: (workspaceSlug: string, projectId: string, issueId: string) => Promise; addCycleToIssue: (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) => Promise; addIssueToCycle: (workspaceSlug: string, projectId: string, cycleId: string, issueIds: string[]) => Promise; removeIssueFromCycle: (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) => Promise; addModulesToIssue: (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => Promise; removeModulesFromIssue: ( workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[] ) => Promise; removeIssueFromModule: (workspaceSlug: string, projectId: string, moduleId: string, issueId: string) => Promise; } export interface IIssueStore extends IIssueStoreActions { // helper methods getIssueById: (issueId: string) => TIssue | undefined; } export class IssueStore implements IIssueStore { // root store rootIssueDetailStore: IIssueDetail; // services issueService; issueArchiveService; issueDraftService; constructor(rootStore: IIssueDetail) { makeObservable(this, {}); // root store this.rootIssueDetailStore = rootStore; // services this.issueService = new IssueService(); this.issueArchiveService = new IssueArchiveService(); this.issueDraftService = new IssueDraftService(); } // helper methods getIssueById = computedFn((issueId: string) => { if (!issueId) return undefined; return this.rootIssueDetailStore.rootIssueStore.issues.getIssueById(issueId) ?? undefined; }); // actions fetchIssue = async (workspaceSlug: string, projectId: string, issueId: string, issueType = "DEFAULT") => { try { const query = { expand: "issue_reactions,issue_attachment,issue_link,parent", }; let issue: TIssue; if (issueType === "ARCHIVED") issue = await this.issueArchiveService.retrieveArchivedIssue(workspaceSlug, projectId, issueId, query); else if (issueType === "DRAFT") issue = await this.issueDraftService.getDraftIssueById(workspaceSlug, projectId, issueId, query); else issue = await this.issueService.retrieve(workspaceSlug, projectId, issueId, query); if (!issue) throw new Error("Issue not found"); const issuePayload: TIssue = { id: issue?.id, sequence_id: issue?.sequence_id, name: issue?.name, description_html: issue?.description_html, sort_order: issue?.sort_order, state_id: issue?.state_id, priority: issue?.priority, label_ids: issue?.label_ids, assignee_ids: issue?.assignee_ids, estimate_point: issue?.estimate_point, sub_issues_count: issue?.sub_issues_count, attachment_count: issue?.attachment_count, link_count: issue?.link_count, project_id: issue?.project_id, parent_id: issue?.parent_id, cycle_id: issue?.cycle_id, module_ids: issue?.module_ids, created_at: issue?.created_at, updated_at: issue?.updated_at, start_date: issue?.start_date, target_date: issue?.target_date, completed_at: issue?.completed_at, archived_at: issue?.archived_at, created_by: issue?.created_by, updated_by: issue?.updated_by, is_draft: issue?.is_draft, is_subscribed: issue?.is_subscribed, }; this.rootIssueDetailStore.rootIssueStore.issues.addIssue([issuePayload], true); // store handlers from issue detail // parent if (issue && issue?.parent && issue?.parent?.id) { const parentIssue = await this.issueService.retrieve(workspaceSlug, projectId, issue?.parent?.id); this.rootIssueDetailStore.rootIssueStore.issues.addIssue([parentIssue]); } // assignees // labels // state // issue reactions if (issue.issue_reactions) this.rootIssueDetailStore.addReactions(issueId, issue.issue_reactions); // fetch issue links if (issue.issue_link) this.rootIssueDetailStore.addLinks(issueId, issue.issue_link); // fetch issue attachments if (issue.issue_attachment) this.rootIssueDetailStore.addAttachments(issueId, issue.issue_attachment); this.rootIssueDetailStore.addSubscription(issueId, issue.is_subscribed); // fetch issue activity this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueId); // fetch issue comments this.rootIssueDetailStore.comment.fetchComments(workspaceSlug, projectId, issueId); // fetch sub issues this.rootIssueDetailStore.subIssues.fetchSubIssues(workspaceSlug, projectId, issueId); // fetch issue relations this.rootIssueDetailStore.relation.fetchRelations(workspaceSlug, projectId, issueId); // fetching states // TODO: check if this function is required this.rootIssueDetailStore.rootIssueStore.state.fetchProjectStates(workspaceSlug, projectId); return issue; } catch (error) { throw error; } }; updateIssue = async (workspaceSlug: string, projectId: string, issueId: string, data: Partial) => { await this.rootIssueDetailStore.rootIssueStore.projectIssues.updateIssue(workspaceSlug, projectId, issueId, data); await this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueId); }; removeIssue = async (workspaceSlug: string, projectId: string, issueId: string) => this.rootIssueDetailStore.rootIssueStore.projectIssues.removeIssue(workspaceSlug, projectId, issueId); archiveIssue = async (workspaceSlug: string, projectId: string, issueId: string) => this.rootIssueDetailStore.rootIssueStore.projectIssues.archiveIssue(workspaceSlug, projectId, issueId); addCycleToIssue = async (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) => { await this.rootIssueDetailStore.rootIssueStore.cycleIssues.addCycleToIssue( workspaceSlug, projectId, cycleId, issueId ); await this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueId); }; addIssueToCycle = async (workspaceSlug: string, projectId: string, cycleId: string, issueIds: string[]) => { await this.rootIssueDetailStore.rootIssueStore.cycleIssues.addIssueToCycle( workspaceSlug, projectId, cycleId, issueIds, false ); if (issueIds && issueIds.length > 0) await this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueIds[0]); }; removeIssueFromCycle = async (workspaceSlug: string, projectId: string, cycleId: string, issueId: string) => { const cycle = await this.rootIssueDetailStore.rootIssueStore.cycleIssues.removeIssueFromCycle( workspaceSlug, projectId, cycleId, issueId ); await this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueId); return cycle; }; addModulesToIssue = async (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => { const currentModule = await this.rootIssueDetailStore.rootIssueStore.moduleIssues.changeModulesInIssue( workspaceSlug, projectId, issueId, moduleIds, [] ); if (moduleIds && moduleIds.length > 0) await this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueId); return currentModule; }; removeModulesFromIssue = async (workspaceSlug: string, projectId: string, issueId: string, moduleIds: string[]) => { const currentModule = await this.rootIssueDetailStore.rootIssueStore.moduleIssues.changeModulesInIssue( workspaceSlug, projectId, issueId, [], moduleIds ); await this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueId); return currentModule; }; removeIssueFromModule = async (workspaceSlug: string, projectId: string, moduleId: string, issueId: string) => { const currentModule = await this.rootIssueDetailStore.rootIssueStore.moduleIssues.removeIssuesFromModule( workspaceSlug, projectId, moduleId, [issueId] ); await this.rootIssueDetailStore.activity.fetchActivities(workspaceSlug, projectId, issueId); return currentModule; }; }