mirror of
https://github.com/makeplane/plane
synced 2024-06-14 14:31:34 +00:00
223 lines
7.6 KiB
TypeScript
223 lines
7.6 KiB
TypeScript
import { FC } from "react";
|
|
import { observer } from "mobx-react";
|
|
import Link from "next/link";
|
|
import { useRouter } from "next/router";
|
|
import { ArchiveRestoreIcon, Link2, MoveDiagonal, MoveRight, Trash2 } from "lucide-react";
|
|
// ui
|
|
import {
|
|
ArchiveIcon,
|
|
CenterPanelIcon,
|
|
CustomSelect,
|
|
FullScreenPanelIcon,
|
|
SidePanelIcon,
|
|
TOAST_TYPE,
|
|
Tooltip,
|
|
setToast,
|
|
} from "@plane/ui";
|
|
// components
|
|
import { IssueSubscription, IssueUpdateStatus } from "@/components/issues";
|
|
// constants
|
|
import { ISSUE_OPENED } from "@/constants/event-tracker";
|
|
import { STATE_GROUPS } from "@/constants/state";
|
|
// helpers
|
|
import { cn } from "@/helpers/common.helper";
|
|
import { getElementIdFromPath } from "@/helpers/event-tracker.helper";
|
|
import { copyUrlToClipboard } from "@/helpers/string.helper";
|
|
// store hooks
|
|
import { useIssueDetail, useProjectState, useUser, useEventTracker } from "@/hooks/store";
|
|
// hooks
|
|
import { usePlatformOS } from "@/hooks/use-platform-os";
|
|
export 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 type PeekOverviewHeaderProps = {
|
|
peekMode: TPeekModes;
|
|
setPeekMode: (value: TPeekModes) => void;
|
|
removeRoutePeekId: () => void;
|
|
workspaceSlug: string;
|
|
projectId: string;
|
|
issueId: string;
|
|
isArchived: boolean;
|
|
disabled: boolean;
|
|
toggleDeleteIssueModal: (issueId: string | null) => void;
|
|
toggleArchiveIssueModal: (issueId: string | null) => void;
|
|
handleRestoreIssue: () => void;
|
|
isSubmitting: "submitting" | "submitted" | "saved";
|
|
};
|
|
|
|
export const IssuePeekOverviewHeader: FC<PeekOverviewHeaderProps> = observer((props) => {
|
|
const {
|
|
peekMode,
|
|
setPeekMode,
|
|
workspaceSlug,
|
|
projectId,
|
|
issueId,
|
|
isArchived,
|
|
disabled,
|
|
removeRoutePeekId,
|
|
toggleDeleteIssueModal,
|
|
toggleArchiveIssueModal,
|
|
handleRestoreIssue,
|
|
isSubmitting,
|
|
} = props;
|
|
// router
|
|
const router = useRouter();
|
|
// store hooks
|
|
const { data: currentUser } = useUser();
|
|
const {
|
|
issue: { getIssueById },
|
|
} = useIssueDetail();
|
|
const { getStateById } = useProjectState();
|
|
const { isMobile } = usePlatformOS();
|
|
const { captureEvent } = useEventTracker();
|
|
// derived values
|
|
const issueDetails = getIssueById(issueId);
|
|
const stateDetails = issueDetails ? getStateById(issueDetails?.state_id) : undefined;
|
|
const currentMode = PEEK_OPTIONS.find((m) => m.key === peekMode);
|
|
|
|
const issueLink = `${workspaceSlug}/projects/${projectId}/${isArchived ? "archives/" : ""}issues/${issueId}`;
|
|
|
|
const handleCopyText = (e: React.MouseEvent<HTMLButtonElement>) => {
|
|
e.stopPropagation();
|
|
e.preventDefault();
|
|
copyUrlToClipboard(issueLink).then(() => {
|
|
setToast({
|
|
type: TOAST_TYPE.SUCCESS,
|
|
title: "Link Copied!",
|
|
message: "Issue link copied to clipboard.",
|
|
});
|
|
});
|
|
};
|
|
// auth
|
|
const isArchivingAllowed = !isArchived && !disabled;
|
|
const isInArchivableGroup =
|
|
!!stateDetails && [STATE_GROUPS.completed.key, STATE_GROUPS.cancelled.key].includes(stateDetails?.group);
|
|
const isRestoringAllowed = isArchived && !disabled;
|
|
|
|
return (
|
|
<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">
|
|
<Tooltip tooltipContent="Close the peek view" isMobile={isMobile}>
|
|
<button onClick={removeRoutePeekId}>
|
|
<MoveRight className="h-4 w-4 text-custom-text-300 hover:text-custom-text-200" />
|
|
</button>
|
|
</Tooltip>
|
|
|
|
<Tooltip tooltipContent="Open issue in full screen" isMobile={isMobile}>
|
|
<Link
|
|
href={`/${issueLink}`}
|
|
onClick={() => {
|
|
removeRoutePeekId();
|
|
captureEvent(ISSUE_OPENED, {
|
|
element: "peek",
|
|
elementId: getElementIdFromPath(router.asPath),
|
|
mode: "detail",
|
|
});
|
|
}}
|
|
>
|
|
<MoveDiagonal className="h-4 w-4 text-custom-text-300 hover:text-custom-text-200" />
|
|
</Link>
|
|
</Tooltip>
|
|
{currentMode && (
|
|
<div className="flex flex-shrink-0 items-center gap-2">
|
|
<CustomSelect
|
|
value={currentMode}
|
|
onChange={(val: any) => setPeekMode(val)}
|
|
customButton={
|
|
<Tooltip tooltipContent="Toggle peek view layout" isMobile={isMobile}>
|
|
<button type="button" className="">
|
|
<currentMode.icon className="h-4 w-4 text-custom-text-300 hover:text-custom-text-200" />
|
|
</button>
|
|
</Tooltip>
|
|
}
|
|
>
|
|
{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">
|
|
{currentUser && !isArchived && (
|
|
<IssueSubscription workspaceSlug={workspaceSlug} projectId={projectId} issueId={issueId} />
|
|
)}
|
|
<Tooltip tooltipContent="Copy link" isMobile={isMobile}>
|
|
<button type="button" onClick={handleCopyText}>
|
|
<Link2 className="h-4 w-4 -rotate-45 text-custom-text-300 hover:text-custom-text-200" />
|
|
</button>
|
|
</Tooltip>
|
|
{isArchivingAllowed && (
|
|
<Tooltip
|
|
isMobile={isMobile}
|
|
tooltipContent={isInArchivableGroup ? "Archive" : "Only completed or canceled issues can be archived"}
|
|
>
|
|
<button
|
|
type="button"
|
|
className={cn("text-custom-text-300", {
|
|
"hover:text-custom-text-200": isInArchivableGroup,
|
|
"cursor-not-allowed text-custom-text-400": !isInArchivableGroup,
|
|
})}
|
|
onClick={() => {
|
|
if (!isInArchivableGroup) return;
|
|
toggleArchiveIssueModal(issueId);
|
|
}}
|
|
>
|
|
<ArchiveIcon className="h-4 w-4" />
|
|
</button>
|
|
</Tooltip>
|
|
)}
|
|
{isRestoringAllowed && (
|
|
<Tooltip tooltipContent="Restore" isMobile={isMobile}>
|
|
<button type="button" onClick={handleRestoreIssue}>
|
|
<ArchiveRestoreIcon className="h-4 w-4 text-custom-text-300 hover:text-custom-text-200" />
|
|
</button>
|
|
</Tooltip>
|
|
)}
|
|
{!disabled && (
|
|
<Tooltip tooltipContent="Delete" isMobile={isMobile}>
|
|
<button type="button" onClick={() => toggleDeleteIssueModal(issueId)}>
|
|
<Trash2 className="h-4 w-4 text-custom-text-300 hover:text-custom-text-200" />
|
|
</button>
|
|
</Tooltip>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
});
|