import { FC, useRef, useState } from "react"; import { observer } from "mobx-react-lite"; import { MoveRight, MoveDiagonal, Bell, Link2, Trash2 } from "lucide-react"; // hooks import useOutsideClickDetector from "hooks/use-outside-click-detector"; // store hooks import { useIssueDetail, useUser } from "hooks/store"; // components import { DeleteArchivedIssueModal, DeleteIssueModal, IssueActivity, IssueUpdateStatus, PeekOverviewIssueDetails, PeekOverviewProperties, } from "components/issues"; // ui import { Button, CenterPanelIcon, CustomSelect, FullScreenPanelIcon, SidePanelIcon, Spinner } from "@plane/ui"; // types import { TIssue } from "@plane/types"; interface IIssueView { workspaceSlug: string; projectId: string; issueId: string; isLoading?: boolean; isArchived?: boolean; issue: TIssue | undefined; handleCopyText: (e: React.MouseEvent<HTMLButtonElement>) => void; redirectToIssueDetail: () => void; issueUpdate: (issue: Partial<TIssue>) => void; issueDelete: () => Promise<void>; issueReactionCreate: (reaction: string) => void; issueReactionRemove: (reaction: string) => void; issueCommentCreate: (comment: any) => void; issueCommentUpdate: (comment: any) => void; issueCommentRemove: (commentId: string) => void; issueCommentReactionCreate: (commentId: string, reaction: string) => void; issueCommentReactionRemove: (commentId: string, reaction: string) => void; issueSubscriptionCreate: () => void; issueSubscriptionRemove: () => void; disableUserActions?: boolean; showCommentAccessSpecifier?: boolean; issueOperations: any; } type TPeekModes = "side-peek" | "modal" | "full-screen"; const PEEK_OPTIONS: { key: TPeekModes; icon: any; title: string }[] = [ { key: "side-peek", icon: SidePanelIcon, title: "Side Peek", }, { key: "modal", icon: CenterPanelIcon, title: "Modal", }, { key: "full-screen", icon: FullScreenPanelIcon, title: "Full Screen", }, ]; export const IssueView: FC<IIssueView> = observer((props) => { const { workspaceSlug, projectId, issueId, issue, isLoading, isArchived, handleCopyText, redirectToIssueDetail, issueUpdate, issueReactionCreate, issueReactionRemove, issueCommentCreate, issueCommentUpdate, issueCommentRemove, issueCommentReactionCreate, issueCommentReactionRemove, issueSubscriptionCreate, issueSubscriptionRemove, issueDelete, disableUserActions = false, showCommentAccessSpecifier = false, issueOperations, } = props; // states const [peekMode, setPeekMode] = useState<TPeekModes>("side-peek"); const [isSubmitting, setIsSubmitting] = useState<"submitting" | "submitted" | "saved">("saved"); // ref const issuePeekOverviewRef = useRef<HTMLDivElement>(null); // store hooks const { activity, reaction, subscription, setPeekIssue, isAnyModalOpen, isDeleteIssueModalOpen, toggleDeleteIssueModal, } = useIssueDetail(); const { currentUser } = useUser(); const removeRoutePeekId = () => { setPeekIssue(undefined); }; const issueReactions = reaction.getReactionsByIssueId(issueId) || []; const issueActivity = activity.getActivitiesByIssueId(issueId); const issueSubscription = subscription.getSubscriptionByIssueId(issueId) || {}; const currentMode = PEEK_OPTIONS.find((m) => m.key === peekMode); useOutsideClickDetector(issuePeekOverviewRef, () => !isAnyModalOpen && removeRoutePeekId()); return ( <> {issue && !isArchived && ( <DeleteIssueModal isOpen={isDeleteIssueModalOpen} handleClose={() => toggleDeleteIssueModal(false)} data={issue} onSubmit={issueDelete} /> )} {issue && isArchived && ( <DeleteArchivedIssueModal data={issue} isOpen={isDeleteIssueModalOpen} handleClose={() => toggleDeleteIssueModal(false)} onSubmit={issueDelete} /> )} <div className="w-full truncate !text-base"> {issueId && ( <div ref={issuePeekOverviewRef} className={`fixed z-20 flex flex-col overflow-hidden rounded border border-custom-border-200 bg-custom-background-100 transition-all duration-300 ${peekMode === "side-peek" ? `bottom-0 right-0 top-0 w-full md:w-[50%]` : ``} ${peekMode === "modal" ? `left-[50%] top-[50%] h-5/6 w-5/6 -translate-x-[50%] -translate-y-[50%]` : ``} ${peekMode === "full-screen" ? `bottom-0 left-0 right-0 top-0 m-4` : ``} `} style={{ boxShadow: "0px 4px 8px 0px rgba(0, 0, 0, 0.12), 0px 6px 12px 0px rgba(16, 24, 40, 0.12), 0px 1px 16px 0px rgba(16, 24, 40, 0.12)", }} > {/* header */} <div className={`relative flex items-center justify-between p-4 ${ currentMode?.key === "full-screen" ? "border-b border-custom-border-200" : "" }`} > <div className="flex items-center gap-4"> <button onClick={removeRoutePeekId}> <MoveRight className="h-4 w-4 text-custom-text-400 hover:text-custom-text-200" /> </button> <button onClick={redirectToIssueDetail}> <MoveDiagonal className="h-4 w-4 text-custom-text-400 hover:text-custom-text-200" /> </button> {currentMode && ( <div className="flex flex-shrink-0 items-center gap-2"> <CustomSelect value={currentMode} onChange={(val: any) => setPeekMode(val)} customButton={ <button type="button" className=""> <currentMode.icon className="h-4 w-4 text-custom-text-400 hover:text-custom-text-200" /> </button> } > {PEEK_OPTIONS.map((mode) => ( <CustomSelect.Option key={mode.key} value={mode.key}> <div className={`flex items-center gap-1.5 ${ currentMode.key === mode.key ? "text-custom-text-200" : "text-custom-text-400 hover:text-custom-text-200" }`} > <mode.icon className="-my-1 h-4 w-4 flex-shrink-0" /> {mode.title} </div> </CustomSelect.Option> ))} </CustomSelect> </div> )} </div> <div className="flex items-center gap-x-4"> <IssueUpdateStatus isSubmitting={isSubmitting} /> <div className="flex items-center gap-4"> {issue?.created_by !== currentUser?.id && !issue?.assignee_ids.includes(currentUser?.id ?? "") && !issue?.archived_at && ( <Button size="sm" prependIcon={<Bell className="h-3 w-3" />} variant="outline-primary" className="hover:!bg-custom-primary-100/20" onClick={() => issueSubscription && issueSubscription.subscribed ? issueSubscriptionRemove() : issueSubscriptionCreate() } > {issueSubscription && issueSubscription.subscribed ? "Unsubscribe" : "Subscribe"} </Button> )} <button onClick={handleCopyText}> <Link2 className="h-4 w-4 -rotate-45 text-custom-text-300 hover:text-custom-text-200" /> </button> {!disableUserActions && ( <button onClick={() => toggleDeleteIssueModal(true)}> <Trash2 className="h-4 w-4 text-custom-text-300 hover:text-custom-text-200" /> </button> )} </div> </div> </div> {/* content */} <div className="relative h-full w-full overflow-hidden overflow-y-auto"> {isLoading && !issue ? ( <div className="flex h-full w-full items-center justify-center"> <Spinner /> </div> ) : ( issue && ( <> {["side-peek", "modal"].includes(peekMode) ? ( <div className="relative flex flex-col gap-3 px-8 py-5"> {isArchived && ( <div className="absolute left-0 top-0 z-[9] flex h-full min-h-full w-full items-center justify-center bg-custom-background-100 opacity-60" /> )} <PeekOverviewIssueDetails setIsSubmitting={(value) => setIsSubmitting(value)} isSubmitting={isSubmitting} workspaceSlug={workspaceSlug} issue={issue} issueUpdate={issueUpdate} issueReactions={issueReactions} user={currentUser} issueReactionCreate={issueReactionCreate} issueReactionRemove={issueReactionRemove} /> <PeekOverviewProperties issue={issue} issueUpdate={issueUpdate} disableUserActions={disableUserActions} issueOperations={issueOperations} /> <IssueActivity workspaceSlug={workspaceSlug} projectId={projectId} issueId={issueId} user={currentUser} issueActivity={issueActivity} issueCommentCreate={issueCommentCreate} issueCommentUpdate={issueCommentUpdate} issueCommentRemove={issueCommentRemove} issueCommentReactionCreate={issueCommentReactionCreate} issueCommentReactionRemove={issueCommentReactionRemove} showCommentAccessSpecifier={showCommentAccessSpecifier} /> </div> ) : ( <div className={`flex h-full w-full overflow-auto ${isArchived ? "opacity-60" : ""}`}> <div className="relative h-full w-full space-y-6 overflow-auto p-4 py-5"> <div className={isArchived ? "pointer-events-none" : ""}> <PeekOverviewIssueDetails setIsSubmitting={(value) => setIsSubmitting(value)} isSubmitting={isSubmitting} workspaceSlug={workspaceSlug} issue={issue} issueReactions={issueReactions} issueUpdate={issueUpdate} user={currentUser} issueReactionCreate={issueReactionCreate} issueReactionRemove={issueReactionRemove} /> <IssueActivity workspaceSlug={workspaceSlug} projectId={projectId} issueId={issueId} user={currentUser} issueActivity={issueActivity} issueCommentCreate={issueCommentCreate} issueCommentUpdate={issueCommentUpdate} issueCommentRemove={issueCommentRemove} issueCommentReactionCreate={issueCommentReactionCreate} issueCommentReactionRemove={issueCommentReactionRemove} showCommentAccessSpecifier={showCommentAccessSpecifier} /> </div> </div> <div className={`h-full !w-[400px] flex-shrink-0 border-l border-custom-border-200 p-4 py-5 ${ isArchived ? "pointer-events-none" : "" }`} > <PeekOverviewProperties issue={issue} issueUpdate={issueUpdate} disableUserActions={disableUserActions} issueOperations={issueOperations} /> </div> </div> )} </> ) )} </div> </div> )} </div> </> ); });