2024-06-10 09:43:10 +00:00
|
|
|
"use client";
|
|
|
|
|
2024-05-21 10:55:57 +00:00
|
|
|
import { MutableRefObject, useEffect, useRef, useState } from "react";
|
|
|
|
import { combine } from "@atlaskit/pragmatic-drag-and-drop/combine";
|
|
|
|
import { dropTargetForElements } from "@atlaskit/pragmatic-drag-and-drop/element/adapter";
|
2024-06-10 14:45:03 +00:00
|
|
|
import { isNil } from "lodash";
|
2024-05-21 10:55:57 +00:00
|
|
|
import { observer } from "mobx-react";
|
|
|
|
import { cn } from "@plane/editor-core";
|
2024-06-10 14:45:03 +00:00
|
|
|
// plane packages
|
2024-05-21 10:55:57 +00:00
|
|
|
import {
|
|
|
|
IGroupByColumn,
|
|
|
|
TIssueMap,
|
2024-06-10 14:45:03 +00:00
|
|
|
TIssueGroupByOptions,
|
2024-05-21 10:55:57 +00:00
|
|
|
TIssueOrderByOptions,
|
2024-06-10 14:45:03 +00:00
|
|
|
TIssue,
|
|
|
|
IIssueDisplayProperties,
|
2024-05-21 10:55:57 +00:00
|
|
|
} from "@plane/types";
|
2024-06-10 14:45:03 +00:00
|
|
|
import { setToast, TOAST_TYPE } from "@plane/ui";
|
|
|
|
// components
|
|
|
|
import { ListLoaderItemRow } from "@/components/ui";
|
2024-05-21 10:55:57 +00:00
|
|
|
// constants
|
2024-06-10 14:45:03 +00:00
|
|
|
import { DRAG_ALLOWED_GROUPS } from "@/constants/issue";
|
2024-05-21 10:55:57 +00:00
|
|
|
// hooks
|
|
|
|
import { useProjectState } from "@/hooks/store";
|
2024-06-10 14:45:03 +00:00
|
|
|
import { useIntersectionObserver } from "@/hooks/use-intersection-observer";
|
|
|
|
import { useIssuesStore } from "@/hooks/use-issue-layout-store";
|
2024-06-04 05:42:24 +00:00
|
|
|
import { TSelectionHelper } from "@/hooks/use-multiple-select";
|
2024-05-21 10:55:57 +00:00
|
|
|
// components
|
|
|
|
import { GroupDragOverlay } from "../group-drag-overlay";
|
|
|
|
import {
|
|
|
|
GroupDropLocation,
|
|
|
|
getDestinationFromDropPayload,
|
|
|
|
getIssueBlockId,
|
|
|
|
getSourceFromDropPayload,
|
|
|
|
highlightIssueOnDrop,
|
|
|
|
} from "../utils";
|
|
|
|
import { IssueBlocksList } from "./blocks-list";
|
|
|
|
import { HeaderGroupByCard } from "./headers/group-by-card";
|
|
|
|
import { TRenderQuickActions } from "./list-view-types";
|
|
|
|
import { ListQuickAddIssueForm } from "./quick-add-issue-form";
|
|
|
|
|
2024-06-10 14:45:03 +00:00
|
|
|
interface Props {
|
|
|
|
groupIssueIds: string[] | undefined;
|
2024-05-21 10:55:57 +00:00
|
|
|
group: IGroupByColumn;
|
|
|
|
issuesMap: TIssueMap;
|
|
|
|
group_by: TIssueGroupByOptions | null;
|
|
|
|
orderBy: TIssueOrderByOptions | undefined;
|
|
|
|
getGroupIndex: (groupId: string | undefined) => number;
|
2024-06-10 14:45:03 +00:00
|
|
|
updateIssue: ((projectId: string | null, issueId: string, data: Partial<TIssue>) => Promise<void>) | undefined;
|
2024-05-21 10:55:57 +00:00
|
|
|
quickActions: TRenderQuickActions;
|
|
|
|
displayProperties: IIssueDisplayProperties | undefined;
|
|
|
|
enableIssueQuickAdd: boolean;
|
|
|
|
canEditProperties: (projectId: string | undefined) => boolean;
|
|
|
|
containerRef: MutableRefObject<HTMLDivElement | null>;
|
2024-06-10 14:45:03 +00:00
|
|
|
quickAddCallback?: ((projectId: string | null | undefined, data: TIssue) => Promise<TIssue | undefined>) | undefined;
|
2024-05-21 10:55:57 +00:00
|
|
|
handleOnDrop: (source: GroupDropLocation, destination: GroupDropLocation) => Promise<void>;
|
|
|
|
disableIssueCreation?: boolean;
|
|
|
|
addIssuesToView?: (issueIds: string[]) => Promise<TIssue>;
|
|
|
|
isCompletedCycle?: boolean;
|
2024-06-10 14:45:03 +00:00
|
|
|
showEmptyGroup?: boolean;
|
|
|
|
loadMoreIssues: (groupId?: string) => void;
|
2024-06-04 05:42:24 +00:00
|
|
|
selectionHelpers: TSelectionHelper;
|
2024-05-21 10:55:57 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
export const ListGroup = observer((props: Props) => {
|
|
|
|
const {
|
2024-06-10 14:45:03 +00:00
|
|
|
groupIssueIds,
|
2024-05-21 10:55:57 +00:00
|
|
|
group,
|
2024-06-10 14:45:03 +00:00
|
|
|
issuesMap,
|
2024-05-21 10:55:57 +00:00
|
|
|
group_by,
|
|
|
|
orderBy,
|
|
|
|
getGroupIndex,
|
|
|
|
updateIssue,
|
|
|
|
quickActions,
|
|
|
|
displayProperties,
|
2024-06-10 14:45:03 +00:00
|
|
|
enableIssueQuickAdd,
|
2024-05-21 10:55:57 +00:00
|
|
|
canEditProperties,
|
|
|
|
containerRef,
|
2024-06-10 14:45:03 +00:00
|
|
|
quickAddCallback,
|
2024-05-21 10:55:57 +00:00
|
|
|
handleOnDrop,
|
2024-06-10 14:45:03 +00:00
|
|
|
disableIssueCreation,
|
|
|
|
addIssuesToView,
|
2024-05-21 10:55:57 +00:00
|
|
|
isCompletedCycle,
|
2024-06-10 14:45:03 +00:00
|
|
|
showEmptyGroup,
|
|
|
|
loadMoreIssues,
|
2024-06-04 05:42:24 +00:00
|
|
|
selectionHelpers,
|
2024-05-21 10:55:57 +00:00
|
|
|
} = props;
|
|
|
|
|
|
|
|
const [isDraggingOverColumn, setIsDraggingOverColumn] = useState(false);
|
|
|
|
const [dragColumnOrientation, setDragColumnOrientation] = useState<"justify-start" | "justify-end">("justify-start");
|
2024-06-10 14:45:03 +00:00
|
|
|
const [isExpanded, setIsExpanded] = useState(true);
|
2024-05-21 10:55:57 +00:00
|
|
|
const groupRef = useRef<HTMLDivElement | null>(null);
|
|
|
|
|
|
|
|
const projectState = useProjectState();
|
|
|
|
|
2024-06-10 14:45:03 +00:00
|
|
|
const {
|
|
|
|
issues: { getGroupIssueCount, getPaginationData, getIssueLoader },
|
|
|
|
} = useIssuesStore();
|
|
|
|
|
|
|
|
const [intersectionElement, setIntersectionElement] = useState<HTMLDivElement | null>(null);
|
|
|
|
|
|
|
|
useIntersectionObserver(containerRef, intersectionElement, loadMoreIssues, `50% 0% 50% 0%`);
|
|
|
|
|
|
|
|
const groupIssueCount = getGroupIssueCount(group.id, undefined, false);
|
|
|
|
const nextPageResults = getPaginationData(group.id, undefined)?.nextPageResults;
|
|
|
|
const isPaginating = !!getIssueLoader(group.id);
|
|
|
|
|
|
|
|
const shouldLoadMore =
|
|
|
|
nextPageResults === undefined && groupIssueCount !== undefined && groupIssueIds
|
|
|
|
? groupIssueIds.length < groupIssueCount
|
|
|
|
: !!nextPageResults;
|
|
|
|
|
|
|
|
const loadMore = isPaginating ? (
|
|
|
|
<ListLoaderItemRow />
|
|
|
|
) : (
|
|
|
|
<div
|
|
|
|
className={
|
|
|
|
"h-11 relative flex items-center gap-3 bg-custom-background-100 border-t border-b-custom-border-200 pl-8 p-3 text-sm font-medium text-custom-text-350 hover:text-custom-text-300 hover:underline cursor-pointer"
|
|
|
|
}
|
|
|
|
onClick={() => loadMoreIssues(group.id)}
|
|
|
|
>
|
|
|
|
Load More ↓
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
|
|
|
|
const validateEmptyIssueGroups = (issueCount: number = 0) => {
|
|
|
|
if (!showEmptyGroup && issueCount <= 0) return false;
|
|
|
|
return true;
|
|
|
|
};
|
|
|
|
|
|
|
|
const toggleListGroup = () => {
|
|
|
|
setIsExpanded((prevState) => !prevState);
|
|
|
|
};
|
|
|
|
|
2024-05-21 10:55:57 +00:00
|
|
|
const prePopulateQuickAddData = (groupByKey: string | null, value: any) => {
|
|
|
|
const defaultState = projectState.projectStates?.find((state) => state.default);
|
|
|
|
let preloadedData: object = { state_id: defaultState?.id };
|
|
|
|
|
|
|
|
if (groupByKey === null) {
|
|
|
|
preloadedData = { ...preloadedData };
|
|
|
|
} else {
|
|
|
|
if (groupByKey === "state") {
|
|
|
|
preloadedData = { ...preloadedData, state_id: value };
|
|
|
|
} else if (groupByKey === "priority") {
|
|
|
|
preloadedData = { ...preloadedData, priority: value };
|
|
|
|
} else if (groupByKey === "labels" && value != "None") {
|
|
|
|
preloadedData = { ...preloadedData, label_ids: [value] };
|
|
|
|
} else if (groupByKey === "assignees" && value != "None") {
|
|
|
|
preloadedData = { ...preloadedData, assignee_ids: [value] };
|
|
|
|
} else if (groupByKey === "cycle" && value != "None") {
|
|
|
|
preloadedData = { ...preloadedData, cycle_id: value };
|
|
|
|
} else if (groupByKey === "module" && value != "None") {
|
|
|
|
preloadedData = { ...preloadedData, module_ids: [value] };
|
|
|
|
} else if (groupByKey === "created_by") {
|
|
|
|
preloadedData = { ...preloadedData };
|
|
|
|
} else {
|
|
|
|
preloadedData = { ...preloadedData, [groupByKey]: value };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return preloadedData;
|
|
|
|
};
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
const element = groupRef.current;
|
|
|
|
|
|
|
|
if (!element) return;
|
|
|
|
|
|
|
|
return combine(
|
|
|
|
dropTargetForElements({
|
|
|
|
element,
|
|
|
|
getData: () => ({ groupId: group.id, type: "COLUMN" }),
|
|
|
|
onDragEnter: () => {
|
|
|
|
setIsDraggingOverColumn(true);
|
|
|
|
},
|
|
|
|
onDragLeave: () => {
|
|
|
|
setIsDraggingOverColumn(false);
|
|
|
|
},
|
|
|
|
onDragStart: () => {
|
|
|
|
setIsDraggingOverColumn(true);
|
|
|
|
},
|
|
|
|
onDrag: ({ source }) => {
|
|
|
|
const sourceGroupId = source?.data?.groupId as string | undefined;
|
|
|
|
const currentGroupId = group.id;
|
|
|
|
|
|
|
|
const sourceIndex = getGroupIndex(sourceGroupId);
|
|
|
|
const currentIndex = getGroupIndex(currentGroupId);
|
|
|
|
|
|
|
|
if (sourceIndex > currentIndex) {
|
|
|
|
setDragColumnOrientation("justify-end");
|
|
|
|
} else {
|
|
|
|
setDragColumnOrientation("justify-start");
|
|
|
|
}
|
|
|
|
},
|
|
|
|
onDrop: (payload) => {
|
|
|
|
setIsDraggingOverColumn(false);
|
|
|
|
const source = getSourceFromDropPayload(payload);
|
|
|
|
const destination = getDestinationFromDropPayload(payload);
|
|
|
|
|
|
|
|
if (!source || !destination) return;
|
|
|
|
|
|
|
|
if (group.isDropDisabled) {
|
|
|
|
group.dropErrorMessage &&
|
|
|
|
setToast({
|
|
|
|
type: TOAST_TYPE.WARNING,
|
|
|
|
title: "Warning!",
|
|
|
|
message: group.dropErrorMessage,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
handleOnDrop(source, destination);
|
|
|
|
|
|
|
|
highlightIssueOnDrop(getIssueBlockId(source.id, destination?.groupId), orderBy !== "sort_order");
|
|
|
|
},
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}, [groupRef?.current, group, orderBy, getGroupIndex, setDragColumnOrientation, setIsDraggingOverColumn]);
|
|
|
|
|
|
|
|
const isDragAllowed = !!group_by && DRAG_ALLOWED_GROUPS.includes(group_by);
|
2024-05-22 11:45:37 +00:00
|
|
|
const canOverlayBeVisible = orderBy !== "sort_order" || !!group.isDropDisabled;
|
2024-05-21 10:55:57 +00:00
|
|
|
|
|
|
|
const isGroupByCreatedBy = group_by === "created_by";
|
2024-06-10 14:45:03 +00:00
|
|
|
const shouldExpand = (!!groupIssueCount && isExpanded) || !group_by;
|
2024-05-21 10:55:57 +00:00
|
|
|
|
2024-06-10 14:45:03 +00:00
|
|
|
return groupIssueIds && !isNil(groupIssueCount) && validateEmptyIssueGroups(groupIssueCount) ? (
|
2024-05-21 10:55:57 +00:00
|
|
|
<div
|
|
|
|
ref={groupRef}
|
|
|
|
className={cn(`relative flex flex-shrink-0 flex-col border-[1px] border-transparent`, {
|
2024-05-22 11:45:37 +00:00
|
|
|
"border-custom-primary-100": isDraggingOverColumn,
|
|
|
|
"border-custom-error-200": isDraggingOverColumn && !!group.isDropDisabled,
|
2024-05-21 10:55:57 +00:00
|
|
|
})}
|
|
|
|
>
|
2024-06-05 08:33:49 +00:00
|
|
|
<div className="sticky top-0 z-[2] w-full flex-shrink-0 border-b border-custom-border-200 bg-custom-background-90 pl-2 pr-3 py-1">
|
2024-05-21 10:55:57 +00:00
|
|
|
<HeaderGroupByCard
|
2024-06-04 05:42:24 +00:00
|
|
|
groupID={group.id}
|
2024-05-21 10:55:57 +00:00
|
|
|
icon={group.icon}
|
|
|
|
title={group.name || ""}
|
2024-06-10 14:45:03 +00:00
|
|
|
count={groupIssueCount}
|
2024-05-21 10:55:57 +00:00
|
|
|
issuePayload={group.payload}
|
2024-06-04 05:42:24 +00:00
|
|
|
canEditProperties={canEditProperties}
|
2024-05-21 10:55:57 +00:00
|
|
|
disableIssueCreation={disableIssueCreation || isGroupByCreatedBy || isCompletedCycle}
|
|
|
|
addIssuesToView={addIssuesToView}
|
2024-06-04 05:42:24 +00:00
|
|
|
selectionHelpers={selectionHelpers}
|
2024-06-10 14:45:03 +00:00
|
|
|
toggleListGroup={toggleListGroup}
|
2024-05-21 10:55:57 +00:00
|
|
|
/>
|
|
|
|
</div>
|
2024-06-10 14:45:03 +00:00
|
|
|
{shouldExpand && (
|
2024-05-21 10:55:57 +00:00
|
|
|
<div className="relative">
|
|
|
|
<GroupDragOverlay
|
|
|
|
dragColumnOrientation={dragColumnOrientation}
|
2024-05-22 11:45:37 +00:00
|
|
|
canOverlayBeVisible={canOverlayBeVisible}
|
2024-05-21 10:55:57 +00:00
|
|
|
isDropDisabled={!!group.isDropDisabled}
|
|
|
|
dropErrorMessage={group.dropErrorMessage}
|
|
|
|
orderBy={orderBy}
|
|
|
|
isDraggingOverColumn={isDraggingOverColumn}
|
|
|
|
/>
|
2024-06-10 14:45:03 +00:00
|
|
|
{groupIssueIds && (
|
2024-05-21 10:55:57 +00:00
|
|
|
<IssueBlocksList
|
2024-06-10 14:45:03 +00:00
|
|
|
issueIds={groupIssueIds}
|
2024-05-21 10:55:57 +00:00
|
|
|
groupId={group.id}
|
|
|
|
issuesMap={issuesMap}
|
|
|
|
updateIssue={updateIssue}
|
|
|
|
quickActions={quickActions}
|
|
|
|
displayProperties={displayProperties}
|
|
|
|
canEditProperties={canEditProperties}
|
|
|
|
containerRef={containerRef}
|
|
|
|
isDragAllowed={isDragAllowed}
|
2024-05-22 11:45:37 +00:00
|
|
|
canDropOverIssue={!canOverlayBeVisible}
|
2024-06-04 05:42:24 +00:00
|
|
|
selectionHelpers={selectionHelpers}
|
2024-05-21 10:55:57 +00:00
|
|
|
/>
|
|
|
|
)}
|
|
|
|
|
2024-06-10 14:45:03 +00:00
|
|
|
{shouldLoadMore && (group_by ? <>{loadMore}</> : <ListLoaderItemRow ref={setIntersectionElement} />)}
|
|
|
|
|
2024-05-21 10:55:57 +00:00
|
|
|
{enableIssueQuickAdd && !disableIssueCreation && !isGroupByCreatedBy && !isCompletedCycle && (
|
|
|
|
<div className="sticky bottom-0 z-[1] w-full flex-shrink-0">
|
|
|
|
<ListQuickAddIssueForm
|
|
|
|
prePopulatedData={prePopulateQuickAddData(group_by, group.id)}
|
|
|
|
quickAddCallback={quickAddCallback}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</div>
|
2024-06-10 14:45:03 +00:00
|
|
|
) : null;
|
2024-05-21 10:55:57 +00:00
|
|
|
});
|