forked from github/plane
442c83eea2
* style: spreadsheet columns * fix: build errors
537 lines
23 KiB
TypeScript
537 lines
23 KiB
TypeScript
import React, { useState } from "react";
|
|
import { useRouter } from "next/router";
|
|
import { mutate } from "swr";
|
|
import { Controller, useFormContext } from "react-hook-form";
|
|
// hooks
|
|
import useUser from "hooks/use-user";
|
|
import { useProjectMyMembership } from "contexts/project-member.context";
|
|
import useEstimateOption from "hooks/use-estimate-option";
|
|
// fetch keys
|
|
import { ISSUE_DETAILS, PROJECT_ISSUES_ACTIVITY } from "constants/fetch-keys";
|
|
// icons
|
|
import { PlayIcon, User, X, CalendarDays, LayoutGrid, Users, CopyPlus } from "lucide-react";
|
|
//services
|
|
import { IssueService } from "services/issue";
|
|
// components
|
|
import { Button, DiceIcon, BlockedIcon, BlockerIcon, RelatedIcon, ContrastIcon } from "@plane/ui";
|
|
import {
|
|
Label,
|
|
StateSelect,
|
|
PrioritySelect,
|
|
AssigneeSelect,
|
|
EstimateSelect,
|
|
ParentSelect,
|
|
BlockerSelect,
|
|
BlockedBySelect,
|
|
RelatesSelect,
|
|
DuplicateSelect,
|
|
ModuleSelect,
|
|
CycleSelect,
|
|
DateSelector,
|
|
} from "components/web-view";
|
|
// types
|
|
import type { IIssue } from "types";
|
|
|
|
type Props = {
|
|
submitChanges: (data: Partial<IIssue>) => Promise<void>;
|
|
};
|
|
|
|
const issueService = new IssueService();
|
|
|
|
export const IssuePropertiesDetail: React.FC<Props> = (props) => {
|
|
const { submitChanges } = props;
|
|
|
|
const { watch, control } = useFormContext<IIssue>();
|
|
|
|
const blockerIssues = watch("issue_relations")?.filter((i) => i.relation_type === "blocked_by") || [];
|
|
|
|
const blockedByIssues = watch("related_issues")?.filter((i) => i.relation_type === "blocked_by");
|
|
|
|
const relatedToIssueRelation = [
|
|
...(watch("related_issues")?.filter((i) => i.relation_type === "relates_to") ?? []),
|
|
...(watch("issue_relations") ?? [])
|
|
?.filter((i) => i.relation_type === "relates_to")
|
|
.map((i) => ({
|
|
...i,
|
|
issue_detail: i.issue_detail,
|
|
related_issue: i.issue_detail?.id,
|
|
})),
|
|
];
|
|
|
|
const duplicateIssuesRelation = [
|
|
...(watch("related_issues")?.filter((i) => i.relation_type === "duplicate") ?? []),
|
|
...(watch("issue_relations") ?? [])
|
|
?.filter((i) => i.relation_type === "duplicate")
|
|
.map((i) => ({
|
|
...i,
|
|
issue_detail: i.issue_detail,
|
|
related_issue: i.issue_detail?.id,
|
|
})),
|
|
];
|
|
|
|
const startDate = watch("start_date");
|
|
|
|
const router = useRouter();
|
|
const { workspaceSlug, projectId, issueId } = router.query;
|
|
|
|
const isArchive = Boolean(router.query.archive);
|
|
|
|
const { memberRole } = useProjectMyMembership();
|
|
|
|
const { user } = useUser();
|
|
|
|
const [isViewAllOpen, setIsViewAllOpen] = useState(false);
|
|
|
|
const { isEstimateActive } = useEstimateOption();
|
|
|
|
return (
|
|
<div className="space-y-[6px]">
|
|
<Label>Details</Label>
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2 flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<LayoutGrid className="h-4 w-4 flex-shrink-0 text-custom-text-400" />
|
|
<span className="text-sm text-custom-text-400">State</span>
|
|
</div>
|
|
<div>
|
|
<Controller
|
|
control={control}
|
|
name="state"
|
|
render={({ field: { value } }) => (
|
|
<StateSelect
|
|
value={value}
|
|
disabled={isArchive || memberRole.isGuest || memberRole.isViewer}
|
|
onChange={(val: string) => submitChanges({ state: val })}
|
|
/>
|
|
)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2 flex justify-between items-center">
|
|
<div className="flex items-center gap-1 text-custom-text-400">
|
|
<svg width="18" height="18" viewBox="0 0 18 18" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
<path
|
|
d="M13.5862 14.5239C13.3459 14.5239 13.1416 14.4398 12.9733 14.2715C12.805 14.1032 12.7209 13.8989 12.7209 13.6585V3.76429C12.7209 3.52391 12.805 3.31958 12.9733 3.15132C13.1416 2.98306 13.3459 2.89893 13.5862 2.89893C13.8266 2.89893 14.031 2.98306 14.1992 3.15132C14.3675 3.31958 14.4516 3.52391 14.4516 3.76429V13.6585C14.4516 13.8989 14.3675 14.1032 14.1992 14.2715C14.031 14.4398 13.8266 14.5239 13.5862 14.5239ZM5.1629 14.5239C5.04676 14.5239 4.93557 14.5018 4.82932 14.4576C4.72308 14.4133 4.63006 14.3513 4.55025 14.2715C4.47045 14.1917 4.40843 14.0986 4.36419 13.9922C4.31996 13.8858 4.29785 13.7746 4.29785 13.6585V11.2643C4.29785 11.0239 4.38198 10.8196 4.55025 10.6513C4.71851 10.4831 4.92283 10.3989 5.16322 10.3989C5.40359 10.3989 5.60791 10.4831 5.77618 10.6513C5.94445 10.8196 6.02859 11.0239 6.02859 11.2643V13.6585C6.02859 13.7746 6.00647 13.8858 5.96223 13.9922C5.91801 14.0986 5.85599 14.1917 5.77618 14.2715C5.69638 14.3513 5.60325 14.4133 5.49678 14.4576C5.39033 14.5018 5.27904 14.5239 5.1629 14.5239ZM9.37473 14.5239C9.13436 14.5239 8.93003 14.4398 8.76176 14.2715C8.59349 14.1032 8.50936 13.8989 8.50936 13.6585V7.5143C8.50936 7.27391 8.59349 7.06958 8.76176 6.90132C8.93003 6.73306 9.13436 6.64893 9.37473 6.64893C9.61511 6.64893 9.81943 6.73306 9.98771 6.90132C10.156 7.06958 10.2401 7.27391 10.2401 7.5143V13.6585C10.2401 13.8989 10.156 14.1032 9.98771 14.2715C9.81943 14.4398 9.61511 14.5239 9.37473 14.5239Z"
|
|
fill="currentColor"
|
|
/>
|
|
</svg>
|
|
|
|
<span className="text-sm text-custom-text-400">Priority</span>
|
|
</div>
|
|
<div>
|
|
<Controller
|
|
control={control}
|
|
name="priority"
|
|
render={({ field: { value } }) => (
|
|
<PrioritySelect
|
|
value={value}
|
|
disabled={isArchive || memberRole.isGuest || memberRole.isViewer}
|
|
onChange={(val) => submitChanges({ priority: val })}
|
|
/>
|
|
)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2 flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<Users className="h-4 w-4 flex-shrink-0 text-custom-text-400" />
|
|
<span className="text-sm text-custom-text-400">Assignee</span>
|
|
</div>
|
|
<div>
|
|
<Controller
|
|
control={control}
|
|
name="assignees"
|
|
render={({ field: { value } }) => (
|
|
<AssigneeSelect
|
|
value={value}
|
|
disabled={isArchive || memberRole.isGuest || memberRole.isViewer}
|
|
onChange={(val: string) => {
|
|
const assignees = value?.includes(val) ? value?.filter((i) => i !== val) : [...(value ?? []), val];
|
|
|
|
submitChanges({ assignees: assignees });
|
|
}}
|
|
/>
|
|
)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
{isViewAllOpen && (
|
|
<>
|
|
{isEstimateActive && (
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2 flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<PlayIcon className="h-4 w-4 flex-shrink-0 -rotate-90 text-custom-text-400" />
|
|
<span className="text-sm text-custom-text-400">Estimate</span>
|
|
</div>
|
|
<div>
|
|
<Controller
|
|
control={control}
|
|
name="estimate_point"
|
|
render={({ field: { value } }) => (
|
|
<EstimateSelect
|
|
value={value}
|
|
disabled={isArchive || memberRole.isGuest || memberRole.isViewer}
|
|
onChange={(val) => submitChanges({ estimate_point: val })}
|
|
/>
|
|
)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)}
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2 flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<User className="h-4 w-4 flex-shrink-0 text-custom-text-400" />
|
|
<span className="text-sm text-custom-text-400">Parent</span>
|
|
</div>
|
|
<div>
|
|
<Controller
|
|
control={control}
|
|
name="parent"
|
|
render={({ field: { value } }) => (
|
|
<ParentSelect
|
|
value={value}
|
|
disabled={isArchive || memberRole.isGuest || memberRole.isViewer}
|
|
onChange={(val) => submitChanges({ parent: val })}
|
|
/>
|
|
)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* blocker to / blocking */}
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2">
|
|
<div className="flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<BlockerIcon height={16} width={16} />
|
|
<span className="text-sm text-custom-text-400">Blocking</span>
|
|
</div>
|
|
<div>
|
|
<BlockerSelect disabled={isArchive || memberRole.isGuest || memberRole.isViewer} />
|
|
</div>
|
|
</div>
|
|
{blockerIssues &&
|
|
blockerIssues.map((relation) => (
|
|
<div
|
|
key={relation.issue_detail?.id}
|
|
className="group inline-flex mr-1 cursor-pointer items-center gap-1 rounded-2xl border border-custom-border-200 px-1.5 py-0.5 text-xs text-yellow-500 duration-300 hover:border-yellow-500/20 hover:bg-yellow-500/20"
|
|
>
|
|
<span
|
|
onClick={() =>
|
|
console.log(
|
|
"issue",
|
|
JSON.stringify({
|
|
issue_id: relation.issue_detail?.id,
|
|
project_id: relation.issue_detail?.project_detail.id,
|
|
issue_identifier: `${relation.issue_detail?.project_detail.identifier}-${relation.issue_detail?.sequence_id}`,
|
|
})
|
|
)
|
|
}
|
|
className="flex items-center gap-1"
|
|
>
|
|
<BlockerIcon height={10} width={10} />
|
|
{`${relation.issue_detail?.project_detail.identifier}-${relation.issue_detail?.sequence_id}`}
|
|
</span>
|
|
{!isArchive && (
|
|
<button
|
|
type="button"
|
|
className="duration-300"
|
|
onClick={() => {
|
|
if (memberRole.isGuest || memberRole.isViewer || !user) return;
|
|
|
|
issueService
|
|
.deleteIssueRelation(
|
|
workspaceSlug as string,
|
|
projectId as string,
|
|
relation.issue,
|
|
relation.id,
|
|
user
|
|
)
|
|
.then(() => {
|
|
mutate(ISSUE_DETAILS(issueId as string));
|
|
mutate(PROJECT_ISSUES_ACTIVITY(issueId as string));
|
|
});
|
|
}}
|
|
>
|
|
<X className="h-2 w-2" />
|
|
</button>
|
|
)}
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
|
|
{/* blocked by */}
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2">
|
|
<div className="flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<BlockedIcon height={16} width={16} />
|
|
<span className="text-sm text-custom-text-400">Blocked by</span>
|
|
</div>
|
|
<div>
|
|
<BlockedBySelect disabled={isArchive || memberRole.isGuest || memberRole.isViewer} />
|
|
</div>
|
|
</div>
|
|
{blockedByIssues &&
|
|
blockedByIssues.map((relation) => (
|
|
<div
|
|
key={relation.issue_detail?.id}
|
|
className="group inline-flex mr-1 cursor-pointer items-center gap-1 rounded-2xl border border-custom-border-200 px-1.5 py-0.5 text-xs text-red-500 duration-300 hover:border-red-500/20 hover:bg-red-500/20"
|
|
>
|
|
<span
|
|
onClick={() =>
|
|
console.log(
|
|
"issue",
|
|
JSON.stringify({
|
|
issue_id: relation.issue_detail?.id,
|
|
project_id: relation.issue_detail?.project_detail.id,
|
|
issue_identifier: `${relation.issue_detail?.project_detail.identifier}-${relation.issue_detail?.sequence_id}`,
|
|
})
|
|
)
|
|
}
|
|
className="flex items-center gap-1"
|
|
>
|
|
<BlockedIcon height={10} width={10} />
|
|
{`${relation?.issue_detail?.project_detail?.identifier}-${relation?.issue_detail?.sequence_id}`}
|
|
</span>
|
|
{!isArchive && !(memberRole.isGuest || memberRole.isViewer) && (
|
|
<button
|
|
type="button"
|
|
className="duration-300"
|
|
onClick={() => {
|
|
if (memberRole.isGuest || memberRole.isViewer || !user) return;
|
|
|
|
issueService
|
|
.deleteIssueRelation(
|
|
workspaceSlug as string,
|
|
projectId as string,
|
|
issueId as string,
|
|
relation.id,
|
|
user
|
|
)
|
|
.then(() => {
|
|
mutate(ISSUE_DETAILS(issueId as string));
|
|
mutate(PROJECT_ISSUES_ACTIVITY(issueId as string));
|
|
});
|
|
}}
|
|
>
|
|
<X className="h-2 w-2" />
|
|
</button>
|
|
)}
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
|
|
{/* duplicate */}
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2">
|
|
<div className="flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<CopyPlus height={16} width={16} className="text-custom-text-400" />
|
|
<span className="text-sm text-custom-text-400">Duplicate</span>
|
|
</div>
|
|
<div>
|
|
<DuplicateSelect disabled={isArchive || memberRole.isGuest || memberRole.isViewer} />
|
|
</div>
|
|
</div>
|
|
{duplicateIssuesRelation &&
|
|
duplicateIssuesRelation.map((relation) => (
|
|
<div
|
|
key={relation.issue_detail?.id}
|
|
className="group inline-flex mr-1 cursor-pointer items-center gap-1 rounded-2xl border border-custom-border-200 px-1.5 py-0.5 text-xs text-red-500 duration-300 hover:border-red-500/20 hover:bg-red-500/20"
|
|
>
|
|
<span
|
|
onClick={() =>
|
|
console.log(
|
|
"issue",
|
|
JSON.stringify({
|
|
issue_id: relation.issue_detail?.id,
|
|
project_id: relation.issue_detail?.project_detail.id,
|
|
issue_identifier: `${relation.issue_detail?.project_detail.identifier}-${relation.issue_detail?.sequence_id}`,
|
|
})
|
|
)
|
|
}
|
|
className="flex items-center gap-1"
|
|
>
|
|
<CopyPlus height={10} width={10} />
|
|
{`${relation?.issue_detail?.project_detail?.identifier}-${relation?.issue_detail?.sequence_id}`}
|
|
</span>
|
|
{!isArchive && !(memberRole.isGuest || memberRole.isViewer) && (
|
|
<button
|
|
type="button"
|
|
className="duration-300"
|
|
onClick={() => {
|
|
if (memberRole.isGuest || memberRole.isViewer || !user) return;
|
|
|
|
issueService
|
|
.deleteIssueRelation(
|
|
workspaceSlug as string,
|
|
projectId as string,
|
|
issueId as string,
|
|
relation.id,
|
|
user
|
|
)
|
|
.then(() => {
|
|
mutate(ISSUE_DETAILS(issueId as string));
|
|
mutate(PROJECT_ISSUES_ACTIVITY(issueId as string));
|
|
});
|
|
}}
|
|
>
|
|
<X className="h-2 w-2" />
|
|
</button>
|
|
)}
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
|
|
{/* relates to */}
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2">
|
|
<div className="flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<RelatedIcon height={16} width={16} color="rgb(var(--color-text-400))" />
|
|
<span className="text-sm text-custom-text-400">Relates To</span>
|
|
</div>
|
|
<div>
|
|
<RelatesSelect disabled={isArchive || memberRole.isGuest || memberRole.isViewer} />
|
|
</div>
|
|
</div>
|
|
{relatedToIssueRelation &&
|
|
relatedToIssueRelation.map((relation) => (
|
|
<div
|
|
key={relation.issue_detail?.id}
|
|
className="group inline-flex mr-1 cursor-pointer items-center gap-1 rounded-2xl border border-custom-border-200 px-1.5 py-0.5 text-xs text-red-500 duration-300 hover:border-red-500/20 hover:bg-red-500/20"
|
|
>
|
|
<span
|
|
onClick={() =>
|
|
console.log(
|
|
"issue",
|
|
JSON.stringify({
|
|
issue_id: relation.issue_detail?.id,
|
|
project_id: relation.issue_detail?.project_detail.id,
|
|
issue_identifier: `${relation.issue_detail?.project_detail.identifier}-${relation.issue_detail?.sequence_id}`,
|
|
})
|
|
)
|
|
}
|
|
className="flex items-center gap-1"
|
|
>
|
|
<RelatedIcon height={10} width={10} />
|
|
{`${relation?.issue_detail?.project_detail?.identifier}-${relation?.issue_detail?.sequence_id}`}
|
|
</span>
|
|
{!isArchive && !(memberRole.isGuest || memberRole.isViewer) && (
|
|
<button
|
|
type="button"
|
|
className="duration-300"
|
|
onClick={() => {
|
|
if (memberRole.isGuest || memberRole.isViewer || !user) return;
|
|
|
|
issueService
|
|
.deleteIssueRelation(
|
|
workspaceSlug as string,
|
|
projectId as string,
|
|
issueId as string,
|
|
relation.id,
|
|
user
|
|
)
|
|
.then(() => {
|
|
mutate(ISSUE_DETAILS(issueId as string));
|
|
mutate(PROJECT_ISSUES_ACTIVITY(issueId as string));
|
|
});
|
|
}}
|
|
>
|
|
<X className="h-2 w-2" />
|
|
</button>
|
|
)}
|
|
</div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2">
|
|
<div className="flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<CalendarDays className="w-4 h-4 text-custom-text-400" />
|
|
<span className="text-sm text-custom-text-400">Due date</span>
|
|
</div>
|
|
<div>
|
|
<Controller
|
|
control={control}
|
|
name="target_date"
|
|
render={({ field: { value } }) => (
|
|
<DateSelector
|
|
placeholderText="Due date"
|
|
value={value ?? undefined}
|
|
wrapperClassName="w-full"
|
|
disabled={isArchive || memberRole.isGuest || memberRole.isViewer}
|
|
onChange={(val) =>
|
|
submitChanges({
|
|
target_date: val,
|
|
})
|
|
}
|
|
className="border-transparent !shadow-none !w-[6.75rem]"
|
|
minDate={startDate ? new Date(startDate) : undefined}
|
|
/>
|
|
)}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2">
|
|
<div className="flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<DiceIcon className="h-4 w-4 flex-shrink-0 text-custom-text-400" />
|
|
<span className="text-sm text-custom-text-400">Module</span>
|
|
</div>
|
|
<div>
|
|
<ModuleSelect
|
|
value={watch("issue_module")}
|
|
disabled={isArchive || memberRole.isGuest || memberRole.isViewer}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div>
|
|
<div className="border border-custom-border-200 rounded-[4px] p-2">
|
|
<div className="flex justify-between items-center">
|
|
<div className="flex items-center gap-1">
|
|
<ContrastIcon color="rgba(var(--color-text-400))" className="h-4 w-4 flex-shrink-0" />
|
|
<span className="text-sm text-custom-text-400">Cycle</span>
|
|
</div>
|
|
<div>
|
|
<CycleSelect
|
|
value={watch("issue_cycle")}
|
|
disabled={isArchive || memberRole.isGuest || memberRole.isViewer}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</>
|
|
)}
|
|
<div>
|
|
<Button type="button" onClick={() => setIsViewAllOpen((prev) => !prev)}>
|
|
{isViewAllOpen ? "View less" : "View all"}
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|