From 231fd52992085dfd4b7c2f123b82da9d37177f1a Mon Sep 17 00:00:00 2001 From: Prateek Shourya Date: Thu, 21 Mar 2024 20:59:34 +0530 Subject: [PATCH] [WEB-447] feat: projects archive. (#4014) * dev: project archive response * feat: projects archive. * dev: response changes for cycle and module * chore: status message changed * chore: update clear all applied display filters logic. * style: archived project card UI update. * chore: archive/ restore taost message update. * fix: clear all applied display filter logic. * chore: project empty state update to handle archived projects. * chore: minor typo fix in cycles and modules archive. * chore: close cycle/ module overview sidebar if it's already open when clicked on overview button. * chore: optimize current workspace applied display filter logic. * chore: update all `archived_at` type from `Date` to `string`. --------- Co-authored-by: NarayanBavisetti --- apiserver/plane/app/views/cycle/base.py | 5 +- apiserver/plane/app/views/module/base.py | 7 +- apiserver/plane/app/views/project/base.py | 8 +- .../types/src/project/project_filters.d.ts | 5 + packages/types/src/project/projects.d.ts | 1 + .../cycles/archived-cycles/modal.tsx | 4 +- .../cycles/board/cycles-board-card.tsx | 20 +- .../cycles/list/cycles-list-item.tsx | 16 +- .../modules/archived-modules/modal.tsx | 4 +- web/components/modules/module-card-item.tsx | 16 +- web/components/modules/module-list-item.tsx | 16 +- .../project/applied-filters/index.ts | 1 + .../project-display-filters.tsx | 39 +++ .../project/applied-filters/root.tsx | 40 +++- web/components/project/card-list.tsx | 4 +- web/components/project/card.tsx | 223 +++++++++++------- .../project/dropdowns/filters/root.tsx | 9 + .../archive-project/archive-restore-modal.tsx | 135 +++++++++++ .../settings/archive-project/index.tsx | 2 + .../settings/archive-project/selection.tsx | 60 +++++ .../settings/delete-project-section.tsx | 18 +- web/components/project/settings/index.ts | 1 + web/constants/empty-state.ts | 4 +- web/constants/project.ts | 16 +- web/helpers/project.helper.ts | 8 +- .../projects/[projectId]/settings/index.tsx | 42 +++- web/pages/[workspaceSlug]/projects/index.tsx | 36 ++- web/services/project/index.ts | 1 + .../project/project-archive.service.ts | 31 +++ web/store/project/project.store.ts | 102 +++++++- web/store/project/project_filter.store.ts | 37 ++- 31 files changed, 749 insertions(+), 162 deletions(-) create mode 100644 web/components/project/applied-filters/project-display-filters.tsx create mode 100644 web/components/project/settings/archive-project/archive-restore-modal.tsx create mode 100644 web/components/project/settings/archive-project/index.tsx create mode 100644 web/components/project/settings/archive-project/selection.tsx create mode 100644 web/services/project/project-archive.service.ts diff --git a/apiserver/plane/app/views/cycle/base.py b/apiserver/plane/app/views/cycle/base.py index 6809efbe6..58719e373 100644 --- a/apiserver/plane/app/views/cycle/base.py +++ b/apiserver/plane/app/views/cycle/base.py @@ -878,7 +878,10 @@ class CycleArchiveUnarchiveEndpoint(BaseAPIView): ) cycle.archived_at = timezone.now() cycle.save() - return Response(status=status.HTTP_204_NO_CONTENT) + return Response( + {"archived_at": str(cycle.archived_at)}, + status=status.HTTP_200_OK, + ) def delete(self, request, slug, project_id, cycle_id): cycle = Cycle.objects.get( diff --git a/apiserver/plane/app/views/module/base.py b/apiserver/plane/app/views/module/base.py index 39dbcb751..61bb16e40 100644 --- a/apiserver/plane/app/views/module/base.py +++ b/apiserver/plane/app/views/module/base.py @@ -621,7 +621,7 @@ class ModuleArchiveUnarchiveEndpoint(BaseAPIView): "backlog_issues", "created_at", "updated_at", - "archived_at" + "archived_at", ) return Response(modules, status=status.HTTP_200_OK) @@ -631,7 +631,10 @@ class ModuleArchiveUnarchiveEndpoint(BaseAPIView): ) module.archived_at = timezone.now() module.save() - return Response(status=status.HTTP_204_NO_CONTENT) + return Response( + {"archived_at": str(module.archived_at)}, + status=status.HTTP_200_OK, + ) def delete(self, request, slug, project_id, module_id): module = Module.objects.get( diff --git a/apiserver/plane/app/views/project/base.py b/apiserver/plane/app/views/project/base.py index b2f9f56e9..1672cd47c 100644 --- a/apiserver/plane/app/views/project/base.py +++ b/apiserver/plane/app/views/project/base.py @@ -372,7 +372,7 @@ class ProjectViewSet(WebhookMixin, BaseViewSet): return Response( {"error": "Archived projects cannot be updated"}, status=status.HTTP_400_BAD_REQUEST, - ) + ) serializer = ProjectSerializer( project, @@ -433,11 +433,15 @@ class ProjectArchiveUnarchiveEndpoint(BaseAPIView): permission_classes = [ ProjectBasePermission, ] + def post(self, request, slug, project_id): project = Project.objects.get(pk=project_id, workspace__slug=slug) project.archived_at = timezone.now() project.save() - return Response(status=status.HTTP_204_NO_CONTENT) + return Response( + {"archived_at": str(project.archived_at)}, + status=status.HTTP_200_OK, + ) def delete(self, request, slug, project_id): project = Project.objects.get(pk=project_id, workspace__slug=slug) diff --git a/packages/types/src/project/project_filters.d.ts b/packages/types/src/project/project_filters.d.ts index 02ad09ee1..77da7365f 100644 --- a/packages/types/src/project/project_filters.d.ts +++ b/packages/types/src/project/project_filters.d.ts @@ -9,9 +9,14 @@ export type TProjectOrderByOptions = export type TProjectDisplayFilters = { my_projects?: boolean; + archived_projects?: boolean; order_by?: TProjectOrderByOptions; }; +export type TProjectAppliedDisplayFilterKeys = + | "my_projects" + | "archived_projects"; + export type TProjectFilters = { access?: string[] | null; lead?: string[] | null; diff --git a/packages/types/src/project/projects.d.ts b/packages/types/src/project/projects.d.ts index 86384b401..157ecb16e 100644 --- a/packages/types/src/project/projects.d.ts +++ b/packages/types/src/project/projects.d.ts @@ -23,6 +23,7 @@ export type TProjectLogoProps = { export interface IProject { archive_in: number; + archived_at: string | null; archived_issues: number; archived_sub_issues: number; close_in: number; diff --git a/web/components/cycles/archived-cycles/modal.tsx b/web/components/cycles/archived-cycles/modal.tsx index a9b421351..6e0ddef35 100644 --- a/web/components/cycles/archived-cycles/modal.tsx +++ b/web/components/cycles/archived-cycles/modal.tsx @@ -31,7 +31,7 @@ export const ArchiveCycleModal: React.FC = (props) => { handleClose(); }; - const handleArchiveIssue = async () => { + const handleArchiveCycle = async () => { setIsArchiving(true); await archiveCycle(workspaceSlug, projectId, cycleId) .then(() => { @@ -89,7 +89,7 @@ export const ArchiveCycleModal: React.FC = (props) => { - diff --git a/web/components/cycles/board/cycles-board-card.tsx b/web/components/cycles/board/cycles-board-card.tsx index 8426fe313..34d395db4 100644 --- a/web/components/cycles/board/cycles-board-card.tsx +++ b/web/components/cycles/board/cycles-board-card.tsx @@ -69,8 +69,8 @@ export const CyclesBoardCard: FC = observer((props) => { ? cycleTotalIssues === 0 ? "0 Issue" : cycleTotalIssues === cycleDetails.completed_issues - ? `${cycleTotalIssues} Issue${cycleTotalIssues > 1 ? "s" : ""}` - : `${cycleDetails.completed_issues}/${cycleTotalIssues} Issues` + ? `${cycleTotalIssues} Issue${cycleTotalIssues > 1 ? "s" : ""}` + : `${cycleDetails.completed_issues}/${cycleTotalIssues} Issues` : "0 Issue"; const handleAddToFavorites = (e: MouseEvent) => { @@ -134,10 +134,18 @@ export const CyclesBoardCard: FC = observer((props) => { e.preventDefault(); e.stopPropagation(); - router.push({ - pathname: router.pathname, - query: { ...query, peekCycle: cycleId }, - }); + if (query.peekCycle) { + delete query.peekCycle; + router.push({ + pathname: router.pathname, + query: { ...query }, + }); + } else { + router.push({ + pathname: router.pathname, + query: { ...query, peekCycle: cycleId }, + }); + } }; const daysLeft = findHowManyDaysLeft(cycleDetails.end_date) ?? 0; diff --git a/web/components/cycles/list/cycles-list-item.tsx b/web/components/cycles/list/cycles-list-item.tsx index a418f9b04..c9c3992f2 100644 --- a/web/components/cycles/list/cycles-list-item.tsx +++ b/web/components/cycles/list/cycles-list-item.tsx @@ -106,10 +106,18 @@ export const CyclesListItem: FC = observer((props) => { e.preventDefault(); e.stopPropagation(); - router.push({ - pathname: router.pathname, - query: { ...query, peekCycle: cycleId }, - }); + if (query.peekCycle) { + delete query.peekCycle; + router.push({ + pathname: router.pathname, + query: { ...query }, + }); + } else { + router.push({ + pathname: router.pathname, + query: { ...query, peekCycle: cycleId }, + }); + } }; const cycleDetails = getCycleById(cycleId); diff --git a/web/components/modules/archived-modules/modal.tsx b/web/components/modules/archived-modules/modal.tsx index f34aff260..f922e0ba9 100644 --- a/web/components/modules/archived-modules/modal.tsx +++ b/web/components/modules/archived-modules/modal.tsx @@ -31,7 +31,7 @@ export const ArchiveModuleModal: React.FC = (props) => { handleClose(); }; - const handleArchiveIssue = async () => { + const handleArchiveModule = async () => { setIsArchiving(true); await archiveModule(workspaceSlug, projectId, moduleId) .then(() => { @@ -89,7 +89,7 @@ export const ArchiveModuleModal: React.FC = (props) => { - diff --git a/web/components/modules/module-card-item.tsx b/web/components/modules/module-card-item.tsx index ff228c82e..5f040fdd6 100644 --- a/web/components/modules/module-card-item.tsx +++ b/web/components/modules/module-card-item.tsx @@ -101,10 +101,18 @@ export const ModuleCardItem: React.FC = observer((props) => { e.preventDefault(); const { query } = router; - router.push({ - pathname: router.pathname, - query: { ...query, peekModule: moduleId }, - }); + if (query.peekModule) { + delete query.peekModule; + router.push({ + pathname: router.pathname, + query: { ...query }, + }); + } else { + router.push({ + pathname: router.pathname, + query: { ...query, peekModule: moduleId }, + }); + } }; if (!moduleDetails) return null; diff --git a/web/components/modules/module-list-item.tsx b/web/components/modules/module-list-item.tsx index 3fd630f29..2354ac81c 100644 --- a/web/components/modules/module-list-item.tsx +++ b/web/components/modules/module-list-item.tsx @@ -102,10 +102,18 @@ export const ModuleListItem: React.FC = observer((props) => { e.preventDefault(); const { query } = router; - router.push({ - pathname: router.pathname, - query: { ...query, peekModule: moduleId }, - }); + if (query.peekModule) { + delete query.peekModule; + router.push({ + pathname: router.pathname, + query: { ...query }, + }); + } else { + router.push({ + pathname: router.pathname, + query: { ...query, peekModule: moduleId }, + }); + } }; if (!moduleDetails) return null; diff --git a/web/components/project/applied-filters/index.ts b/web/components/project/applied-filters/index.ts index 818aa6134..85bcda446 100644 --- a/web/components/project/applied-filters/index.ts +++ b/web/components/project/applied-filters/index.ts @@ -1,4 +1,5 @@ export * from "./access"; export * from "./date"; export * from "./members"; +export * from "./project-display-filters"; export * from "./root"; diff --git a/web/components/project/applied-filters/project-display-filters.tsx b/web/components/project/applied-filters/project-display-filters.tsx new file mode 100644 index 000000000..0c8af7097 --- /dev/null +++ b/web/components/project/applied-filters/project-display-filters.tsx @@ -0,0 +1,39 @@ +import { observer } from "mobx-react-lite"; +// icons +import { X } from "lucide-react"; +// types +import { TProjectAppliedDisplayFilterKeys } from "@plane/types"; +// constants +import { PROJECT_DISPLAY_FILTER_OPTIONS } from "@/constants/project"; + +type Props = { + handleRemove: (key: TProjectAppliedDisplayFilterKeys) => void; + values: TProjectAppliedDisplayFilterKeys[]; + editable: boolean | undefined; +}; + +export const AppliedProjectDisplayFilters: React.FC = observer((props) => { + const { handleRemove, values, editable } = props; + + return ( + <> + {values.map((key) => { + const filterLabel = PROJECT_DISPLAY_FILTER_OPTIONS.find((s) => s.key === key)?.label; + return ( +
+ {filterLabel} + {editable && ( + + )} +
+ ); + })} + + ); +}); diff --git a/web/components/project/applied-filters/root.tsx b/web/components/project/applied-filters/root.tsx index 66a2513c4..7c4381989 100644 --- a/web/components/project/applied-filters/root.tsx +++ b/web/components/project/applied-filters/root.tsx @@ -1,17 +1,24 @@ import { X } from "lucide-react"; -import { TProjectFilters } from "@plane/types"; -// components -import { Tooltip } from "@plane/ui"; -import { AppliedAccessFilters, AppliedDateFilters, AppliedMembersFilters } from "@/components/project"; +// types +import { TProjectAppliedDisplayFilterKeys, TProjectFilters } from "@plane/types"; // ui +import { Tooltip } from "@plane/ui"; +// components +import { + AppliedAccessFilters, + AppliedDateFilters, + AppliedMembersFilters, + AppliedProjectDisplayFilters, +} from "@/components/project"; // helpers import { replaceUnderscoreIfSnakeCase } from "@/helpers/string.helper"; -// types type Props = { appliedFilters: TProjectFilters; + appliedDisplayFilters: TProjectAppliedDisplayFilterKeys[]; handleClearAllFilters: () => void; handleRemoveFilter: (key: keyof TProjectFilters, value: string | null) => void; + handleRemoveDisplayFilter: (key: TProjectAppliedDisplayFilterKeys) => void; alwaysAllowEditing?: boolean; filteredProjects: number; totalProjects: number; @@ -23,21 +30,24 @@ const DATE_FILTERS = ["created_at"]; export const ProjectAppliedFiltersList: React.FC = (props) => { const { appliedFilters, + appliedDisplayFilters, handleClearAllFilters, handleRemoveFilter, + handleRemoveDisplayFilter, alwaysAllowEditing, filteredProjects, totalProjects, } = props; - if (!appliedFilters) return null; - if (Object.keys(appliedFilters).length === 0) return null; + if (!appliedFilters && !appliedDisplayFilters) return null; + if (Object.keys(appliedFilters).length === 0 && appliedDisplayFilters.length === 0) return null; const isEditingAllowed = alwaysAllowEditing; return (
+ {/* Applied filters */} {Object.entries(appliedFilters).map(([key, value]) => { const filterKey = key as keyof TProjectFilters; @@ -85,6 +95,22 @@ export const ProjectAppliedFiltersList: React.FC = (props) => {
); })} + {/* Applied display filters */} + {appliedDisplayFilters.length > 0 && ( +
+
+ Projects + handleRemoveDisplayFilter(key)} + /> +
+
+ )} {isEditingAllowed && (
-
- - -
+ {!isArchived && ( +
+ + +
+ )} -
+

{project.description && project.description.trim() !== "" ? project.description : `Created on ${renderFormattedDate(project.created_at)}`}

- 0 ? `${project.members.length} Members` : "No Member" - } - position="top" - > - {projectMembersIds && projectMembersIds.length > 0 ? ( -
- - {projectMembersIds.map((memberId) => { - const member = project.members?.find((m) => m.member_id === memberId); - - if (!member) return null; - - return ; - })} - +
+ 0 ? `${project.members.length} Members` : "No Member" + } + position="top" + > + {projectMembersIds && projectMembersIds.length > 0 ? ( +
+ + {projectMembersIds.map((memberId) => { + const member = project.members?.find((m) => m.member_id === memberId); + if (!member) return null; + return ( + + ); + })} + +
+ ) : ( + No Member Yet + )} +
+ {isArchived &&
Archived
} +
+ {isArchived ? ( + isOwner && ( +
+
{ + e.preventDefault(); + e.stopPropagation(); + setRestoreProject(true); + }} + > +
+ + Restore +
+
+
{ + e.preventDefault(); + e.stopPropagation(); + setDeleteProjectModal(true); + }} + > + +
- ) : ( - No Member Yet - )} - - {project.is_member && - (isOwner || isMember ? ( - { - e.stopPropagation(); - }} - href={`/${workspaceSlug}/projects/${project.id}/settings`} - > - - - ) : ( - - - Joined - - ))} - {!project.is_member && ( -
- -
+ ) + ) : ( + <> + {project.is_member && + (isOwner || isMember ? ( + { + e.stopPropagation(); + }} + href={`/${workspaceSlug}/projects/${project.id}/settings`} + > + + + ) : ( + + + Joined + + ))} + {!project.is_member && ( +
+ +
+ )} + )}
diff --git a/web/components/project/dropdowns/filters/root.tsx b/web/components/project/dropdowns/filters/root.tsx index a7e50cec1..12008eecd 100644 --- a/web/components/project/dropdowns/filters/root.tsx +++ b/web/components/project/dropdowns/filters/root.tsx @@ -51,6 +51,15 @@ export const ProjectFiltersSelection: React.FC = observer((props) => { } title="My projects" /> + + handleDisplayFiltersUpdate({ + archived_projects: !displayFilters.archived_projects, + }) + } + title="Archived" + />
{/* access */} diff --git a/web/components/project/settings/archive-project/archive-restore-modal.tsx b/web/components/project/settings/archive-project/archive-restore-modal.tsx new file mode 100644 index 000000000..88361895c --- /dev/null +++ b/web/components/project/settings/archive-project/archive-restore-modal.tsx @@ -0,0 +1,135 @@ +import { useState, Fragment } from "react"; +import { useRouter } from "next/router"; +import { Dialog, Transition } from "@headlessui/react"; +// ui +import { Button, TOAST_TYPE, setToast } from "@plane/ui"; +// hooks +import { useProject } from "@/hooks/store"; + +type Props = { + workspaceSlug: string; + projectId: string; + isOpen: boolean; + onClose: () => void; + archive: boolean; +}; + +export const ArchiveRestoreProjectModal: React.FC = (props) => { + const { workspaceSlug, projectId, isOpen, onClose, archive } = props; + // router + const router = useRouter(); + // states + const [isLoading, setIsLoading] = useState(false); + // store hooks + const { getProjectById, archiveProject, restoreProject } = useProject(); + + const projectDetails = getProjectById(projectId); + if (!projectDetails) return null; + + const handleClose = () => { + setIsLoading(false); + onClose(); + }; + + const handleArchiveProject = async () => { + setIsLoading(true); + await archiveProject(workspaceSlug, projectId) + .then(() => { + setToast({ + type: TOAST_TYPE.SUCCESS, + title: "Archive success", + message: `${projectDetails.name} has been archived successfully`, + }); + onClose(); + router.push(`/${workspaceSlug}/projects/`); + }) + .catch(() => + setToast({ + type: TOAST_TYPE.ERROR, + title: "Error!", + message: "Project could not be archived. Please try again.", + }) + ) + .finally(() => setIsLoading(false)); + }; + + const handleRestoreProject = async () => { + setIsLoading(true); + await restoreProject(workspaceSlug, projectId) + .then(() => { + setToast({ + type: TOAST_TYPE.SUCCESS, + title: "Restore success", + message: `You can find ${projectDetails.name} in your projects.`, + }); + onClose(); + router.push(`/${workspaceSlug}/projects/`); + }) + .catch(() => + setToast({ + type: TOAST_TYPE.ERROR, + title: "Error!", + message: "Project could not be restored. Please try again.", + }) + ) + .finally(() => setIsLoading(false)); + }; + + return ( + + + +
+ + +
+
+ + +
+

+ {archive ? "Archive" : "Restore"} {projectDetails.name} +

+

+ {archive + ? "This project and its issues, cycles, modules, and pages will be archived. Its issues won’t appear in search. Only project admins can restore the project." + : "Restoring a project will activate it and make it visible to all members of the project. Are you sure you want to continue?"} +

+
+ + +
+
+
+
+
+
+
+
+ ); +}; diff --git a/web/components/project/settings/archive-project/index.tsx b/web/components/project/settings/archive-project/index.tsx new file mode 100644 index 000000000..23da8dcb2 --- /dev/null +++ b/web/components/project/settings/archive-project/index.tsx @@ -0,0 +1,2 @@ +export * from "./selection"; +export * from "./archive-restore-modal"; diff --git a/web/components/project/settings/archive-project/selection.tsx b/web/components/project/settings/archive-project/selection.tsx new file mode 100644 index 000000000..14fb43053 --- /dev/null +++ b/web/components/project/settings/archive-project/selection.tsx @@ -0,0 +1,60 @@ +import React from "react"; +import { ChevronRight, ChevronUp } from "lucide-react"; +import { Disclosure, Transition } from "@headlessui/react"; +// types +import { IProject } from "@plane/types"; +// ui +import { Button, Loader } from "@plane/ui"; + +export interface IArchiveProject { + projectDetails: IProject; + handleArchive: () => void; +} + +export const ArchiveProjectSelection: React.FC = (props) => { + const { projectDetails, handleArchive } = props; + + return ( + + {({ open }) => ( +
+ + Archive project + {open ? : } + + + +
+ + Archiving a project will unlist your project from your side navigation although you will still be able + to access it from your projects page. You can restore the project or delete it whenever you want. + +
+ {projectDetails ? ( +
+ +
+ ) : ( + + + + )} +
+
+
+
+
+ )} +
+ ); +}; diff --git a/web/components/project/settings/delete-project-section.tsx b/web/components/project/settings/delete-project-section.tsx index 991b29207..690f67432 100644 --- a/web/components/project/settings/delete-project-section.tsx +++ b/web/components/project/settings/delete-project-section.tsx @@ -1,12 +1,10 @@ import React from "react"; - -// ui -import { ChevronDown, ChevronUp } from "lucide-react"; +import { ChevronRight, ChevronUp } from "lucide-react"; import { Disclosure, Transition } from "@headlessui/react"; -import { IProject } from "@plane/types"; -import { Button, Loader } from "@plane/ui"; -// icons // types +import { IProject } from "@plane/types"; +// ui +import { Button, Loader } from "@plane/ui"; export interface IDeleteProjectSection { projectDetails: IProject; @@ -17,12 +15,12 @@ export const DeleteProjectSection: React.FC = (props) => const { projectDetails, handleDelete } = props; return ( - + {({ open }) => (
- + Delete Project - {open ? : } + {open ? : } = (props) => leaveTo="transform opacity-0" > -
+
The danger zone of the project delete page is a critical area that requires careful consideration and attention. When deleting a project, all of the data and resources within that project will be diff --git a/web/components/project/settings/index.ts b/web/components/project/settings/index.ts index 0bf79ec17..0f8e9aa6d 100644 --- a/web/components/project/settings/index.ts +++ b/web/components/project/settings/index.ts @@ -1,2 +1,3 @@ export * from "./delete-project-section"; export * from "./features-list"; +export * from "./archive-project"; diff --git a/web/constants/empty-state.ts b/web/constants/empty-state.ts index 8be4a52e2..6705021a5 100644 --- a/web/constants/empty-state.ts +++ b/web/constants/empty-state.ts @@ -125,9 +125,9 @@ const emptyStateDetails = { }, [EmptyStateType.WORKSPACE_PROJECTS]: { key: EmptyStateType.WORKSPACE_PROJECTS, - title: "Start a Project", + title: "No active projects", description: - "Think of each project as the parent for goal-oriented work. Projects are where Jobs, Cycles, and Modules live and, along with your colleagues, help you achieve that goal.", + "Think of each project as the parent for goal-oriented work. Projects are where Jobs, Cycles, and Modules live and, along with your colleagues, help you achieve that goal. Create a new project or filter for archived projects.", path: "/empty-state/onboarding/projects", primaryButton: { text: "Start your first project", diff --git a/web/constants/project.ts b/web/constants/project.ts index 6393a6a6c..c4ef817fd 100644 --- a/web/constants/project.ts +++ b/web/constants/project.ts @@ -1,9 +1,9 @@ // icons import { Globe2, Lock, LucideIcon } from "lucide-react"; +import { TProjectAppliedDisplayFilterKeys, TProjectOrderByOptions } from "@plane/types"; import { SettingIcon } from "@/components/icons"; // types import { Props } from "@/components/icons/types"; -import { TProjectOrderByOptions } from "@plane/types"; export enum EUserProjectRoles { GUEST = 5, @@ -162,3 +162,17 @@ export const PROJECT_ORDER_BY_OPTIONS: { label: "Number of members", }, ]; + +export const PROJECT_DISPLAY_FILTER_OPTIONS: { + key: TProjectAppliedDisplayFilterKeys; + label: string; +}[] = [ + { + key: "my_projects", + label: "My projects", + }, + { + key: "archived_projects", + label: "Archived", + }, +]; diff --git a/web/helpers/project.helper.ts b/web/helpers/project.helper.ts index b4c461bb4..f479eb25c 100644 --- a/web/helpers/project.helper.ts +++ b/web/helpers/project.helper.ts @@ -1,11 +1,11 @@ import sortBy from "lodash/sortBy"; -// helpers -import { satisfiesDateFilter } from "@/helpers/filter.helper"; -import { getDate } from "@/helpers/date-time.helper"; // types import { IProject, TProjectDisplayFilters, TProjectFilters, TProjectOrderByOptions } from "@plane/types"; // constants import { EUserProjectRoles } from "@/constants/project"; +// helpers +import { getDate } from "@/helpers/date-time.helper"; +import { satisfiesDateFilter } from "@/helpers/filter.helper"; /** * Updates the sort order of the project. @@ -93,6 +93,8 @@ export const shouldFilterProject = ( } }); if (displayFilters.my_projects && !project.is_member) fallsInFilters = false; + if (displayFilters.archived_projects && !project.archived_at) fallsInFilters = false; + if (project.archived_at) fallsInFilters = displayFilters.archived_projects ? fallsInFilters : false; return fallsInFilters; }; diff --git a/web/pages/[workspaceSlug]/projects/[projectId]/settings/index.tsx b/web/pages/[workspaceSlug]/projects/[projectId]/settings/index.tsx index f099a97f7..946d9b9cb 100644 --- a/web/pages/[workspaceSlug]/projects/[projectId]/settings/index.tsx +++ b/web/pages/[workspaceSlug]/projects/[projectId]/settings/index.tsx @@ -2,26 +2,29 @@ import { useState, ReactElement } from "react"; import { observer } from "mobx-react-lite"; import { useRouter } from "next/router"; import useSWR from "swr"; -// hooks +// components import { PageHead } from "@/components/core"; import { ProjectSettingHeader } from "@/components/headers"; import { + ArchiveRestoreProjectModal, + ArchiveProjectSelection, DeleteProjectModal, DeleteProjectSection, ProjectDetailsForm, ProjectDetailsFormLoader, } from "@/components/project"; +// hooks import { useProject } from "@/hooks/store"; // layouts import { AppLayout } from "@/layouts/app-layout"; import { ProjectSettingLayout } from "@/layouts/settings-layout"; -// components // types import { NextPageWithLayout } from "@/lib/types"; const GeneralSettingsPage: NextPageWithLayout = observer(() => { // states const [selectProject, setSelectedProject] = useState(null); + const [archiveProject, setArchiveProject] = useState(false); // router const router = useRouter(); const { workspaceSlug, projectId } = router.query; @@ -42,12 +45,21 @@ const GeneralSettingsPage: NextPageWithLayout = observer(() => { return ( <> - {currentProjectDetails && ( - setSelectedProject(null)} - /> + {currentProjectDetails && workspaceSlug && projectId && ( + <> + setArchiveProject(false)} + archive + /> + setSelectedProject(null)} + /> + )}
@@ -63,10 +75,16 @@ const GeneralSettingsPage: NextPageWithLayout = observer(() => { )} {isAdmin && ( - setSelectedProject(currentProjectDetails.id ?? null)} - /> + <> + setArchiveProject(true)} + /> + setSelectedProject(currentProjectDetails.id ?? null)} + /> + )}
diff --git a/web/pages/[workspaceSlug]/projects/index.tsx b/web/pages/[workspaceSlug]/projects/index.tsx index 1e153b688..5db5daa34 100644 --- a/web/pages/[workspaceSlug]/projects/index.tsx +++ b/web/pages/[workspaceSlug]/projects/index.tsx @@ -1,6 +1,6 @@ import { ReactElement, useCallback } from "react"; import { observer } from "mobx-react"; -import { TProjectFilters } from "@plane/types"; +import { TProjectAppliedDisplayFilterKeys, TProjectFilters } from "@plane/types"; // components import { PageHead } from "@/components/core"; import { ProjectsHeader } from "@/components/headers"; @@ -19,8 +19,15 @@ const ProjectsPage: NextPageWithLayout = observer(() => { router: { workspaceSlug }, } = useApplication(); const { currentWorkspace } = useWorkspace(); - const { workspaceProjectIds, filteredProjectIds } = useProject(); - const { currentWorkspaceFilters, clearAllFilters, updateFilters } = useProjectFilter(); + const { totalProjectIds, filteredProjectIds } = useProject(); + const { + currentWorkspaceFilters, + currentWorkspaceAppliedDisplayFilters, + clearAllFilters, + clearAllAppliedDisplayFilters, + updateFilters, + updateDisplayFilters, + } = useProjectFilter(); // derived values const pageTitle = currentWorkspace?.name ? `${currentWorkspace?.name} - Projects` : undefined; @@ -37,18 +44,35 @@ const ProjectsPage: NextPageWithLayout = observer(() => { [currentWorkspaceFilters, updateFilters, workspaceSlug] ); + const handleRemoveDisplayFilter = useCallback( + (key: TProjectAppliedDisplayFilterKeys) => { + if (!workspaceSlug) return; + updateDisplayFilters(workspaceSlug.toString(), { [key]: false }); + }, + [updateDisplayFilters, workspaceSlug] + ); + + const handleClearAllFilters = useCallback(() => { + if (!workspaceSlug) return; + clearAllFilters(workspaceSlug.toString()); + clearAllAppliedDisplayFilters(workspaceSlug.toString()); + }, [clearAllFilters, clearAllAppliedDisplayFilters, workspaceSlug]); + return ( <>
- {calculateTotalFilters(currentWorkspaceFilters ?? {}) !== 0 && ( + {(calculateTotalFilters(currentWorkspaceFilters ?? {}) !== 0 || + currentWorkspaceAppliedDisplayFilters?.length !== 0) && (
clearAllFilters(`${workspaceSlug}`)} + appliedDisplayFilters={currentWorkspaceAppliedDisplayFilters ?? []} + handleClearAllFilters={handleClearAllFilters} handleRemoveFilter={handleRemoveFilter} + handleRemoveDisplayFilter={handleRemoveDisplayFilter} filteredProjects={filteredProjectIds?.length ?? 0} - totalProjects={workspaceProjectIds?.length ?? 0} + totalProjects={totalProjectIds?.length ?? 0} alwaysAllowEditing />
diff --git a/web/services/project/index.ts b/web/services/project/index.ts index 18cf1200a..d131ceb6b 100644 --- a/web/services/project/index.ts +++ b/web/services/project/index.ts @@ -4,3 +4,4 @@ export * from "./project-export.service"; export * from "./project-member.service"; export * from "./project-state.service"; export * from "./project-publish.service"; +export * from "./project-archive.service"; diff --git a/web/services/project/project-archive.service.ts b/web/services/project/project-archive.service.ts new file mode 100644 index 000000000..5fdca54b6 --- /dev/null +++ b/web/services/project/project-archive.service.ts @@ -0,0 +1,31 @@ +// helpers +import { API_BASE_URL } from "@/helpers/common.helper"; +// services +import { APIService } from "@/services/api.service"; + +export class ProjectArchiveService extends APIService { + constructor() { + super(API_BASE_URL); + } + + async archiveProject( + workspaceSlug: string, + projectId: string + ): Promise<{ + archived_at: string; + }> { + return this.post(`/api/workspaces/${workspaceSlug}/projects/${projectId}/archive/`) + .then((response) => response?.data) + .catch((error) => { + throw error?.response?.data; + }); + } + + async restoreProject(workspaceSlug: string, projectId: string): Promise { + return this.delete(`/api/workspaces/${workspaceSlug}/projects/${projectId}/archive/`) + .then((response) => response?.data) + .catch((error) => { + throw error?.response?.data; + }); + } +} diff --git a/web/store/project/project.store.ts b/web/store/project/project.store.ts index dfd35ac59..a12039ba6 100644 --- a/web/store/project/project.store.ts +++ b/web/store/project/project.store.ts @@ -3,12 +3,15 @@ import sortBy from "lodash/sortBy"; import { observable, action, computed, makeObservable, runInAction } from "mobx"; import { computedFn } from "mobx-utils"; // types -import { IssueLabelService, IssueService } from "@/services/issue"; -import { ProjectService, ProjectStateService } from "@/services/project"; import { IProject } from "@plane/types"; -import { RootStore } from "../root.store"; +// helpers import { orderProjects, shouldFilterProject } from "@/helpers/project.helper"; // services +import { IssueLabelService, IssueService } from "@/services/issue"; +import { ProjectService, ProjectStateService, ProjectArchiveService } from "@/services/project"; +// store +import { RootStore } from "../root.store"; + export interface IProjectStore { // observables projectMap: { @@ -17,6 +20,8 @@ export interface IProjectStore { // computed filteredProjectIds: string[] | undefined; workspaceProjectIds: string[] | undefined; + archivedProjectIds: string[] | undefined; + totalProjectIds: string[] | undefined; joinedProjectIds: string[]; favoriteProjectIds: string[]; currentProjectDetails: IProject | undefined; @@ -35,6 +40,9 @@ export interface IProjectStore { createProject: (workspaceSlug: string, data: Partial) => Promise; updateProject: (workspaceSlug: string, projectId: string, data: Partial) => Promise; deleteProject: (workspaceSlug: string, projectId: string) => Promise; + // archive actions + archiveProject: (workspaceSlug: string, projectId: string) => Promise; + restoreProject: (workspaceSlug: string, projectId: string) => Promise; } export class ProjectStore implements IProjectStore { @@ -46,6 +54,7 @@ export class ProjectStore implements IProjectStore { rootStore: RootStore; // service projectService; + projectArchiveService; issueLabelService; issueService; stateService; @@ -57,6 +66,8 @@ export class ProjectStore implements IProjectStore { // computed filteredProjectIds: computed, workspaceProjectIds: computed, + archivedProjectIds: computed, + totalProjectIds: computed, currentProjectDetails: computed, joinedProjectIds: computed, favoriteProjectIds: computed, @@ -76,6 +87,7 @@ export class ProjectStore implements IProjectStore { this.rootStore = _rootStore; // services this.projectService = new ProjectService(); + this.projectArchiveService = new ProjectArchiveService(); this.issueService = new IssueService(); this.issueLabelService = new IssueLabelService(); this.stateService = new ProjectStateService(); @@ -109,11 +121,42 @@ export class ProjectStore implements IProjectStore { get workspaceProjectIds() { const workspaceDetails = this.rootStore.workspaceRoot.currentWorkspace; if (!workspaceDetails) return; - const workspaceProjects = Object.values(this.projectMap).filter((p) => p.workspace === workspaceDetails.id); + const workspaceProjects = Object.values(this.projectMap).filter( + (p) => p.workspace === workspaceDetails.id && !p.archived_at + ); const projectIds = workspaceProjects.map((p) => p.id); return projectIds ?? null; } + /** + * Returns archived project IDs belong to current workspace. + */ + get archivedProjectIds() { + const currentWorkspace = this.rootStore.workspaceRoot.currentWorkspace; + if (!currentWorkspace) return; + + let projects = Object.values(this.projectMap ?? {}); + projects = sortBy(projects, "archived_at"); + + const projectIds = projects + .filter((project) => project.workspace === currentWorkspace.id && !!project.archived_at) + .map((project) => project.id); + return projectIds; + } + + /** + * Returns total project IDs belong to the current workspace + */ + // workspaceProjectIds + archivedProjectIds + get totalProjectIds() { + const currentWorkspace = this.rootStore.workspaceRoot.currentWorkspace; + if (!currentWorkspace) return; + + const workspaceProjects = this.workspaceProjectIds ?? []; + const archivedProjects = this.archivedProjectIds ?? []; + return [...workspaceProjects, ...archivedProjects]; + } + /** * Returns current project details */ @@ -133,7 +176,7 @@ export class ProjectStore implements IProjectStore { projects = sortBy(projects, "sort_order"); const projectIds = projects - .filter((project) => project.workspace === currentWorkspace.id && project.is_member) + .filter((project) => project.workspace === currentWorkspace.id && project.is_member && !project.archived_at) .map((project) => project.id); return projectIds; } @@ -149,7 +192,10 @@ export class ProjectStore implements IProjectStore { projects = sortBy(projects, "created_at"); const projectIds = projects - .filter((project) => project.workspace === currentWorkspace.id && project.is_member && project.is_favorite) + .filter( + (project) => + project.workspace === currentWorkspace.id && project.is_member && project.is_favorite && !project.archived_at + ) .map((project) => project.id); return projectIds; } @@ -348,4 +394,48 @@ export class ProjectStore implements IProjectStore { this.fetchProjects(workspaceSlug); } }; + + /** + * Archives a project from specific workspace and updates it in the store + * @param workspaceSlug + * @param projectId + * @returns Promise + */ + archiveProject = async (workspaceSlug: string, projectId: string) => { + await this.projectArchiveService + .archiveProject(workspaceSlug, projectId) + .then((response) => { + runInAction(() => { + set(this.projectMap, [projectId, "archived_at"], response.archived_at); + }); + }) + .catch((error) => { + console.log("Failed to archive project from project store"); + this.fetchProjects(workspaceSlug); + this.fetchProjectDetails(workspaceSlug, projectId); + throw error; + }); + }; + + /** + * Restores a project from specific workspace and updates it in the store + * @param workspaceSlug + * @param projectId + * @returns Promise + */ + restoreProject = async (workspaceSlug: string, projectId: string) => { + await this.projectArchiveService + .restoreProject(workspaceSlug, projectId) + .then(() => { + runInAction(() => { + set(this.projectMap, [projectId, "archived_at"], null); + }); + }) + .catch((error) => { + console.log("Failed to restore project from project store"); + this.fetchProjects(workspaceSlug); + this.fetchProjectDetails(workspaceSlug, projectId); + throw error; + }); + }; } diff --git a/web/store/project/project_filter.store.ts b/web/store/project/project_filter.store.ts index 7d6aff96f..94d5f7073 100644 --- a/web/store/project/project_filter.store.ts +++ b/web/store/project/project_filter.store.ts @@ -1,9 +1,10 @@ +import set from "lodash/set"; import { action, computed, observable, makeObservable, runInAction, reaction } from "mobx"; import { computedFn } from "mobx-utils"; -import set from "lodash/set"; // types +import { TProjectDisplayFilters, TProjectFilters, TProjectAppliedDisplayFilterKeys } from "@plane/types"; +// store import { RootStore } from "@/store/root.store"; -import { TProjectDisplayFilters, TProjectFilters } from "@plane/types"; export interface IProjectFilterStore { // observables @@ -12,6 +13,7 @@ export interface IProjectFilterStore { searchQuery: string; // computed currentWorkspaceDisplayFilters: TProjectDisplayFilters | undefined; + currentWorkspaceAppliedDisplayFilters: TProjectAppliedDisplayFilterKeys[] | undefined; currentWorkspaceFilters: TProjectFilters | undefined; // computed functions getDisplayFiltersByWorkspaceSlug: (workspaceSlug: string) => TProjectDisplayFilters | undefined; @@ -21,6 +23,7 @@ export interface IProjectFilterStore { updateFilters: (workspaceSlug: string, filters: TProjectFilters) => void; updateSearchQuery: (query: string) => void; clearAllFilters: (workspaceSlug: string) => void; + clearAllAppliedDisplayFilters: (workspaceSlug: string) => void; } export class ProjectFilterStore implements IProjectFilterStore { @@ -39,12 +42,14 @@ export class ProjectFilterStore implements IProjectFilterStore { searchQuery: observable.ref, // computed currentWorkspaceDisplayFilters: computed, + currentWorkspaceAppliedDisplayFilters: computed, currentWorkspaceFilters: computed, // actions updateDisplayFilters: action, updateFilters: action, updateSearchQuery: action, clearAllFilters: action, + clearAllAppliedDisplayFilters: action, }); // root store this.rootStore = _rootStore; @@ -67,6 +72,21 @@ export class ProjectFilterStore implements IProjectFilterStore { return this.displayFilters[workspaceSlug]; } + /** + * @description get project state applied display filter of the current workspace + * @returns {TProjectAppliedDisplayFilterKeys[] | undefined} // An array of keys of applied display filters + */ + // TODO: Figure out a better approach for this + get currentWorkspaceAppliedDisplayFilters() { + const workspaceSlug = this.rootStore.app.router.workspaceSlug; + if (!workspaceSlug) return; + const displayFilters = this.displayFilters[workspaceSlug]; + return Object.keys(displayFilters).filter( + (key): key is TProjectAppliedDisplayFilterKeys => + ["my_projects", "archived_projects"].includes(key) && !!displayFilters[key as keyof TProjectDisplayFilters] + ); + } + /** * @description get filters of the current workspace */ @@ -143,4 +163,17 @@ export class ProjectFilterStore implements IProjectFilterStore { this.filters[workspaceSlug] = {}; }); }; + + /** + * @description clear project display filters of a workspace + * @param {string} workspaceSlug + */ + clearAllAppliedDisplayFilters = (workspaceSlug: string) => { + runInAction(() => { + if (!this.currentWorkspaceAppliedDisplayFilters) return; + this.currentWorkspaceAppliedDisplayFilters.forEach((key) => { + set(this.displayFilters, [workspaceSlug, key], false); + }); + }); + }; }