2023-09-25 09:30:35 +00:00
|
|
|
import { useCallback, useState } from "react";
|
2023-09-22 09:52:05 +00:00
|
|
|
import { useRouter } from "next/router";
|
|
|
|
|
|
|
|
import useSWR from "swr";
|
|
|
|
|
|
|
|
// hook
|
|
|
|
import useUser from "hooks/use-user";
|
2023-09-25 14:37:08 +00:00
|
|
|
// context
|
|
|
|
import { useProjectMyMembership } from "contexts/project-member.context";
|
2023-09-22 09:52:05 +00:00
|
|
|
// services
|
|
|
|
import workspaceService from "services/workspace.service";
|
|
|
|
// layouts
|
|
|
|
import { WorkspaceAuthorizationLayout } from "layouts/auth-layout";
|
|
|
|
// components
|
|
|
|
import { SpreadsheetView } from "components/core";
|
|
|
|
import { WorkspaceViewsNavigation } from "components/workspace/views/workpace-view-navigation";
|
|
|
|
import { WorkspaceIssuesViewOptions } from "components/issues/workspace-views/workspace-issue-view-option";
|
2023-09-25 09:30:35 +00:00
|
|
|
import { CreateUpdateViewModal } from "components/views";
|
|
|
|
import { CreateUpdateIssueModal, DeleteIssueModal } from "components/issues";
|
2023-09-22 09:52:05 +00:00
|
|
|
// ui
|
|
|
|
import { EmptyState, PrimaryButton } from "components/ui";
|
|
|
|
// icons
|
|
|
|
import { PlusIcon } from "@heroicons/react/24/outline";
|
|
|
|
import { CheckCircle } from "lucide-react";
|
|
|
|
// images
|
|
|
|
import emptyView from "public/empty-state/view.svg";
|
|
|
|
// fetch-keys
|
|
|
|
import { WORKSPACE_VIEW_DETAILS, WORKSPACE_VIEW_ISSUES } from "constants/fetch-keys";
|
2023-09-25 09:30:35 +00:00
|
|
|
// types
|
|
|
|
import { IIssue } from "types";
|
2023-09-22 09:52:05 +00:00
|
|
|
|
|
|
|
const WorkspaceViewAssignedIssue: React.FC = () => {
|
2023-09-25 09:30:35 +00:00
|
|
|
const [createViewModal, setCreateViewModal] = useState<any>(null);
|
|
|
|
|
|
|
|
// create issue modal
|
|
|
|
const [createIssueModal, setCreateIssueModal] = useState(false);
|
|
|
|
const [preloadedData, setPreloadedData] = useState<
|
|
|
|
(Partial<IIssue> & { actionType: "createIssue" | "edit" | "delete" }) | undefined
|
|
|
|
>(undefined);
|
|
|
|
|
|
|
|
// update issue modal
|
|
|
|
const [editIssueModal, setEditIssueModal] = useState(false);
|
|
|
|
const [issueToEdit, setIssueToEdit] = useState<
|
|
|
|
(IIssue & { actionType: "edit" | "delete" }) | undefined
|
|
|
|
>(undefined);
|
|
|
|
|
|
|
|
// delete issue modal
|
|
|
|
const [deleteIssueModal, setDeleteIssueModal] = useState(false);
|
|
|
|
const [issueToDelete, setIssueToDelete] = useState<IIssue | null>(null);
|
|
|
|
|
2023-09-22 09:52:05 +00:00
|
|
|
const router = useRouter();
|
|
|
|
const { workspaceSlug, workspaceViewId } = router.query;
|
|
|
|
|
|
|
|
const { user } = useUser();
|
|
|
|
|
2023-09-25 14:37:08 +00:00
|
|
|
const { memberRole } = useProjectMyMembership();
|
|
|
|
|
|
|
|
const params: any = {
|
|
|
|
assignees: user?.id ?? undefined,
|
|
|
|
sub_issue: false,
|
|
|
|
};
|
|
|
|
|
2023-09-22 09:52:05 +00:00
|
|
|
const { data: viewDetails, error } = useSWR(
|
|
|
|
workspaceSlug && workspaceViewId ? WORKSPACE_VIEW_DETAILS(workspaceViewId.toString()) : null,
|
|
|
|
workspaceSlug && workspaceViewId
|
|
|
|
? () => workspaceService.getViewDetails(workspaceSlug.toString(), workspaceViewId.toString())
|
|
|
|
: null
|
|
|
|
);
|
|
|
|
|
|
|
|
const { data: viewIssues, mutate: mutateIssues } = useSWR(
|
|
|
|
workspaceSlug ? WORKSPACE_VIEW_ISSUES(workspaceSlug.toString(), params) : null,
|
|
|
|
workspaceSlug ? () => workspaceService.getViewIssues(workspaceSlug.toString(), params) : null
|
|
|
|
);
|
|
|
|
|
2023-09-25 09:30:35 +00:00
|
|
|
const makeIssueCopy = useCallback(
|
|
|
|
(issue: IIssue) => {
|
|
|
|
setCreateIssueModal(true);
|
|
|
|
|
|
|
|
setPreloadedData({ ...issue, name: `${issue.name} (Copy)`, actionType: "createIssue" });
|
|
|
|
},
|
|
|
|
[setCreateIssueModal, setPreloadedData]
|
|
|
|
);
|
|
|
|
|
|
|
|
const handleEditIssue = useCallback(
|
|
|
|
(issue: IIssue) => {
|
|
|
|
setEditIssueModal(true);
|
|
|
|
setIssueToEdit({
|
|
|
|
...issue,
|
|
|
|
actionType: "edit",
|
|
|
|
cycle: issue.issue_cycle ? issue.issue_cycle.cycle : null,
|
|
|
|
module: issue.issue_module ? issue.issue_module.module : null,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
[setEditIssueModal, setIssueToEdit]
|
|
|
|
);
|
|
|
|
|
|
|
|
const handleDeleteIssue = useCallback(
|
|
|
|
(issue: IIssue) => {
|
|
|
|
setDeleteIssueModal(true);
|
|
|
|
setIssueToDelete(issue);
|
|
|
|
},
|
|
|
|
[setDeleteIssueModal, setIssueToDelete]
|
|
|
|
);
|
|
|
|
|
|
|
|
const handleIssueAction = useCallback(
|
|
|
|
(issue: IIssue, action: "copy" | "edit" | "delete" | "updateDraft") => {
|
|
|
|
if (action === "copy") makeIssueCopy(issue);
|
|
|
|
else if (action === "edit") handleEditIssue(issue);
|
|
|
|
else if (action === "delete") handleDeleteIssue(issue);
|
|
|
|
},
|
|
|
|
[makeIssueCopy, handleEditIssue, handleDeleteIssue]
|
|
|
|
);
|
|
|
|
|
2023-09-22 09:52:05 +00:00
|
|
|
return (
|
2023-09-25 09:30:35 +00:00
|
|
|
<WorkspaceAuthorizationLayout
|
|
|
|
breadcrumbs={
|
|
|
|
<div className="flex gap-2 items-center">
|
|
|
|
<CheckCircle className="h-[18px] w-[18px] stroke-[1.5]" />
|
|
|
|
<span className="text-sm font-medium">
|
|
|
|
{viewDetails ? `${viewDetails.name} Issues` : "Workspace Issues"}
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
right={
|
|
|
|
<div className="flex items-center gap-2">
|
|
|
|
<WorkspaceIssuesViewOptions />
|
|
|
|
<PrimaryButton
|
|
|
|
className="flex items-center gap-2"
|
|
|
|
onClick={() => {
|
|
|
|
const e = new KeyboardEvent("keydown", { key: "c" });
|
|
|
|
document.dispatchEvent(e);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<PlusIcon className="h-4 w-4" />
|
|
|
|
Add Issue
|
|
|
|
</PrimaryButton>
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
>
|
|
|
|
<CreateUpdateIssueModal
|
|
|
|
isOpen={createIssueModal && preloadedData?.actionType === "createIssue"}
|
|
|
|
handleClose={() => setCreateIssueModal(false)}
|
|
|
|
prePopulateData={{
|
|
|
|
...preloadedData,
|
|
|
|
}}
|
|
|
|
onSubmit={async () => {
|
|
|
|
mutateIssues();
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
<CreateUpdateIssueModal
|
|
|
|
isOpen={editIssueModal && issueToEdit?.actionType !== "delete"}
|
|
|
|
handleClose={() => setEditIssueModal(false)}
|
|
|
|
data={issueToEdit}
|
|
|
|
onSubmit={async () => {
|
|
|
|
mutateIssues();
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
<DeleteIssueModal
|
|
|
|
handleClose={() => setDeleteIssueModal(false)}
|
|
|
|
isOpen={deleteIssueModal}
|
|
|
|
data={issueToDelete}
|
|
|
|
user={user}
|
|
|
|
onSubmit={async () => {
|
|
|
|
mutateIssues();
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
<CreateUpdateViewModal
|
|
|
|
isOpen={createViewModal !== null}
|
|
|
|
handleClose={() => setCreateViewModal(null)}
|
|
|
|
viewType="workspace"
|
|
|
|
preLoadedData={createViewModal}
|
|
|
|
user={user}
|
|
|
|
/>
|
|
|
|
<div className="h-full flex flex-col overflow-hidden bg-custom-background-100">
|
|
|
|
<div className="h-full w-full border-b border-custom-border-300">
|
|
|
|
<WorkspaceViewsNavigation handleAddView={() => setCreateViewModal(true)} />
|
|
|
|
{error ? (
|
|
|
|
<EmptyState
|
|
|
|
image={emptyView}
|
|
|
|
title="View does not exist"
|
|
|
|
description="The view you are looking for does not exist or has been deleted."
|
|
|
|
primaryButton={{
|
|
|
|
text: "View other views",
|
|
|
|
onClick: () => router.push(`/${workspaceSlug}/workspace-views`),
|
2023-09-22 09:52:05 +00:00
|
|
|
}}
|
2023-09-25 09:30:35 +00:00
|
|
|
/>
|
|
|
|
) : (
|
|
|
|
<div className="h-full w-full flex flex-col">
|
|
|
|
<SpreadsheetView
|
|
|
|
spreadsheetIssues={viewIssues}
|
|
|
|
mutateIssues={mutateIssues}
|
|
|
|
handleIssueAction={handleIssueAction}
|
|
|
|
disableUserActions={false}
|
|
|
|
user={user}
|
2023-09-25 14:37:08 +00:00
|
|
|
userAuth={memberRole}
|
2023-09-22 09:52:05 +00:00
|
|
|
/>
|
2023-09-25 09:30:35 +00:00
|
|
|
</div>
|
|
|
|
)}
|
2023-09-22 09:52:05 +00:00
|
|
|
</div>
|
2023-09-25 09:30:35 +00:00
|
|
|
</div>
|
|
|
|
</WorkspaceAuthorizationLayout>
|
2023-09-22 09:52:05 +00:00
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default WorkspaceViewAssignedIssue;
|