forked from github/plane
9075f9441c
* style: added cta at the bottom of sidebar, added missing icons as well, showing dynamic workspace member count on workspace dropdown * refractor: running parallel request, made create/edit label function to async function * fix: sidebar dropdown content going below kanban items outside click detection in need help dropdown * refractor: making parallel api calls fix: create state input comes at bottom, create state input gets on focus automatically, form is getting submitted on enter click * refactoring file structure and signin page * style: changed text and added spinner for signing in loading * refractor: removed unused type * fix: my issue cta in profile page sending to 404 page * fix: added new s3 bucket url in next.config.js file increased image modal height * packaging UI components * eslint config * eslint fixes * refactoring changes * build fixes * minor fixes * adding todo comments for reference * refactor: cleared unused imports and re ordered imports * refactor: removed unused imports * fix: added workspace argument to useissues hook * refactor: removed api-routes file, unnecessary constants * refactor: created helpers folder, removed unnecessary constants * refactor: new context for issue view * refactoring issues page * build fixes * refactoring * refactor: create issue modal * refactor: module ui * fix: sub-issues mutation * fix: create more option in create issue modal * description form debounce issue * refactor: global component for assignees list * fix: link module interface * fix: priority icons and sub-issues count added * fix: cycle mutation in issue details page * fix: remove issue from cycle mutation * fix: create issue modal in home page * fix: removed unnecessary props * fix: updated create issue form status * fix: settings auth breaking * refactor: issue details page Co-authored-by: Dakshesh Jain <dakshesh.jain14@gmail.com> Co-authored-by: Dakshesh Jain <65905942+dakshesh14@users.noreply.github.com> Co-authored-by: venkatesh-soulpage <venkatesh.marreboyina@soulpageit.com> Co-authored-by: Aaryan Khandelwal <aaryankhandu123@gmail.com> Co-authored-by: Anmol Singh Bhatia <anmolsinghbhatia1001@gmail.com>
175 lines
5.7 KiB
TypeScript
175 lines
5.7 KiB
TypeScript
import React, { useState } from "react";
|
|
|
|
import { useRouter } from "next/router";
|
|
|
|
import useSWR from "swr";
|
|
|
|
// react-beautiful-dnd
|
|
import { Draggable } from "react-beautiful-dnd";
|
|
// components
|
|
import StrictModeDroppable from "components/dnd/StrictModeDroppable";
|
|
import SingleIssue from "components/common/board-view/single-issue";
|
|
import BoardHeader from "components/common/board-view/board-header";
|
|
// icons
|
|
import { PlusIcon } from "@heroicons/react/24/outline";
|
|
// services
|
|
import workspaceService from "services/workspace.service";
|
|
// types
|
|
import { IIssue, Properties, NestedKeyOf, IWorkspaceMember } from "types";
|
|
// fetch-keys
|
|
import { WORKSPACE_MEMBERS } from "constants/fetch-keys";
|
|
|
|
type Props = {
|
|
selectedGroup: NestedKeyOf<IIssue> | null;
|
|
groupTitle: string;
|
|
groupedByIssues: {
|
|
[key: string]: IIssue[];
|
|
};
|
|
index: number;
|
|
setIsIssueOpen: React.Dispatch<React.SetStateAction<boolean>>;
|
|
properties: Properties;
|
|
setPreloadedData: React.Dispatch<
|
|
React.SetStateAction<
|
|
| (Partial<IIssue> & {
|
|
actionType: "createIssue" | "edit" | "delete";
|
|
})
|
|
| undefined
|
|
>
|
|
>;
|
|
bgColor?: string;
|
|
stateId: string | null;
|
|
createdBy: string | null;
|
|
handleDeleteIssue: React.Dispatch<React.SetStateAction<string | undefined>>;
|
|
partialUpdateIssue: (formData: Partial<IIssue>, childIssueId: string) => void;
|
|
};
|
|
|
|
const SingleBoard: React.FC<Props> = ({
|
|
selectedGroup,
|
|
groupTitle,
|
|
groupedByIssues,
|
|
index,
|
|
setIsIssueOpen,
|
|
properties,
|
|
setPreloadedData,
|
|
bgColor = "#0f2b16",
|
|
stateId,
|
|
createdBy,
|
|
handleDeleteIssue,
|
|
partialUpdateIssue,
|
|
}) => {
|
|
// Collapse/Expand
|
|
const [isCollapsed, setIsCollapsed] = useState(true);
|
|
|
|
const router = useRouter();
|
|
const { workspaceSlug } = router.query;
|
|
|
|
if (selectedGroup === "priority")
|
|
groupTitle === "high"
|
|
? (bgColor = "#dc2626")
|
|
: groupTitle === "medium"
|
|
? (bgColor = "#f97316")
|
|
: groupTitle === "low"
|
|
? (bgColor = "#22c55e")
|
|
: (bgColor = "#ff0000");
|
|
|
|
const { data: people } = useSWR<IWorkspaceMember[]>(
|
|
workspaceSlug ? WORKSPACE_MEMBERS : null,
|
|
workspaceSlug ? () => workspaceService.workspaceMembers(workspaceSlug as string) : null
|
|
);
|
|
|
|
const addIssueToState = () => {
|
|
setIsIssueOpen(true);
|
|
if (selectedGroup)
|
|
setPreloadedData({
|
|
state: stateId !== null ? stateId : undefined,
|
|
[selectedGroup]: groupTitle,
|
|
actionType: "createIssue",
|
|
});
|
|
};
|
|
|
|
return (
|
|
<Draggable draggableId={groupTitle} index={index}>
|
|
{(provided, snapshot) => (
|
|
<div
|
|
className={`h-full flex-shrink-0 rounded ${
|
|
snapshot.isDragging ? "border-theme shadow-lg" : ""
|
|
} ${!isCollapsed ? "" : "w-80 border bg-gray-50"}`}
|
|
ref={provided.innerRef}
|
|
{...provided.draggableProps}
|
|
>
|
|
<div
|
|
className={`${!isCollapsed ? "" : "flex h-full flex-col space-y-3 overflow-y-auto"}`}
|
|
>
|
|
<BoardHeader
|
|
addIssueToState={addIssueToState}
|
|
bgColor={bgColor}
|
|
createdBy={createdBy}
|
|
groupTitle={groupTitle}
|
|
groupedByIssues={groupedByIssues}
|
|
isCollapsed={isCollapsed}
|
|
setIsCollapsed={setIsCollapsed}
|
|
selectedGroup={selectedGroup}
|
|
provided={provided}
|
|
/>
|
|
<StrictModeDroppable key={groupTitle} droppableId={groupTitle}>
|
|
{(provided, snapshot) => (
|
|
<div
|
|
className={`mt-3 h-full space-y-3 overflow-y-auto px-3 pb-3 ${
|
|
snapshot.isDraggingOver ? "bg-indigo-50 bg-opacity-50" : ""
|
|
} ${!isCollapsed ? "hidden" : "block"}`}
|
|
{...provided.droppableProps}
|
|
ref={provided.innerRef}
|
|
>
|
|
{groupedByIssues[groupTitle].map((childIssue, index: number) => {
|
|
const assignees = [
|
|
...(childIssue?.assignees_list ?? []),
|
|
...(childIssue?.assignees ?? []),
|
|
]?.map((assignee) => {
|
|
const tempPerson = people?.find((p) => p.member.id === assignee)?.member;
|
|
|
|
return tempPerson;
|
|
});
|
|
|
|
return (
|
|
<Draggable key={childIssue.id} draggableId={childIssue.id} index={index}>
|
|
{(provided, snapshot) => (
|
|
<div
|
|
ref={provided.innerRef}
|
|
{...provided.draggableProps}
|
|
{...provided.dragHandleProps}
|
|
>
|
|
<SingleIssue
|
|
issue={childIssue}
|
|
properties={properties}
|
|
snapshot={snapshot}
|
|
people={people}
|
|
assignees={assignees}
|
|
handleDeleteIssue={handleDeleteIssue}
|
|
partialUpdateIssue={partialUpdateIssue}
|
|
/>
|
|
</div>
|
|
)}
|
|
</Draggable>
|
|
);
|
|
})}
|
|
{provided.placeholder}
|
|
<button
|
|
type="button"
|
|
className="flex items-center rounded p-2 text-xs font-medium outline-none duration-300 hover:bg-gray-100"
|
|
onClick={addIssueToState}
|
|
>
|
|
<PlusIcon className="mr-1 h-3 w-3" />
|
|
Create
|
|
</button>
|
|
</div>
|
|
)}
|
|
</StrictModeDroppable>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</Draggable>
|
|
);
|
|
};
|
|
|
|
export default SingleBoard;
|