NODEDC_TASKMANAGER/plane-src/apps/web/ce/components/projects/external-contours/create-properties.tsx

175 lines
8.1 KiB
TypeScript

/**
* Copyright (c) 2023-present Plane Software, Inc. and contributors
* SPDX-License-Identifier: AGPL-3.0-only
* See the LICENSE file for details.
*/
import { CalendarDays, SignalHigh } from "lucide-react";
import { observer } from "mobx-react";
import { ISSUE_PRIORITIES } from "@plane/constants";
import { useTranslation } from "@plane/i18n";
import { LabelPropertyIcon, MembersPropertyIcon, PriorityIcon, ProjectIcon } from "@plane/propel/icons";
import type { TIssue, TIssuePriorities } from "@plane/types";
import { cn, renderFormattedDate, renderFormattedPayloadDate } from "@plane/utils";
import { DateDropdown } from "@/components/dropdowns/date";
import { ButtonAvatars } from "@/components/dropdowns/member/avatar";
import { MemberDropdownBase } from "@/components/dropdowns/member/base";
import { PriorityDropdown } from "@/components/dropdowns/priority";
import { ProjectDropdownBase } from "@/components/dropdowns/project/base";
import { WorkItemLabelSelectBase } from "@/components/issues/select/base";
import { useMember } from "@/hooks/store/use-member";
import { useProjectExternalContours } from "@/hooks/store/use-project-external-contours";
type Props = {
currentProjectName?: string;
data: Partial<TIssue> & { target_project_id?: string | null; priority?: TIssuePriorities };
handleData: (issueKey: keyof Props["data"], issueValue: Props["data"][keyof Props["data"]]) => void;
};
export const ExternalContoursCreateProperties = observer(function ExternalContoursCreateProperties(props: Props) {
const { currentProjectName, data, handleData } = props;
const { t } = useTranslation();
const { getUserDetails } = useMember();
const { targetProjectIds, getTargetOptionsByProjectId, getTargetProjectById } = useProjectExternalContours();
const controlClassName =
"nodedc-modal-field flex h-10 min-w-fit items-center gap-2 rounded-[1.25rem] px-3.5 text-[12px] font-medium text-secondary";
const selectedTargetProject = data.target_project_id ? getTargetProjectById(data.target_project_id) : undefined;
const selectedTargetOptions = getTargetOptionsByProjectId(data.target_project_id);
const getTargetLabelById = (labelId: string) =>
selectedTargetOptions?.labels?.find((label) => label.id === labelId) ?? null;
const targetLabelIds = selectedTargetOptions?.labels?.map((label) => label.id) ?? [];
const selectedLabels = (data.label_ids || []).map((labelId) => getTargetLabelById(labelId)).filter((label) => !!label);
const assigneeIds = data.assignee_ids || [];
const assigneeLabel = !assigneeIds.length
? t("external_contours_page.form.assignee")
: assigneeIds.length === 1
? (getUserDetails(assigneeIds[0])?.display_name ?? t("external_contours_page.form.assignee"))
: `${assigneeIds.length} ${t("assignees").toLocaleLowerCase()}`;
const priorityDetails = ISSUE_PRIORITIES.find((priority) => priority.key === data.priority);
return (
<div className="relative flex flex-wrap items-center gap-2.5">
<div className={cn(controlClassName, "min-w-[16rem] justify-start")}>
<span className="text-tertiary">{t("external_contours_page.form.source_project")}</span>
<span className="truncate text-primary">{currentProjectName || "NODE.DC"}</span>
</div>
<div className="h-10">
<ProjectDropdownBase
value={data.target_project_id ?? null}
onChange={(value) => {
if (!Array.isArray(value)) {
handleData("target_project_id", value);
handleData("assignee_ids", []);
handleData("label_ids", []);
}
}}
multiple={false}
projectIds={targetProjectIds}
getProjectById={getTargetProjectById}
buttonVariant="transparent-without-text"
buttonContainerClassName="h-full"
button={
<div className={cn(controlClassName, "min-w-[16rem] justify-start")}>
<ProjectIcon className="h-3.5 w-3.5 flex-shrink-0 text-tertiary" />
<span className={cn("truncate", selectedTargetProject ? "text-primary" : "text-tertiary")}>
{selectedTargetProject?.name ?? t("external_contours_page.form.target_project")}
</span>
</div>
}
placeholder={t("external_contours_page.form.target_project")}
disabled={targetProjectIds.length === 0}
/>
</div>
<div className="h-10">
<MemberDropdownBase
value={assigneeIds}
onChange={(nextAssigneeIds) => handleData("assignee_ids", nextAssigneeIds)}
getUserDetails={getUserDetails}
memberIds={selectedTargetOptions?.member_ids ?? []}
button={
<div className={cn(controlClassName, "min-w-[12rem] justify-start")}>
<ButtonAvatars showTooltip={false} userIds={assigneeIds} icon={MembersPropertyIcon} />
<span className={cn("truncate", assigneeIds.length > 0 ? "text-primary" : "text-tertiary")}>
{assigneeLabel}
</span>
</div>
}
buttonVariant="transparent-without-text"
optionsClassName="z-[60]"
placeholder={t("external_contours_page.form.assignee")}
disabled={!data.target_project_id || !selectedTargetOptions}
multiple
/>
</div>
<div className="h-10">
<PriorityDropdown
value={data.priority}
onChange={(priority) => handleData("priority", priority)}
buttonVariant="transparent-without-text"
buttonContainerClassName="h-full"
button={
<div className={cn(controlClassName, "min-w-[12rem] justify-start")}>
{data.priority && data.priority !== "none" ? (
<PriorityIcon priority={data.priority} size={14} />
) : (
<SignalHigh className="h-3.5 w-3.5 flex-shrink-0 text-tertiary" />
)}
<span className={cn("truncate", data.priority && data.priority !== "none" ? "text-primary" : "text-tertiary")}>
{data.priority && data.priority !== "none"
? priorityDetails?.title
: t("external_contours_page.form.priority")}
</span>
</div>
}
placeholder={t("external_contours_page.form.priority")}
/>
</div>
<div className="h-10">
<WorkItemLabelSelectBase
value={data.label_ids || []}
onChange={(labelIds) => handleData("label_ids", labelIds)}
getLabelById={getTargetLabelById}
labelIds={targetLabelIds}
buttonContainerClassName="h-full !text-[12px]"
label={
<div className={cn(controlClassName, "min-w-[9rem] justify-start !text-[12px]")}>
<LabelPropertyIcon className="h-3.5 w-3.5 flex-shrink-0 text-tertiary" />
<span className={cn("truncate", selectedLabels.length > 0 ? "text-primary" : "text-tertiary")}>
{selectedLabels.length > 0
? selectedLabels.length === 1
? selectedLabels[0]?.name
: `${selectedLabels.length} ${t("labels").toLocaleLowerCase()}`
: t("labels")}
</span>
</div>
}
disabled={!data.target_project_id || !selectedTargetOptions}
/>
</div>
<div className="h-10">
<DateDropdown
value={data.target_date || null}
onChange={(date) => handleData("target_date", date ? renderFormattedPayloadDate(date) : "")}
buttonVariant="transparent-without-text"
buttonContainerClassName="h-full"
button={
<div className={cn(controlClassName, "min-w-[10rem] justify-start")}>
<CalendarDays className="h-3.5 w-3.5 flex-shrink-0 text-tertiary" />
<span className={cn("truncate", data.target_date ? "text-primary" : "text-tertiary")}>
{data.target_date ? renderFormattedDate(data.target_date) : t("external_contours_page.form.due_date")}
</span>
</div>
}
placeholder={t("external_contours_page.form.due_date")}
/>
</div>
</div>
);
});