diff --git a/web/components/issues/issue-layouts/calendar/base-calendar-root.tsx b/web/components/issues/issue-layouts/calendar/base-calendar-root.tsx index 1d70e2289..b080bc838 100644 --- a/web/components/issues/issue-layouts/calendar/base-calendar-root.tsx +++ b/web/components/issues/issue-layouts/calendar/base-calendar-root.tsx @@ -120,8 +120,8 @@ export const BaseCalendarRoot = observer((props: IBaseCalendarRoot) => { workspaceSlug={workspaceSlug.toString()} projectId={peekProjectId.toString()} issueId={peekIssueId.toString()} - handleIssue={async (issueToUpdate) => - await handleIssues(issueToUpdate.target_date ?? "", issueToUpdate as IIssue, EIssueActions.UPDATE) + handleIssue={async (issueToUpdate, action: EIssueActions) => + await handleIssues(issueToUpdate.target_date ?? "", issueToUpdate as IIssue, action) } /> )} diff --git a/web/components/issues/issue-layouts/gantt/base-gantt-root.tsx b/web/components/issues/issue-layouts/gantt/base-gantt-root.tsx index 62df8fc79..15b851661 100644 --- a/web/components/issues/issue-layouts/gantt/base-gantt-root.tsx +++ b/web/components/issues/issue-layouts/gantt/base-gantt-root.tsx @@ -1,4 +1,4 @@ -import React from "react"; +import React, { useCallback } from "react"; import { useRouter } from "next/router"; import { observer } from "mobx-react-lite"; // mobx store @@ -25,6 +25,8 @@ import { IViewIssuesStore, } from "store/issues"; import { TUnGroupedIssues } from "store/issues/types"; +import { EIssueActions } from "../types"; +// constants import { EUserWorkspaceRoles } from "constants/workspace"; interface IBaseGanttRoot { @@ -35,10 +37,15 @@ interface IBaseGanttRoot { | IViewIssuesFilterStore; issueStore: IProjectIssuesStore | IModuleIssuesStore | ICycleIssuesStore | IViewIssuesStore; viewId?: string; + issueActions: { + [EIssueActions.DELETE]: (issue: IIssue) => Promise; + [EIssueActions.UPDATE]?: (issue: IIssue) => Promise; + [EIssueActions.REMOVE]?: (issue: IIssue) => Promise; + }; } export const BaseGanttRoot: React.FC = observer((props: IBaseGanttRoot) => { - const { issueFiltersStore, issueStore, viewId } = props; + const { issueFiltersStore, issueStore, viewId, issueActions } = props; const router = useRouter(); const { workspaceSlug, peekIssueId, peekProjectId } = router.query; @@ -64,11 +71,14 @@ export const BaseGanttRoot: React.FC = observer((props: IBaseGan await issueStore.updateIssue(workspaceSlug.toString(), issue.project, issue.id, payload, viewId); }; - const updateIssue = async (projectId: string, issueId: string, payload: Partial) => { - if (!workspaceSlug) return; - - await issueStore.updateIssue(workspaceSlug.toString(), projectId, issueId, payload, viewId); - }; + const handleIssues = useCallback( + async (issue: IIssue, action: EIssueActions) => { + if (issueActions[action]) { + await issueActions[action]!(issue); + } + }, + [issueActions] + ); const isAllowed = !!currentProjectRole && currentProjectRole >= EUserWorkspaceRoles.MEMBER; @@ -102,8 +112,8 @@ export const BaseGanttRoot: React.FC = observer((props: IBaseGan workspaceSlug={workspaceSlug.toString()} projectId={peekProjectId.toString()} issueId={peekIssueId.toString()} - handleIssue={async (issueToUpdate) => { - await updateIssue(peekProjectId.toString(), peekIssueId.toString(), issueToUpdate); + handleIssue={async (issueToUpdate, action) => { + await handleIssues(issueToUpdate as IIssue, action); }} /> )} diff --git a/web/components/issues/issue-layouts/gantt/cycle-root.tsx b/web/components/issues/issue-layouts/gantt/cycle-root.tsx index 536650694..e09092fec 100644 --- a/web/components/issues/issue-layouts/gantt/cycle-root.tsx +++ b/web/components/issues/issue-layouts/gantt/cycle-root.tsx @@ -4,15 +4,43 @@ import { useMobxStore } from "lib/mobx/store-provider"; // components import { BaseGanttRoot } from "./base-gantt-root"; import { useRouter } from "next/router"; +// types +import { EIssueActions } from "../types"; +import { IIssue } from "types"; export const CycleGanttLayout: React.FC = observer(() => { const router = useRouter(); - const { cycleId } = router.query; + const { cycleId, workspaceSlug } = router.query; const { cycleIssues: cycleIssueStore, cycleIssuesFilter: cycleIssueFilterStore } = useMobxStore(); + const issueActions = { + [EIssueActions.UPDATE]: async (issue: IIssue) => { + if (!workspaceSlug || !cycleId) return; + + await cycleIssueStore.updateIssue(workspaceSlug.toString(), issue.project, issue.id, issue, cycleId.toString()); + }, + [EIssueActions.DELETE]: async (issue: IIssue) => { + if (!workspaceSlug || !cycleId) return; + + await cycleIssueStore.removeIssue(workspaceSlug.toString(), issue.project, issue.id, cycleId.toString()); + }, + [EIssueActions.REMOVE]: async (issue: IIssue) => { + if (!workspaceSlug || !cycleId || !issue.bridge_id) return; + + await cycleIssueStore.removeIssueFromCycle( + workspaceSlug.toString(), + issue.project, + cycleId.toString(), + issue.id, + issue.bridge_id + ); + }, + }; + return ( { const router = useRouter(); - const { moduleId } = router.query; + const { moduleId, workspaceSlug } = router.query; const { moduleIssues: moduleIssueStore, moduleIssuesFilter: moduleIssueFilterStore } = useMobxStore(); + const issueActions = { + [EIssueActions.UPDATE]: async (issue: IIssue) => { + if (!workspaceSlug || !moduleId) return; + + await moduleIssueStore.updateIssue(workspaceSlug.toString(), issue.project, issue.id, issue, moduleId.toString()); + }, + [EIssueActions.DELETE]: async (issue: IIssue) => { + if (!workspaceSlug || !moduleId) return; + + await moduleIssueStore.removeIssue(workspaceSlug.toString(), issue.project, issue.id, moduleId.toString()); + }, + [EIssueActions.REMOVE]: async (issue: IIssue) => { + if (!workspaceSlug || !moduleId || !issue.bridge_id) return; + + await moduleIssueStore.removeIssueFromModule( + workspaceSlug.toString(), + issue.project, + moduleId.toString(), + issue.id, + issue.bridge_id + ); + }, + }; + return ( { const { projectIssues: projectIssuesStore, projectIssuesFilter: projectIssueFiltersStore } = useMobxStore(); + const router = useRouter(); + const { workspaceSlug } = router.query; - return ; + const issueActions = { + [EIssueActions.UPDATE]: async (issue: IIssue) => { + if (!workspaceSlug) return; + + await projectIssuesStore.updateIssue(workspaceSlug.toString(), issue.project, issue.id, issue); + }, + [EIssueActions.DELETE]: async (issue: IIssue) => { + if (!workspaceSlug) return; + + await projectIssuesStore.removeIssue(workspaceSlug.toString(), issue.project, issue.id); + }, + }; + return ( + + ); }); diff --git a/web/components/issues/issue-layouts/gantt/project-view-root.tsx b/web/components/issues/issue-layouts/gantt/project-view-root.tsx index 3155aae6f..c39c32fe8 100644 --- a/web/components/issues/issue-layouts/gantt/project-view-root.tsx +++ b/web/components/issues/issue-layouts/gantt/project-view-root.tsx @@ -1,11 +1,35 @@ import { observer } from "mobx-react-lite"; +import { useRouter } from "next/router"; // mobx store import { useMobxStore } from "lib/mobx/store-provider"; // components import { BaseGanttRoot } from "./base-gantt-root"; +// types +import { EIssueActions } from "../types"; +import { IIssue } from "types"; export const ProjectViewGanttLayout: React.FC = observer(() => { const { viewIssues: projectIssueViewStore, viewIssuesFilter: projectIssueViewFiltersStore } = useMobxStore(); + const router = useRouter(); + const { workspaceSlug } = router.query; - return ; + const issueActions = { + [EIssueActions.UPDATE]: async (issue: IIssue) => { + if (!workspaceSlug) return; + + await projectIssueViewStore.updateIssue(workspaceSlug.toString(), issue.project, issue.id, issue); + }, + [EIssueActions.DELETE]: async (issue: IIssue) => { + if (!workspaceSlug) return; + + await projectIssueViewStore.removeIssue(workspaceSlug.toString(), issue.project, issue.id); + }, + }; + return ( + + ); }); diff --git a/web/components/issues/issue-layouts/kanban/base-kanban-root.tsx b/web/components/issues/issue-layouts/kanban/base-kanban-root.tsx index 057ba0b87..b536b1fa8 100644 --- a/web/components/issues/issue-layouts/kanban/base-kanban-root.tsx +++ b/web/components/issues/issue-layouts/kanban/base-kanban-root.tsx @@ -346,8 +346,8 @@ export const BaseKanBanRoot: React.FC = observer((props: IBas workspaceSlug={workspaceSlug.toString()} projectId={peekProjectId.toString()} issueId={peekIssueId.toString()} - handleIssue={async (issueToUpdate) => - await handleIssues(sub_group_by, group_by, issueToUpdate as IIssue, EIssueActions.UPDATE) + handleIssue={async (issueToUpdate, action: EIssueActions) => + await handleIssues(sub_group_by, group_by, issueToUpdate as IIssue, action) } /> )} diff --git a/web/components/issues/issue-layouts/list/base-list-root.tsx b/web/components/issues/issue-layouts/list/base-list-root.tsx index 92956509e..55b2fce55 100644 --- a/web/components/issues/issue-layouts/list/base-list-root.tsx +++ b/web/components/issues/issue-layouts/list/base-list-root.tsx @@ -168,7 +168,9 @@ export const BaseListRoot = observer((props: IBaseListRoot) => { workspaceSlug={workspaceSlug.toString()} projectId={peekProjectId.toString()} issueId={peekIssueId.toString()} - handleIssue={async (issueToUpdate) => await handleIssues(issueToUpdate as IIssue, EIssueActions.UPDATE)} + handleIssue={async (issueToUpdate, action: EIssueActions) => + await handleIssues(issueToUpdate as IIssue, action) + } /> )} diff --git a/web/components/issues/issue-layouts/spreadsheet/spreadsheet-view.tsx b/web/components/issues/issue-layouts/spreadsheet/spreadsheet-view.tsx index 370bbfa0f..4b7ffe6da 100644 --- a/web/components/issues/issue-layouts/spreadsheet/spreadsheet-view.tsx +++ b/web/components/issues/issue-layouts/spreadsheet/spreadsheet-view.tsx @@ -194,7 +194,7 @@ export const SpreadsheetView: React.FC = observer((props) => { workspaceSlug={workspaceSlug.toString()} projectId={peekProjectId.toString()} issueId={peekIssueId.toString()} - handleIssue={async (issueToUpdate: any) => await handleIssues(issueToUpdate, EIssueActions.UPDATE)} + handleIssue={async (issueToUpdate: any, action: EIssueActions) => await handleIssues(issueToUpdate, action)} /> )} diff --git a/web/components/issues/peek-overview/root.tsx b/web/components/issues/peek-overview/root.tsx index 783958f5d..c00074af9 100644 --- a/web/components/issues/peek-overview/root.tsx +++ b/web/components/issues/peek-overview/root.tsx @@ -11,6 +11,7 @@ import { IssueView } from "components/issues"; import { copyUrlToClipboard } from "helpers/string.helper"; // types import { IIssue, IIssueLink } from "types"; +import { EIssueActions } from "../issue-layouts/types"; // constants import { EUserWorkspaceRoles } from "constants/workspace"; @@ -18,7 +19,7 @@ interface IIssuePeekOverview { workspaceSlug: string; projectId: string; issueId: string; - handleIssue: (issue: Partial) => void; + handleIssue: (issue: Partial, action: EIssueActions) => Promise; isArchived?: boolean; children?: ReactNode; } @@ -31,7 +32,6 @@ export const IssuePeekOverview: FC = observer((props) => { const { user: { currentProjectRole }, - issue: { removeIssueFromStructure }, issueDetail: { createIssueComment, updateIssueComment, @@ -98,7 +98,7 @@ export const IssuePeekOverview: FC = observer((props) => { const issueUpdate = async (_data: Partial) => { if (handleIssue) { - await handleIssue(_data); + await handleIssue(_data, EIssueActions.UPDATE); fetchIssueActivity(workspaceSlug, projectId, issueId); } }; @@ -133,7 +133,7 @@ export const IssuePeekOverview: FC = observer((props) => { const handleDeleteIssue = async () => { if (isArchived) await deleteArchivedIssue(workspaceSlug, projectId, issue!); - else removeIssueFromStructure(workspaceSlug, projectId, issue!); + else await handleIssue(issue!, EIssueActions.DELETE); const { query } = router; if (query.peekIssueId) { setPeekId(null); diff --git a/web/components/issues/sub-issues/issue.tsx b/web/components/issues/sub-issues/issue.tsx index e1e983c80..13b3ff36a 100644 --- a/web/components/issues/sub-issues/issue.tsx +++ b/web/components/issues/sub-issues/issue.tsx @@ -10,6 +10,7 @@ import { CustomMenu, Tooltip } from "@plane/ui"; // types import { IUser, IIssue } from "types"; import { ISubIssuesRootLoaders, ISubIssuesRootLoadersHandler } from "./root"; +import { EIssueActions } from "../issue-layouts/types"; export interface ISubIssues { workspaceSlug: string; @@ -29,6 +30,7 @@ export interface ISubIssues { issue?: IIssue | null ) => void; handleUpdateIssue: (issue: IIssue, data: Partial) => void; + handleDeleteIssue: (issue: IIssue) => Promise; } export const SubIssues: React.FC = ({ @@ -45,6 +47,7 @@ export const SubIssues: React.FC = ({ copyText, handleIssueCrudOperation, handleUpdateIssue, + handleDeleteIssue, }) => { const router = useRouter(); const { peekProjectId, peekIssueId } = router.query; @@ -69,7 +72,13 @@ export const SubIssues: React.FC = ({ workspaceSlug={workspaceSlug} projectId={peekProjectId.toString()} issueId={peekIssueId.toString()} - handleIssue={async (issueToUpdate) => await handleUpdateIssue(issue, { ...issue, ...issueToUpdate })} + handleIssue={async (issueToUpdate, action) => { + if (action === EIssueActions.UPDATE) { + await handleUpdateIssue(issue, { ...issue, ...issueToUpdate }); + } else if (action === EIssueActions.DELETE) { + await handleDeleteIssue(issue); + } + }} /> )}
@@ -180,6 +189,7 @@ export const SubIssues: React.FC = ({ {issuesLoader.visibility.includes(issue?.id) && issue?.sub_issues_count > 0 && ( void; handleUpdateIssue: (issue: IIssue, data: Partial) => void; + handleDeleteIssue: (issue: IIssue) => Promise } const issueService = new IssueService(); @@ -44,6 +45,7 @@ export const SubIssuesRootList: React.FC = ({ copyText, handleIssueCrudOperation, handleUpdateIssue, + handleDeleteIssue }) => { const { data: issues, isLoading } = useSWR( workspaceSlug && projectId && parentIssue && parentIssue?.id ? SUB_ISSUES(parentIssue?.id) : null, @@ -70,6 +72,7 @@ export const SubIssuesRootList: React.FC = ({ issues.sub_issues.length > 0 && issues.sub_issues.map((issue: IIssue) => ( = observer((props) => { [updateIssueStructure, projectId, updateIssue, user, workspaceSlug] ); + const handleDeleteIssue = useCallback( + async (issue: IIssue) => { + if (!workspaceSlug || !projectId || !user) return; + + await removeIssue(workspaceSlug.toString(), projectId.toString(), issue.id); + await mutate(SUB_ISSUES(parentIssue?.id)); + }, + [removeIssue, projectId, user, workspaceSlug, parentIssue?.id] + ); + const isEditable = !!currentProjectRole && currentProjectRole >= EUserWorkspaceRoles.MEMBER; const mutateSubIssues = (parentIssueId: string | null) => { @@ -236,6 +246,7 @@ export const SubIssuesRoot: React.FC = observer((props) => { {issuesLoader.visibility.includes(parentIssue?.id) && workspaceSlug && projectId && (
{ description_html: newDescription, }) .then(() => { - mutatePageDetails((prevData) => ({ ...prevData, description_html: newDescription }) as IPage, false); + mutatePageDetails((prevData) => ({ ...prevData, description_html: newDescription } as IPage), false); }); }; @@ -162,15 +162,12 @@ const PageDetailsPage: NextPageWithLayout = observer(() => { }, [pageDetails?.description_html]); // TODO: Verify the exhaustive-deps warning function createObjectFromArray(keys: string[], options: any): any { - return keys.reduce( - (obj, key) => { - if (options[key] !== undefined) { - obj[key] = options[key]; - } - return obj; - }, - {} as { [key: string]: any } - ); + return keys.reduce((obj, key) => { + if (options[key] !== undefined) { + obj[key] = options[key]; + } + return obj; + }, {} as { [key: string]: any }); } const mutatePageDetailsHelper = ( @@ -499,7 +496,7 @@ const PageDetailsPage: NextPageWithLayout = observer(() => { projectId={projectId as string} issueId={peekIssueId ? (peekIssueId as string) : ""} isArchived={false} - handleIssue={(issueToUpdate) => { + handleIssue={async (issueToUpdate, action) => { if (peekIssueId && typeof peekIssueId === "string") { handleUpdateIssue(peekIssueId, issueToUpdate); }