forked from github/plane
c9252c9713
* chore: new link endpoints
* chore: added created by info for link
* chore: cannot have empty state group
* feat: filtering for cycle and module issue and updated grouper function for grouping in modules and cycles (#342)
* docs: github integration (#346)
* fix: add pagination for github repositories endpoint (#345)
* fix: remove bot accounts from list api (#344)
* refactor: create new endpoints for date checking getting current upcoming and past cycles (#343)
* refactor: create new endpoints for date checking getting current upcoming and past cycles
* refactor: rename endpoint to match consistency
* fix: remove project slug (#340)
* refactor: update links to different endpoints (#338)
* chore: cycle validation services and constants added
* style: kanban board
* chore: cycle type and services updated
* chore: completed cycle dynamic importing and refactor
* feat: cycle modal date validation
* fix: build fix
* style: redesigned sidebar, added new icons and spacing changes
* style: changed app header color to white
* feat: cover image selector for project create
* style/projects_page
* style: added dragging state design
* fix: cycle form date
* chore: draft cycle services and types
* feat: draft tab and cycle sidebar update
* style: projects list page
* fix: image aspect ratio
* style: assignee drop down label
* style: new primary button design
* style: assignee dropdown
* style: assignee dropdown stlye fix
* style: state dropdown redesign
* style: dropdown ui consisteny
* style: priority dropdown redesign
* style: label dropdown redesign
* style: issue dropdown re-order
* style: state Icon
* style: date dropdown redesign
* fix: dropdown issue label
* style: transsition
* style: color fixed
* chore: labels list file and function rename
* style: redesigned create project modal
style: changed image picker to pop-over instread of modal
* fix: upload button on workspace settings page not working, UX of workspace settings image upload
* feat: date range status function added
* style: project settings pages
* fix: merge conflicts
* fix: mutation fix and date range helper fn added
* style: workspace settings pages
* style: dropdowns, feat: favorite projects in sidebar
* feat: global component for combobox with new design
* feat: custom context menu for issues in kanban board
* refactor: global context menu component
* chore: updated context menu component
* chore: updated sidebar selects
* style: kanban horizontal scrollbar added (#372)
* style: new cycle list (#374)
* feat: short date helper function
* feat: linear progress indicator added
* style: new cyce list and cycle card design
* feat: short date function improve
* feat: linear progress indicator improvement
* style: cycle card and progress indicator
* fix: helper date function and progress indicator fix
* fix: build error
---------
Co-authored-by: Aaryan Khandelwal <aaryankhandu123@gmail.com>
* chore: updated project favorites endpoints (#375)
* feat: favorite cycle and style: style improvements (#376)
* style: consistent btn
* style: caret direction for disclosure
* fix: progress tooltip value rounded
* chore: favorite cycle serivces
* chore: favorite cycle type and constant
* feat: favorite cycle feat added
* refactor: favorite services and type
* fix: build fix
* refactor: sidebar projects menu (#377)
* feat: add endpoint for draft cycles and add validation for creating draft cycles (#355)
* feat: add endpoint for draft cycles and add validation for creating draft cycles
* fix: key error in cycle create endpoint
* feat: delete file assets from storage (#373)
* chore: rename past cycle to completed cycle (#347)
* fix: workspace member listing endpoint (#348)
* fix: module issue viewset typo (#349)
* feat: add project to favourites (#352)
* feat: add project to favourites
* feat: add project is_favourite attribute to list endpoints
* refactor: updated destroy endpoint to send project_id
* chore: nomenclature update
* feat: add cover image to project (#353)
* fix: cycle date filtering for current and upcoming cycle (#357)
* fix: update filtering for completed cycles
* fix: filter updated for upcoming cycles
* fix: cycle and module issue filtering (#363)
* feat: already exisiting url validation (#368)
* feat: cycle favourites for user (#369)
* feat: cycle favourites for user
* chore: update nomenclature
* chore: update on nomenclature
* feat: add favorites for completed and current cycle endpoints
* feat: module favourites for user (#370)
* feat: added floating toolbar on text selection (#378)
style: re-designed create-issue modal
* dev: migrations added for ProjectFavorite, ModuleFavorite, CycleFavorite including a bunch of other attribs
* chore: cycles loading, fix: cycles favorite mutation (#379)
* style: cycle sidebar, fix: cycle card bug fix (#383)
* style: new cycle sidebar
* style: other information section
* style: progress bar bg fix
* fix: cycle card bug fix
* style: progress chart
* style: chart tooltip
* style : module sidebar (#385)
* style: new cycle sidebar
* style: other information section
* style: progress bar bg fix
* fix: cycle card bug fix
* style: progress chart
* style: chart tooltip
* style: module link tab added in sidebar stats
* style: lead and member select
* fix: text selection moving when typing in between (#384)
* feat: added floating toolbar on text selection (#386)
style: re-designed create-issue modal
* style :module list (#387)
* chore: module favorite type and services
* style: module list
* style: module list and card
* fix: link fix
* style: truncate (#388)
* style: truncate
* fix: truncate text added to cycle and module card
* fix: custom menu link item (#390)
* fix: ui fixes (#392)
* fix: ui fixes
* chore: kanban issue title length
* style: ui fix (#393)
* style: truncate
* fix: truncate text added to cycle and module card
* fix: progress percentage
* feat: cycle card tooltip
* fix: sidebar fix
* fix: edit module mutation error (#394)
* fix: issue details mutation (#389)
* fix: ui improvement (#395)
* fix: current cycle date updation
* fix: sidebar overflow fix , date helper fn added
* chore: update module dropdowns (#396)
* fix: project member filter for bot accounts (#391)
* fix: make api token only view once (#382)
* dev: add back migration for project cover images (#381)
* fix: rename db host name for docker setup (#380)
* dev: promote to staging (#397)
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: sriram veeraghanta <veeraghanta.sriram@gmail.com>
Co-authored-by: pablohashescobar <nikhilschacko@gmail.com>
Co-authored-by: pablohashescobar <118773738+pablohashescobar@users.noreply.github.com>
Co-authored-by: Anmol Singh Bhatia <121005188+anmolsinghbhatia@users.noreply.github.com>
Co-authored-by: Anmol Singh Bhatia <anmolsinghbhatia1001@gmail.com>
Co-authored-by: Aaryan Khandelwal <65252264+aaryan610@users.noreply.github.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
Co-authored-by: Dakshesh Jain <65905942+dakshesh14@users.noreply.github.com>
Co-authored-by: Aaryan Khandelwal <aaryankhandu123@gmail.com>
Co-authored-by: Narayana <narayana.vadapalli1996@gmail.com>
* Revert "dev: promote to staging (#397)" (#398)
This reverts commit f7405ba1d6
.
---------
Signed-off-by: dependabot[bot] <support@github.com>
Co-authored-by: Aaryan Khandelwal <aaryankhandu123@gmail.com>
Co-authored-by: Aaryan Khandelwal <65252264+aaryan610@users.noreply.github.com>
Co-authored-by: pablohashescobar <118773738+pablohashescobar@users.noreply.github.com>
Co-authored-by: sphynxux <122926002+sphynxux@users.noreply.github.com>
Co-authored-by: Anmol Singh Bhatia <anmolsinghbhatia@caravel.tech>
Co-authored-by: Dakshesh Jain <dakshesh.jain14@gmail.com>
Co-authored-by: Dakshesh Jain <65905942+dakshesh14@users.noreply.github.com>
Co-authored-by: Anmol Singh Bhatia <121005188+anmolsinghbhatia@users.noreply.github.com>
Co-authored-by: sriram veeraghanta <veeraghanta.sriram@gmail.com>
Co-authored-by: pablohashescobar <nikhilschacko@gmail.com>
Co-authored-by: Anmol Singh Bhatia <anmolsinghbhatia1001@gmail.com>
Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
Co-authored-by: Narayana <narayana.vadapalli1996@gmail.com>
441 lines
14 KiB
TypeScript
441 lines
14 KiB
TypeScript
import { useCallback, useState } from "react";
|
|
|
|
import { useRouter } from "next/router";
|
|
|
|
import useSWR, { mutate } from "swr";
|
|
|
|
// react-beautiful-dnd
|
|
import { DragDropContext, DropResult } from "react-beautiful-dnd";
|
|
// services
|
|
import issuesService from "services/issues.service";
|
|
import stateService from "services/state.service";
|
|
import projectService from "services/project.service";
|
|
import modulesService from "services/modules.service";
|
|
// hooks
|
|
import useIssueView from "hooks/use-issue-view";
|
|
// components
|
|
import { AllLists, AllBoards } from "components/core";
|
|
import { CreateUpdateIssueModal, DeleteIssueModal } from "components/issues";
|
|
import StrictModeDroppable from "components/dnd/StrictModeDroppable";
|
|
// icons
|
|
import { TrashIcon } from "@heroicons/react/24/outline";
|
|
// helpers
|
|
import { getStatesList } from "helpers/state.helper";
|
|
// types
|
|
import { CycleIssueResponse, IIssue, ModuleIssueResponse, UserAuth } from "types";
|
|
// fetch-keys
|
|
import {
|
|
CYCLE_ISSUES,
|
|
MODULE_ISSUES,
|
|
PROJECT_ISSUES_LIST,
|
|
PROJECT_MEMBERS,
|
|
STATE_LIST,
|
|
} from "constants/fetch-keys";
|
|
|
|
type Props = {
|
|
type?: "issue" | "cycle" | "module";
|
|
issues: IIssue[];
|
|
openIssuesListModal?: () => void;
|
|
userAuth: UserAuth;
|
|
};
|
|
|
|
export const IssuesView: React.FC<Props> = ({
|
|
type = "issue",
|
|
issues,
|
|
openIssuesListModal,
|
|
userAuth,
|
|
}) => {
|
|
// create issue modal
|
|
const [createIssueModal, setCreateIssueModal] = useState(false);
|
|
const [preloadedData, setPreloadedData] = useState<
|
|
(Partial<IIssue> & { actionType: "createIssue" | "edit" | "delete" }) | undefined
|
|
>(undefined);
|
|
|
|
// updates issue modal
|
|
const [editIssueModal, setEditIssueModal] = useState(false);
|
|
const [issueToEdit, setIssueToEdit] = useState<
|
|
(IIssue & { actionType: "edit" | "delete" }) | undefined
|
|
>(undefined);
|
|
|
|
// delete issue modal
|
|
const [deleteIssueModal, setDeleteIssueModal] = useState(false);
|
|
const [issueToDelete, setIssueToDelete] = useState<IIssue | null>(null);
|
|
|
|
// trash box
|
|
const [trashBox, setTrashBox] = useState(false);
|
|
|
|
const router = useRouter();
|
|
const { workspaceSlug, projectId, cycleId, moduleId } = router.query;
|
|
|
|
const {
|
|
issueView,
|
|
groupedByIssues,
|
|
groupByProperty: selectedGroup,
|
|
orderBy,
|
|
} = useIssueView(issues);
|
|
|
|
const { data: stateGroups } = useSWR(
|
|
workspaceSlug && projectId ? STATE_LIST(projectId as string) : null,
|
|
workspaceSlug
|
|
? () => stateService.getStates(workspaceSlug as string, projectId as string)
|
|
: null
|
|
);
|
|
const states = getStatesList(stateGroups ?? {});
|
|
|
|
const { data: members } = useSWR(
|
|
projectId ? PROJECT_MEMBERS(projectId as string) : null,
|
|
workspaceSlug && projectId
|
|
? () => projectService.projectMembers(workspaceSlug as string, projectId as string)
|
|
: null
|
|
);
|
|
|
|
const handleDeleteIssue = useCallback(
|
|
(issue: IIssue) => {
|
|
setDeleteIssueModal(true);
|
|
setIssueToDelete(issue);
|
|
},
|
|
[setDeleteIssueModal, setIssueToDelete]
|
|
);
|
|
|
|
const handleOnDragEnd = useCallback(
|
|
(result: DropResult) => {
|
|
setTrashBox(false);
|
|
|
|
if (!result.destination || !workspaceSlug || !projectId) return;
|
|
|
|
const { source, destination } = result;
|
|
|
|
const draggedItem = groupedByIssues[source.droppableId][source.index];
|
|
|
|
if (destination.droppableId === "trashBox") {
|
|
handleDeleteIssue(draggedItem);
|
|
} else {
|
|
if (orderBy === "sort_order") {
|
|
let newSortOrder = draggedItem.sort_order;
|
|
|
|
const destinationGroupArray = groupedByIssues[destination.droppableId];
|
|
|
|
if (destinationGroupArray.length !== 0) {
|
|
// check if dropping in the same group
|
|
if (source.droppableId === destination.droppableId) {
|
|
// check if dropping at beginning
|
|
if (destination.index === 0)
|
|
newSortOrder = destinationGroupArray[0].sort_order - 10000;
|
|
// check if dropping at last
|
|
else if (destination.index === destinationGroupArray.length - 1)
|
|
newSortOrder =
|
|
destinationGroupArray[destinationGroupArray.length - 1].sort_order + 10000;
|
|
else {
|
|
if (destination.index > source.index)
|
|
newSortOrder =
|
|
(destinationGroupArray[source.index + 1].sort_order +
|
|
destinationGroupArray[source.index + 2].sort_order) /
|
|
2;
|
|
else if (destination.index < source.index)
|
|
newSortOrder =
|
|
(destinationGroupArray[source.index - 1].sort_order +
|
|
destinationGroupArray[source.index - 2].sort_order) /
|
|
2;
|
|
}
|
|
} else {
|
|
// check if dropping at beginning
|
|
if (destination.index === 0)
|
|
newSortOrder = destinationGroupArray[0].sort_order - 10000;
|
|
// check if dropping at last
|
|
else if (destination.index === destinationGroupArray.length)
|
|
newSortOrder =
|
|
destinationGroupArray[destinationGroupArray.length - 1].sort_order + 10000;
|
|
else
|
|
newSortOrder =
|
|
(destinationGroupArray[destination.index - 1].sort_order +
|
|
destinationGroupArray[destination.index].sort_order) /
|
|
2;
|
|
}
|
|
}
|
|
|
|
draggedItem.sort_order = newSortOrder;
|
|
}
|
|
|
|
if (orderBy === "sort_order" || source.droppableId !== destination.droppableId) {
|
|
const sourceGroup = source.droppableId; // source group id
|
|
const destinationGroup = destination.droppableId; // destination group id
|
|
|
|
if (!sourceGroup || !destinationGroup) return;
|
|
|
|
if (selectedGroup === "priority") draggedItem.priority = destinationGroup;
|
|
else if (selectedGroup === "state_detail.name") {
|
|
const destinationState = states?.find((s) => s.name === destinationGroup);
|
|
|
|
if (!destinationState) return;
|
|
|
|
draggedItem.state = destinationState.id;
|
|
draggedItem.state_detail = destinationState;
|
|
}
|
|
|
|
if (cycleId)
|
|
mutate<CycleIssueResponse[]>(
|
|
CYCLE_ISSUES(cycleId as string),
|
|
(prevData) => {
|
|
if (!prevData) return prevData;
|
|
const updatedIssues = prevData.map((issue) => {
|
|
if (issue.issue_detail.id === draggedItem.id) {
|
|
return {
|
|
...issue,
|
|
issue_detail: draggedItem,
|
|
};
|
|
}
|
|
return issue;
|
|
});
|
|
return [...updatedIssues];
|
|
},
|
|
false
|
|
);
|
|
|
|
if (moduleId)
|
|
mutate<ModuleIssueResponse[]>(
|
|
MODULE_ISSUES(moduleId as string),
|
|
(prevData) => {
|
|
if (!prevData) return prevData;
|
|
const updatedIssues = prevData.map((issue) => {
|
|
if (issue.issue_detail.id === draggedItem.id) {
|
|
return {
|
|
...issue,
|
|
issue_detail: draggedItem,
|
|
};
|
|
}
|
|
return issue;
|
|
});
|
|
return [...updatedIssues];
|
|
},
|
|
false
|
|
);
|
|
|
|
mutate<IIssue[]>(
|
|
PROJECT_ISSUES_LIST(workspaceSlug as string, projectId as string),
|
|
(prevData) => {
|
|
if (!prevData) return prevData;
|
|
|
|
const updatedIssues = prevData.map((i) => {
|
|
if (i.id === draggedItem.id) return draggedItem;
|
|
|
|
return i;
|
|
});
|
|
|
|
return updatedIssues;
|
|
},
|
|
false
|
|
);
|
|
|
|
// patch request
|
|
issuesService
|
|
.patchIssue(workspaceSlug as string, projectId as string, draggedItem.id, {
|
|
priority: draggedItem.priority,
|
|
state: draggedItem.state,
|
|
sort_order: draggedItem.sort_order,
|
|
})
|
|
.then((res) => {
|
|
if (cycleId) mutate(CYCLE_ISSUES(cycleId as string));
|
|
if (moduleId) mutate(MODULE_ISSUES(moduleId as string));
|
|
|
|
mutate(PROJECT_ISSUES_LIST(workspaceSlug as string, projectId as string));
|
|
});
|
|
}
|
|
}
|
|
},
|
|
[
|
|
workspaceSlug,
|
|
cycleId,
|
|
moduleId,
|
|
groupedByIssues,
|
|
projectId,
|
|
selectedGroup,
|
|
orderBy,
|
|
states,
|
|
handleDeleteIssue,
|
|
]
|
|
);
|
|
|
|
const addIssueToState = useCallback(
|
|
(groupTitle: string, stateId: string | null) => {
|
|
setCreateIssueModal(true);
|
|
if (selectedGroup)
|
|
setPreloadedData({
|
|
state: stateId ?? undefined,
|
|
[selectedGroup]: groupTitle,
|
|
actionType: "createIssue",
|
|
});
|
|
else setPreloadedData({ actionType: "createIssue" });
|
|
},
|
|
[setCreateIssueModal, setPreloadedData, selectedGroup]
|
|
);
|
|
|
|
const makeIssueCopy = useCallback(
|
|
(issue: IIssue) => {
|
|
setCreateIssueModal(true);
|
|
|
|
setPreloadedData({ ...issue, name: `${issue.name} (Copy)`, actionType: "createIssue" });
|
|
},
|
|
[setCreateIssueModal, setPreloadedData]
|
|
);
|
|
|
|
const handleEditIssue = useCallback(
|
|
(issue: IIssue) => {
|
|
setEditIssueModal(true);
|
|
setIssueToEdit({
|
|
...issue,
|
|
actionType: "edit",
|
|
cycle: issue.issue_cycle ? issue.issue_cycle.cycle : null,
|
|
module: issue.issue_module ? issue.issue_module.module : null,
|
|
});
|
|
},
|
|
[setEditIssueModal, setIssueToEdit]
|
|
);
|
|
|
|
const removeIssueFromCycle = useCallback(
|
|
(bridgeId: string) => {
|
|
if (!workspaceSlug || !projectId) return;
|
|
|
|
mutate<CycleIssueResponse[]>(
|
|
CYCLE_ISSUES(cycleId as string),
|
|
(prevData) => prevData?.filter((p) => p.id !== bridgeId),
|
|
false
|
|
);
|
|
|
|
issuesService
|
|
.removeIssueFromCycle(
|
|
workspaceSlug as string,
|
|
projectId as string,
|
|
cycleId as string,
|
|
bridgeId
|
|
)
|
|
.then((res) => {
|
|
console.log(res);
|
|
})
|
|
.catch((e) => {
|
|
console.log(e);
|
|
});
|
|
},
|
|
[workspaceSlug, projectId, cycleId]
|
|
);
|
|
|
|
const removeIssueFromModule = useCallback(
|
|
(bridgeId: string) => {
|
|
if (!workspaceSlug || !projectId) return;
|
|
|
|
mutate<ModuleIssueResponse[]>(
|
|
MODULE_ISSUES(moduleId as string),
|
|
(prevData) => prevData?.filter((p) => p.id !== bridgeId),
|
|
false
|
|
);
|
|
|
|
modulesService
|
|
.removeIssueFromModule(
|
|
workspaceSlug as string,
|
|
projectId as string,
|
|
moduleId as string,
|
|
bridgeId
|
|
)
|
|
.then((res) => {
|
|
console.log(res);
|
|
})
|
|
.catch((e) => {
|
|
console.log(e);
|
|
});
|
|
},
|
|
[workspaceSlug, projectId, moduleId]
|
|
);
|
|
|
|
const handleTrashBox = useCallback(
|
|
(isDragging: boolean) => {
|
|
if (isDragging && !trashBox) setTrashBox(true);
|
|
},
|
|
[trashBox, setTrashBox]
|
|
);
|
|
|
|
return (
|
|
<>
|
|
<CreateUpdateIssueModal
|
|
isOpen={createIssueModal && preloadedData?.actionType === "createIssue"}
|
|
handleClose={() => setCreateIssueModal(false)}
|
|
prePopulateData={{
|
|
...preloadedData,
|
|
}}
|
|
/>
|
|
<CreateUpdateIssueModal
|
|
isOpen={editIssueModal && issueToEdit?.actionType !== "delete"}
|
|
prePopulateData={{ ...issueToEdit }}
|
|
handleClose={() => setEditIssueModal(false)}
|
|
data={issueToEdit}
|
|
/>
|
|
<DeleteIssueModal
|
|
handleClose={() => setDeleteIssueModal(false)}
|
|
isOpen={deleteIssueModal}
|
|
data={issueToDelete}
|
|
/>
|
|
|
|
<div className="relative">
|
|
<DragDropContext onDragEnd={handleOnDragEnd}>
|
|
<StrictModeDroppable droppableId="trashBox">
|
|
{(provided, snapshot) => (
|
|
<div
|
|
className={`${
|
|
trashBox ? "pointer-events-auto opacity-100" : "pointer-events-none opacity-0"
|
|
} fixed top-9 right-9 z-20 flex h-28 w-96 items-center justify-center gap-2 rounded border-2 border-red-500 bg-red-100 p-3 text-xs font-medium italic text-red-500 ${
|
|
snapshot.isDraggingOver ? "bg-red-500 text-white" : ""
|
|
} duration-200`}
|
|
ref={provided.innerRef}
|
|
{...provided.droppableProps}
|
|
>
|
|
<TrashIcon className="h-4 w-4" />
|
|
Drop issue here to delete
|
|
</div>
|
|
)}
|
|
</StrictModeDroppable>
|
|
{issueView === "list" ? (
|
|
<AllLists
|
|
type={type}
|
|
issues={issues}
|
|
states={states}
|
|
members={members}
|
|
addIssueToState={addIssueToState}
|
|
makeIssueCopy={makeIssueCopy}
|
|
handleEditIssue={handleEditIssue}
|
|
handleDeleteIssue={handleDeleteIssue}
|
|
openIssuesListModal={type !== "issue" ? openIssuesListModal : null}
|
|
removeIssue={
|
|
type === "cycle"
|
|
? removeIssueFromCycle
|
|
: type === "module"
|
|
? removeIssueFromModule
|
|
: null
|
|
}
|
|
userAuth={userAuth}
|
|
/>
|
|
) : (
|
|
<AllBoards
|
|
type={type}
|
|
issues={issues}
|
|
states={states}
|
|
members={members}
|
|
addIssueToState={addIssueToState}
|
|
makeIssueCopy={makeIssueCopy}
|
|
handleEditIssue={handleEditIssue}
|
|
openIssuesListModal={type !== "issue" ? openIssuesListModal : null}
|
|
handleDeleteIssue={handleDeleteIssue}
|
|
handleTrashBox={handleTrashBox}
|
|
removeIssue={
|
|
type === "cycle"
|
|
? removeIssueFromCycle
|
|
: type === "module"
|
|
? removeIssueFromModule
|
|
: null
|
|
}
|
|
userAuth={userAuth}
|
|
/>
|
|
)}
|
|
</DragDropContext>
|
|
</div>
|
|
</>
|
|
);
|
|
};
|