2024-01-24 13:51:59 +00:00
|
|
|
import React from "react";
|
|
|
|
import Link from "next/link";
|
2024-01-10 14:39:45 +00:00
|
|
|
import { observer } from "mobx-react-lite";
|
2024-01-24 13:51:59 +00:00
|
|
|
import { Pencil, X } from "lucide-react";
|
2024-01-10 14:39:45 +00:00
|
|
|
// hooks
|
|
|
|
import { useIssueDetail, useProject } from "hooks/store";
|
|
|
|
// components
|
|
|
|
import { ParentIssuesListModal } from "components/issues";
|
2024-01-24 13:51:59 +00:00
|
|
|
// ui
|
|
|
|
import { Tooltip } from "@plane/ui";
|
|
|
|
// helpers
|
|
|
|
import { cn } from "helpers/common.helper";
|
|
|
|
// types
|
2024-01-10 14:39:45 +00:00
|
|
|
import { TIssueOperations } from "./root";
|
|
|
|
|
|
|
|
type TIssueParentSelect = {
|
2024-01-24 13:51:59 +00:00
|
|
|
className?: string;
|
|
|
|
disabled?: boolean;
|
2024-01-10 14:39:45 +00:00
|
|
|
issueId: string;
|
|
|
|
issueOperations: TIssueOperations;
|
2024-01-24 13:51:59 +00:00
|
|
|
projectId: string;
|
|
|
|
workspaceSlug: string;
|
2024-01-10 14:39:45 +00:00
|
|
|
};
|
|
|
|
|
2024-01-24 13:51:59 +00:00
|
|
|
export const IssueParentSelect: React.FC<TIssueParentSelect> = observer((props) => {
|
|
|
|
const { className = "", disabled = false, issueId, issueOperations, projectId, workspaceSlug } = props;
|
|
|
|
// store hooks
|
|
|
|
const { getProjectById } = useProject();
|
|
|
|
const {
|
|
|
|
issue: { getIssueById },
|
|
|
|
} = useIssueDetail();
|
|
|
|
const { isParentIssueModalOpen, toggleParentIssueModal } = useIssueDetail();
|
|
|
|
// derived values
|
|
|
|
const issue = getIssueById(issueId);
|
|
|
|
const parentIssue = issue?.parent_id ? getIssueById(issue.parent_id) : undefined;
|
|
|
|
const parentIssueProjectDetails =
|
|
|
|
parentIssue && parentIssue.project_id ? getProjectById(parentIssue.project_id) : undefined;
|
2024-01-10 14:39:45 +00:00
|
|
|
|
2024-01-24 13:51:59 +00:00
|
|
|
const handleParentIssue = async (_issueId: string | null = null) => {
|
|
|
|
try {
|
|
|
|
await issueOperations.update(workspaceSlug, projectId, issueId, { parent_id: _issueId });
|
|
|
|
await issueOperations.fetch(workspaceSlug, projectId, issueId);
|
|
|
|
toggleParentIssueModal(false);
|
|
|
|
} catch (error) {
|
|
|
|
console.error("something went wrong while fetching the issue");
|
|
|
|
}
|
|
|
|
};
|
2024-01-10 14:39:45 +00:00
|
|
|
|
2024-01-24 13:51:59 +00:00
|
|
|
if (!issue) return <></>;
|
2024-01-10 14:39:45 +00:00
|
|
|
|
2024-01-24 13:51:59 +00:00
|
|
|
return (
|
|
|
|
<>
|
|
|
|
<ParentIssuesListModal
|
|
|
|
projectId={projectId}
|
|
|
|
issueId={issueId}
|
|
|
|
isOpen={isParentIssueModalOpen}
|
|
|
|
handleClose={() => toggleParentIssueModal(false)}
|
|
|
|
onChange={(issue: any) => handleParentIssue(issue?.id)}
|
|
|
|
/>
|
|
|
|
<button
|
|
|
|
type="button"
|
|
|
|
className={cn(
|
|
|
|
"group flex items-center justify-between gap-2 px-2 py-0.5 rounded outline-none",
|
|
|
|
{
|
|
|
|
"cursor-not-allowed": disabled,
|
|
|
|
"hover:bg-custom-background-80": !disabled,
|
|
|
|
},
|
|
|
|
className
|
|
|
|
)}
|
|
|
|
onClick={() => toggleParentIssueModal(true)}
|
|
|
|
disabled={disabled}
|
|
|
|
>
|
|
|
|
{issue.parent_id && parentIssue ? (
|
2024-01-29 14:23:46 +00:00
|
|
|
<div className="flex items-center gap-1 bg-success-component-surface-light text-success-text-medium rounded px-1.5 py-1">
|
2024-01-24 13:51:59 +00:00
|
|
|
<Link
|
|
|
|
href={`/${workspaceSlug}/projects/${projectId}/issues/${parentIssue?.id}`}
|
|
|
|
className="text-xs font-medium"
|
|
|
|
>
|
|
|
|
{parentIssueProjectDetails?.identifier}-{parentIssue.sequence_id}
|
|
|
|
</Link>
|
2024-01-10 14:39:45 +00:00
|
|
|
|
2024-01-24 13:51:59 +00:00
|
|
|
{!disabled && (
|
|
|
|
<Tooltip tooltipContent="Remove">
|
|
|
|
<span
|
|
|
|
onClick={(e) => {
|
|
|
|
e.preventDefault();
|
|
|
|
e.stopPropagation();
|
|
|
|
handleParentIssue(null);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<X className="h-2.5 w-2.5 text-custom-text-300 hover:text-red-500" />
|
|
|
|
</span>
|
|
|
|
</Tooltip>
|
2024-01-10 14:39:45 +00:00
|
|
|
)}
|
|
|
|
</div>
|
2024-01-24 13:51:59 +00:00
|
|
|
) : (
|
|
|
|
<span className="text-sm text-custom-text-400">Add parent issue</span>
|
|
|
|
)}
|
|
|
|
{!disabled && <Pencil className="h-4 w-4 flex-shrink-0 hidden group-hover:inline" />}
|
|
|
|
</button>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
});
|