133 lines
5.8 KiB
TypeScript
133 lines
5.8 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 { useMemo } from "react";
|
|
import { observer } from "mobx-react";
|
|
import { useTranslation } from "@plane/i18n";
|
|
import { Badge } from "@plane/propel/badge";
|
|
import { MembersPropertyIcon } from "@plane/propel/icons";
|
|
import type { TIssue, TIssuePriorities } from "@plane/types";
|
|
import { 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 selectedTargetProject = data.target_project_id ? getTargetProjectById(data.target_project_id) : undefined;
|
|
const selectedTargetOptions = getTargetOptionsByProjectId(data.target_project_id);
|
|
const targetLabelIds = useMemo(
|
|
() => selectedTargetOptions?.labels?.map((label) => label.id) ?? [],
|
|
[selectedTargetOptions?.labels]
|
|
);
|
|
const assigneeLabel = useMemo(() => {
|
|
const assigneeIds = data.assignee_ids || [];
|
|
if (!assigneeIds.length) return t("external_contours_page.form.assignee");
|
|
if (assigneeIds.length === 1) return getUserDetails(assigneeIds[0])?.display_name || t("external_contours_page.form.assignee");
|
|
return `${assigneeIds.length} ${t("assignees").toLocaleLowerCase()}`;
|
|
}, [data.assignee_ids, getUserDetails, t]);
|
|
const getTargetLabelById = (labelId: string) =>
|
|
selectedTargetOptions?.labels?.find((label) => label.id === labelId) ?? null;
|
|
|
|
return (
|
|
<div className="relative flex flex-wrap items-center gap-2">
|
|
<div className="rounded-md border border-subtle bg-surface-2 px-3 py-1.5 text-11 text-secondary">
|
|
<span className="mr-2 text-tertiary">{t("external_contours_page.form.source_project")}</span>
|
|
<Badge variant="neutral">{currentProjectName || "NODE.DC"}</Badge>
|
|
</div>
|
|
|
|
<div className="h-7">
|
|
<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="border-with-text"
|
|
placeholder={t("external_contours_page.form.target_project")}
|
|
disabled={targetProjectIds.length === 0}
|
|
/>
|
|
</div>
|
|
|
|
{selectedTargetProject && (
|
|
<div className="rounded-md border border-subtle bg-surface-2 px-3 py-1.5 text-11 text-secondary">
|
|
<span className="mr-2 text-tertiary">{t("external_contours_page.form.selected_target")}</span>
|
|
<Badge variant="neutral">{selectedTargetProject.name}</Badge>
|
|
</div>
|
|
)}
|
|
|
|
<div className="h-7">
|
|
<PriorityDropdown
|
|
value={data.priority}
|
|
onChange={(priority) => handleData("priority", priority)}
|
|
buttonVariant="border-with-text"
|
|
/>
|
|
</div>
|
|
|
|
<div className="h-7">
|
|
<MemberDropdownBase
|
|
value={data.assignee_ids || []}
|
|
onChange={(assigneeIds) => handleData("assignee_ids", assigneeIds)}
|
|
getUserDetails={getUserDetails}
|
|
memberIds={selectedTargetOptions?.member_ids ?? []}
|
|
button={
|
|
<div className="flex h-full items-center justify-start gap-1.5 rounded-sm border-[0.5px] border-strong px-1.5 text-11 text-secondary">
|
|
<ButtonAvatars showTooltip={false} userIds={data.assignee_ids || []} icon={MembersPropertyIcon} />
|
|
<span className="flex-grow truncate text-left text-body-xs-medium leading-5">{assigneeLabel}</span>
|
|
</div>
|
|
}
|
|
buttonVariant={(data.assignee_ids || []).length > 0 ? "transparent-without-text" : "border-with-text"}
|
|
buttonClassName={(data.assignee_ids || []).length > 0 ? "hover:bg-transparent" : ""}
|
|
optionsClassName="z-[60]"
|
|
placeholder={t("external_contours_page.form.assignee")}
|
|
disabled={!data.target_project_id || !selectedTargetOptions}
|
|
multiple
|
|
/>
|
|
</div>
|
|
|
|
<div className="h-7">
|
|
<WorkItemLabelSelectBase
|
|
value={data.label_ids || []}
|
|
onChange={(labelIds) => handleData("label_ids", labelIds)}
|
|
getLabelById={getTargetLabelById}
|
|
labelIds={targetLabelIds}
|
|
disabled={!data.target_project_id || !selectedTargetOptions}
|
|
/>
|
|
</div>
|
|
|
|
<div className="h-7">
|
|
<DateDropdown
|
|
value={data.target_date || null}
|
|
onChange={(date) => handleData("target_date", date ? renderFormattedPayloadDate(date) : "")}
|
|
buttonVariant="border-with-text"
|
|
placeholder={t("external_contours_page.form.due_date")}
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
});
|