import { FC, useCallback, useEffect, useMemo, useState } from "react"; import { observer } from "mobx-react-lite"; import { useRouter } from "next/router"; import { DayPicker } from "react-day-picker"; import { Popover } from "@headlessui/react"; // icons import { CheckCircle2, ChevronDown, ChevronUp, Clock, FileStack, Trash2, XCircle } from "lucide-react"; // ui import { Button, TOAST_TYPE, setToast } from "@plane/ui"; // components import { AcceptIssueModal, DeclineIssueModal, DeleteInboxIssueModal, SelectDuplicateInboxIssueModal, } from "components/inbox"; import { E_INBOX, ISSUE_DELETED } from "constants/event-tracker"; import { EUserProjectRoles } from "constants/project"; // hooks import { useUser, useInboxIssues, useIssueDetail, useWorkspace, useEventTracker } from "hooks/store"; // types import type { TInboxDetailedStatus } from "@plane/types"; type TInboxIssueActionsHeader = { workspaceSlug: string; projectId: string; inboxId: string; inboxIssueId: string | undefined; }; type TInboxIssueOperations = { updateInboxIssueStatus: (data: TInboxDetailedStatus) => Promise; removeInboxIssue: () => Promise; }; export const InboxIssueActionsHeader: FC = observer((props) => { const { workspaceSlug, projectId, inboxId, inboxIssueId } = props; // router const router = useRouter(); // hooks const { captureIssueEvent } = useEventTracker(); const { currentWorkspace } = useWorkspace(); const { issues: { getInboxIssuesByInboxId, getInboxIssueByIssueId, updateInboxIssueStatus, removeInboxIssue }, } = useInboxIssues(); const { issue: { getIssueById }, } = useIssueDetail(); const { currentUser, membership: { currentProjectRole }, } = useUser(); // states const [date, setDate] = useState(new Date()); const [selectDuplicateIssue, setSelectDuplicateIssue] = useState(false); const [acceptIssueModal, setAcceptIssueModal] = useState(false); const [declineIssueModal, setDeclineIssueModal] = useState(false); const [deleteIssueModal, setDeleteIssueModal] = useState(false); // derived values const inboxIssues = getInboxIssuesByInboxId(inboxId); const issueStatus = (inboxIssueId && inboxId && getInboxIssueByIssueId(inboxId, inboxIssueId)) || undefined; const issue = (inboxIssueId && getIssueById(inboxIssueId)) || undefined; const currentIssueIndex = inboxIssues?.findIndex((issue) => issue === inboxIssueId) ?? 0; const inboxIssueOperations: TInboxIssueOperations = useMemo( () => ({ updateInboxIssueStatus: async (data: TInboxDetailedStatus) => { try { if (!workspaceSlug || !projectId || !inboxId || !inboxIssueId) throw new Error("Missing required parameters"); await updateInboxIssueStatus(workspaceSlug, projectId, inboxId, inboxIssueId, data); } catch (error) { setToast({ type: TOAST_TYPE.ERROR, title: "Error!", message: "Something went wrong while updating inbox status. Please try again.", }); } }, removeInboxIssue: async () => { try { if (!workspaceSlug || !projectId || !inboxId || !inboxIssueId || !currentWorkspace) throw new Error("Missing required parameters"); await removeInboxIssue(workspaceSlug, projectId, inboxId, inboxIssueId); captureIssueEvent({ eventName: ISSUE_DELETED, payload: { id: inboxIssueId, state: "SUCCESS", element: E_INBOX, }, }); router.push({ pathname: `/${workspaceSlug}/projects/${projectId}/inbox/${inboxId}`, }); } catch (error) { setToast({ type: TOAST_TYPE.ERROR, title: "Error!", message: "Something went wrong while deleting inbox issue. Please try again.", }); captureIssueEvent({ eventName: ISSUE_DELETED, payload: { id: inboxIssueId, state: "FAILED", element: E_INBOX, }, }); } }, }), [ currentWorkspace, workspaceSlug, projectId, inboxId, inboxIssueId, updateInboxIssueStatus, removeInboxIssue, captureIssueEvent, router, ] ); const handleInboxIssueNavigation = useCallback( (direction: "next" | "prev") => { if (!inboxIssues || !inboxIssueId) return; const activeElement = document.activeElement as HTMLElement; if (activeElement && (activeElement.classList.contains("tiptap") || activeElement.id === "title-input")) return; const nextIssueIndex = direction === "next" ? (currentIssueIndex + 1) % inboxIssues.length : (currentIssueIndex - 1 + inboxIssues.length) % inboxIssues.length; const nextIssueId = inboxIssues[nextIssueIndex]; if (!nextIssueId) return; router.push({ pathname: `/${workspaceSlug}/projects/${projectId}/inbox/${inboxId}`, query: { inboxIssueId: nextIssueId, }, }); }, [workspaceSlug, projectId, inboxId, inboxIssues, inboxIssueId, currentIssueIndex, router] ); const onKeyDown = useCallback( (e: KeyboardEvent) => { if (e.key === "ArrowUp") { handleInboxIssueNavigation("prev"); } else if (e.key === "ArrowDown") { handleInboxIssueNavigation("next"); } }, [handleInboxIssueNavigation] ); useEffect(() => { document.addEventListener("keydown", onKeyDown); return () => { document.removeEventListener("keydown", onKeyDown); }; }, [onKeyDown]); const isAllowed = !!currentProjectRole && currentProjectRole >= EUserProjectRoles.MEMBER; const today = new Date(); const tomorrow = new Date(today); tomorrow.setDate(today.getDate() + 1); useEffect(() => { if (!issueStatus || !issueStatus.snoozed_till) return; setDate(new Date(issueStatus.snoozed_till)); }, [issueStatus]); if (!issueStatus || !issue || !inboxIssues) return <>; return ( <> {issue && ( <> setSelectDuplicateIssue(false)} value={issueStatus.duplicate_to} onSubmit={(dupIssueId) => { inboxIssueOperations .updateInboxIssueStatus({ status: 2, duplicate_to: dupIssueId, }) .finally(() => setSelectDuplicateIssue(false)); }} /> setAcceptIssueModal(false)} onSubmit={async () => { await inboxIssueOperations .updateInboxIssueStatus({ status: 1, }) .finally(() => setAcceptIssueModal(false)); }} /> setDeclineIssueModal(false)} onSubmit={async () => { await inboxIssueOperations .updateInboxIssueStatus({ status: -1, }) .finally(() => setDeclineIssueModal(false)); }} /> setDeleteIssueModal(false)} onSubmit={async () => { await inboxIssueOperations.removeInboxIssue().finally(() => setDeclineIssueModal(false)); }} /> )} {inboxIssueId && (
{currentIssueIndex + 1}/{inboxIssues?.length ?? 0}
{isAllowed && (issueStatus.status === 0 || issueStatus.status === -2) && (
{({ close }) => (
{ if (!date) return; setDate(date); }} mode="single" className="rounded-md border border-custom-border-200 p-3" disabled={[ { before: tomorrow, }, ]} />
)}
)} {isAllowed && issueStatus.status === -2 && (
)} {isAllowed && (issueStatus.status === 0 || issueStatus.status === -2) && (
)} {isAllowed && issueStatus.status === -2 && (
)} {(isAllowed || currentUser?.id === issue?.created_by) && (
)}
)} ); });