mirror of
https://github.com/makeplane/plane
synced 2024-06-14 14:31:34 +00:00
2dcaccd4ec
* chore: dynamic position dropdown (#2138) * chore: dynamic position state dropdown for issue view * style: state select dropdown styling * fix: state icon attribute names * chore: state select dynamic dropdown * chore: member select dynamic dropdown * chore: label select dynamic dropdown * chore: priority select dynamic dropdown * chore: label select dropdown improvement * refactor: state dropdown location * chore: dropdown improvement and code refactor * chore: dynamic dropdown hook type added --------- Co-authored-by: Aaryan Khandelwal <aaryankhandu123@gmail.com> * fix: fields not getting selected in the create issue form (#2212) * fix: hydration error and draft issue workflow * fix: build error * fix: properties getting de-selected after create, module & cycle not getting auto-select on the form * fix: display layout, props being updated directly * chore: sub issues count in individual issue (#2221) * fix: service imports * chore: rename csv service file --------- Co-authored-by: Anmol Singh Bhatia <121005188+anmolsinghbhatia@users.noreply.github.com> Co-authored-by: Dakshesh Jain <65905942+dakshesh14@users.noreply.github.com> Co-authored-by: Bavisetti Narayan <72156168+NarayanBavisetti@users.noreply.github.com>
204 lines
6.5 KiB
TypeScript
204 lines
6.5 KiB
TypeScript
import React from "react";
|
|
|
|
import { useRouter } from "next/router";
|
|
|
|
import useSWR from "swr";
|
|
|
|
// services
|
|
import issuesService from "services/issue.service";
|
|
import projectService from "services/project.service";
|
|
// hooks
|
|
import useProjects from "hooks/use-projects";
|
|
// component
|
|
import { Avatar, Icon } from "components/ui";
|
|
// icons
|
|
import { PlusIcon } from "@heroicons/react/24/outline";
|
|
import { PriorityIcon, StateGroupIcon } from "components/icons";
|
|
// helpers
|
|
import { addSpaceIfCamelCase } from "helpers/string.helper";
|
|
import { renderEmoji } from "helpers/emoji.helper";
|
|
// types
|
|
import { IIssueViewProps, IState, TIssuePriorities, TStateGroups } from "types";
|
|
// fetch-keys
|
|
import { PROJECT_ISSUE_LABELS, PROJECT_MEMBERS } from "constants/fetch-keys";
|
|
// constants
|
|
import { STATE_GROUP_COLORS } from "constants/state";
|
|
|
|
type Props = {
|
|
currentState?: IState | null;
|
|
groupTitle: string;
|
|
addIssueToGroup: () => void;
|
|
isCollapsed: boolean;
|
|
setIsCollapsed: React.Dispatch<React.SetStateAction<boolean>>;
|
|
disableUserActions: boolean;
|
|
disableAddIssue: boolean;
|
|
viewProps: IIssueViewProps;
|
|
};
|
|
|
|
export const BoardHeader: React.FC<Props> = ({
|
|
currentState,
|
|
groupTitle,
|
|
addIssueToGroup,
|
|
isCollapsed,
|
|
setIsCollapsed,
|
|
disableUserActions,
|
|
disableAddIssue,
|
|
viewProps,
|
|
}) => {
|
|
const router = useRouter();
|
|
const { workspaceSlug, projectId } = router.query;
|
|
|
|
const { displayFilters, groupedIssues } = viewProps;
|
|
|
|
const { data: issueLabels } = useSWR(
|
|
workspaceSlug && projectId && displayFilters?.group_by === "labels"
|
|
? PROJECT_ISSUE_LABELS(projectId.toString())
|
|
: null,
|
|
workspaceSlug && projectId && displayFilters?.group_by === "labels"
|
|
? () => issuesService.getIssueLabels(workspaceSlug.toString(), projectId.toString())
|
|
: null
|
|
);
|
|
|
|
const { data: members } = useSWR(
|
|
workspaceSlug &&
|
|
projectId &&
|
|
(displayFilters?.group_by === "created_by" || displayFilters?.group_by === "assignees")
|
|
? PROJECT_MEMBERS(projectId.toString())
|
|
: null,
|
|
workspaceSlug &&
|
|
projectId &&
|
|
(displayFilters?.group_by === "created_by" || displayFilters?.group_by === "assignees")
|
|
? () => projectService.projectMembers(workspaceSlug.toString(), projectId.toString())
|
|
: null
|
|
);
|
|
|
|
const { projects } = useProjects();
|
|
|
|
const getGroupTitle = () => {
|
|
let title = addSpaceIfCamelCase(groupTitle);
|
|
|
|
switch (displayFilters?.group_by) {
|
|
case "state":
|
|
title = addSpaceIfCamelCase(currentState?.name ?? "");
|
|
break;
|
|
case "labels":
|
|
title = issueLabels?.find((label) => label.id === groupTitle)?.name ?? "None";
|
|
break;
|
|
case "project":
|
|
title = projects?.find((p) => p.id === groupTitle)?.name ?? "None";
|
|
break;
|
|
case "assignees":
|
|
case "created_by":
|
|
const member = members?.find((member) => member.member.id === groupTitle)?.member;
|
|
title = member ? member.display_name : "None";
|
|
|
|
break;
|
|
}
|
|
|
|
return title;
|
|
};
|
|
|
|
const getGroupIcon = () => {
|
|
let icon;
|
|
|
|
switch (displayFilters?.group_by) {
|
|
case "state":
|
|
icon = currentState && (
|
|
<StateGroupIcon stateGroup={currentState.group} color={currentState.color} height="16px" width="16px" />
|
|
);
|
|
break;
|
|
case "state_detail.group":
|
|
icon = (
|
|
<StateGroupIcon
|
|
stateGroup={groupTitle as TStateGroups}
|
|
color={STATE_GROUP_COLORS[groupTitle as TStateGroups]}
|
|
height="16px"
|
|
width="16px"
|
|
/>
|
|
);
|
|
break;
|
|
case "priority":
|
|
icon = <PriorityIcon priority={groupTitle as TIssuePriorities} className="text-lg" />;
|
|
break;
|
|
case "project":
|
|
const project = projects?.find((p) => p.id === groupTitle);
|
|
icon =
|
|
project &&
|
|
(project.emoji !== null
|
|
? renderEmoji(project.emoji)
|
|
: project.icon_prop !== null
|
|
? renderEmoji(project.icon_prop)
|
|
: null);
|
|
break;
|
|
case "labels":
|
|
const labelColor = issueLabels?.find((label) => label.id === groupTitle)?.color ?? "#000000";
|
|
icon = <span className="h-3.5 w-3.5 flex-shrink-0 rounded-full" style={{ backgroundColor: labelColor }} />;
|
|
break;
|
|
case "assignees":
|
|
case "created_by":
|
|
const member = members?.find((member) => member.member.id === groupTitle)?.member;
|
|
icon = member ? <Avatar user={member} height="24px" width="24px" fontSize="12px" /> : <></>;
|
|
|
|
break;
|
|
}
|
|
|
|
return icon;
|
|
};
|
|
|
|
return (
|
|
<div
|
|
className={`flex items-center justify-between px-1 ${
|
|
!isCollapsed ? "flex-col rounded-md bg-custom-background-90" : ""
|
|
}`}
|
|
>
|
|
<div className={`flex items-center ${isCollapsed ? "gap-1" : "flex-col gap-2"}`}>
|
|
<div
|
|
className={`flex cursor-pointer items-center gap-x-2 max-w-[316px] ${
|
|
!isCollapsed ? "mb-2 flex-col gap-y-2 py-2" : ""
|
|
}`}
|
|
>
|
|
<span className="flex items-center">{getGroupIcon()}</span>
|
|
<h2
|
|
className={`text-lg font-semibold truncate ${
|
|
displayFilters?.group_by === "created_by" ? "" : "capitalize"
|
|
}`}
|
|
style={{
|
|
writingMode: isCollapsed ? "horizontal-tb" : "vertical-rl",
|
|
}}
|
|
>
|
|
{getGroupTitle()}
|
|
</h2>
|
|
<span className={`${isCollapsed ? "ml-0.5" : ""} py-1 text-center text-sm`}>
|
|
{groupedIssues?.[groupTitle].length ?? 0}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
|
|
<div className={`flex items-center ${!isCollapsed ? "flex-col pb-2" : ""}`}>
|
|
<button
|
|
type="button"
|
|
className="grid h-7 w-7 place-items-center rounded p-1 text-custom-text-200 outline-none duration-300 hover:bg-custom-background-80"
|
|
onClick={() => {
|
|
setIsCollapsed((prevData) => !prevData);
|
|
}}
|
|
>
|
|
{isCollapsed ? (
|
|
<Icon iconName="close_fullscreen" className="text-base font-medium text-custom-text-900" />
|
|
) : (
|
|
<Icon iconName="open_in_full" className="text-base font-medium text-custom-text-900" />
|
|
)}
|
|
</button>
|
|
{!disableAddIssue && !disableUserActions && displayFilters?.group_by !== "created_by" && (
|
|
<button
|
|
type="button"
|
|
className="grid h-7 w-7 place-items-center rounded p-1 text-custom-text-200 outline-none duration-300 hover:bg-custom-background-80"
|
|
onClick={addIssueToGroup}
|
|
>
|
|
<PlusIcon className="h-4 w-4" />
|
|
</button>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|