forked from github/plane
4c16ed8b23
* fix: calendar dnd for due dates before issue start date * chore: start date in calender view * fix: add existing issues to calendar layout --------- Co-authored-by: NarayanBavisetti <narayan3119@gmail.com>
213 lines
7.5 KiB
TypeScript
213 lines
7.5 KiB
TypeScript
import { useEffect, useRef, useState } from "react";
|
|
import { combine } from "@atlaskit/pragmatic-drag-and-drop/combine";
|
|
import { dropTargetForElements } from "@atlaskit/pragmatic-drag-and-drop/element/adapter";
|
|
import { differenceInCalendarDays } from "date-fns";
|
|
import { observer } from "mobx-react-lite";
|
|
// types
|
|
import { TGroupedIssues, TIssue, TIssueMap } from "@plane/types";
|
|
// ui
|
|
import { TOAST_TYPE, setToast } from "@plane/ui";
|
|
// components
|
|
import { CalendarIssueBlocks, ICalendarDate } from "@/components/issues";
|
|
import { highlightIssueOnDrop } from "@/components/issues/issue-layouts/utils";
|
|
// helpers
|
|
import { MONTHS_LIST } from "@/constants/calendar";
|
|
// helpers
|
|
import { cn } from "@/helpers/common.helper";
|
|
import { renderFormattedPayloadDate } from "@/helpers/date-time.helper";
|
|
// types
|
|
import { ICycleIssuesFilter } from "@/store/issue/cycle";
|
|
import { IModuleIssuesFilter } from "@/store/issue/module";
|
|
import { IProjectIssuesFilter } from "@/store/issue/project";
|
|
import { IProjectViewIssuesFilter } from "@/store/issue/project-views";
|
|
import { TRenderQuickActions } from "../list/list-view-types";
|
|
|
|
type Props = {
|
|
issuesFilterStore: IProjectIssuesFilter | IModuleIssuesFilter | ICycleIssuesFilter | IProjectViewIssuesFilter;
|
|
date: ICalendarDate;
|
|
issues: TIssueMap | undefined;
|
|
groupedIssueIds: TGroupedIssues;
|
|
quickActions: TRenderQuickActions;
|
|
enableQuickIssueCreate?: boolean;
|
|
disableIssueCreation?: boolean;
|
|
handleDragAndDrop: (
|
|
issueId: string | undefined,
|
|
sourceDate: string | undefined,
|
|
destinationDate: string | undefined
|
|
) => Promise<void>;
|
|
quickAddCallback?: (
|
|
workspaceSlug: string,
|
|
projectId: string,
|
|
data: TIssue,
|
|
viewId?: string
|
|
) => Promise<TIssue | undefined>;
|
|
addIssuesToView?: (issueIds: string[]) => Promise<any>;
|
|
viewId?: string;
|
|
readOnly?: boolean;
|
|
selectedDate: Date;
|
|
setSelectedDate: (date: Date) => void;
|
|
};
|
|
|
|
export const CalendarDayTile: React.FC<Props> = observer((props) => {
|
|
const {
|
|
issuesFilterStore,
|
|
date,
|
|
issues,
|
|
groupedIssueIds,
|
|
quickActions,
|
|
enableQuickIssueCreate,
|
|
disableIssueCreation,
|
|
quickAddCallback,
|
|
addIssuesToView,
|
|
viewId,
|
|
readOnly = false,
|
|
selectedDate,
|
|
handleDragAndDrop,
|
|
setSelectedDate,
|
|
} = props;
|
|
|
|
const [isDraggingOver, setIsDraggingOver] = useState(false);
|
|
const [showAllIssues, setShowAllIssues] = useState(false);
|
|
|
|
const calendarLayout = issuesFilterStore?.issueFilters?.displayFilters?.calendar?.layout ?? "month";
|
|
|
|
const formattedDatePayload = renderFormattedPayloadDate(date.date);
|
|
|
|
const dayTileRef = useRef<HTMLDivElement | null>(null);
|
|
|
|
useEffect(() => {
|
|
const element = dayTileRef.current;
|
|
|
|
if (!element) return;
|
|
|
|
return combine(
|
|
dropTargetForElements({
|
|
element,
|
|
getData: () => ({ date: formattedDatePayload }),
|
|
onDragEnter: () => {
|
|
setIsDraggingOver(true);
|
|
},
|
|
onDragLeave: () => {
|
|
setIsDraggingOver(false);
|
|
},
|
|
onDrop: ({ source, self }) => {
|
|
setIsDraggingOver(false);
|
|
const sourceData = source?.data as { id: string; date: string } | undefined;
|
|
const destinationData = self?.data as { date: string } | undefined;
|
|
if (!sourceData || !destinationData) return;
|
|
|
|
const issueDetails = issues?.[sourceData?.id];
|
|
if (issueDetails?.start_date) {
|
|
const issueStartDate = new Date(issueDetails.start_date);
|
|
const targetDate = new Date(destinationData?.date);
|
|
const diffInDays = differenceInCalendarDays(targetDate, issueStartDate);
|
|
if (diffInDays < 0) {
|
|
setToast({
|
|
type: TOAST_TYPE.ERROR,
|
|
title: "Error!",
|
|
message: "Due date cannot be before the start date of the issue.",
|
|
});
|
|
return;
|
|
}
|
|
}
|
|
|
|
handleDragAndDrop(sourceData?.id, sourceData?.date, destinationData?.date);
|
|
setShowAllIssues(true);
|
|
highlightIssueOnDrop(source?.element?.id, false);
|
|
},
|
|
})
|
|
);
|
|
}, [dayTileRef?.current, formattedDatePayload]);
|
|
|
|
if (!formattedDatePayload) return null;
|
|
const issueIdList = groupedIssueIds ? groupedIssueIds[formattedDatePayload] : null;
|
|
|
|
const totalIssues = issueIdList?.length ?? 0;
|
|
|
|
const isToday = date.date.toDateString() === new Date().toDateString();
|
|
const isSelectedDate = date.date.toDateString() == selectedDate.toDateString();
|
|
|
|
const isWeekend = [0, 6].includes(date.date.getDay());
|
|
const isMonthLayout = calendarLayout === "month";
|
|
|
|
const normalBackground = isWeekend ? "bg-custom-background-90" : "bg-custom-background-100";
|
|
const draggingOverBackground = isWeekend ? "bg-custom-background-80" : "bg-custom-background-90";
|
|
|
|
return (
|
|
<>
|
|
<div ref={dayTileRef} className="group relative flex h-full w-full flex-col bg-custom-background-90">
|
|
{/* header */}
|
|
<div
|
|
className={`hidden flex-shrink-0 items-center justify-end px-2 py-1.5 text-right text-xs md:flex ${
|
|
isMonthLayout // if month layout, highlight current month days
|
|
? date.is_current_month
|
|
? "font-medium"
|
|
: "text-custom-text-300"
|
|
: "font-medium" // if week layout, highlight all days
|
|
} ${isWeekend ? "bg-custom-background-90" : "bg-custom-background-100"} `}
|
|
>
|
|
{date.date.getDate() === 1 && MONTHS_LIST[date.date.getMonth() + 1].shortTitle + " "}
|
|
{isToday ? (
|
|
<span className="flex h-5 w-5 items-center justify-center rounded-full bg-custom-primary-100 text-white">
|
|
{date.date.getDate()}
|
|
</span>
|
|
) : (
|
|
<>{date.date.getDate()}</>
|
|
)}
|
|
</div>
|
|
|
|
{/* content */}
|
|
<div className="h-full w-full hidden md:block">
|
|
<div
|
|
className={cn(
|
|
`h-full w-full select-none ${isDraggingOver ? `${draggingOverBackground} opacity-70` : normalBackground}`,
|
|
{
|
|
"min-h-[5rem]": isMonthLayout,
|
|
}
|
|
)}
|
|
>
|
|
<CalendarIssueBlocks
|
|
date={date.date}
|
|
issues={issues}
|
|
issueIdList={issueIdList}
|
|
showAllIssues={showAllIssues}
|
|
setShowAllIssues={setShowAllIssues}
|
|
quickActions={quickActions}
|
|
isDragDisabled={readOnly}
|
|
addIssuesToView={addIssuesToView}
|
|
disableIssueCreation={disableIssueCreation}
|
|
enableQuickIssueCreate={enableQuickIssueCreate}
|
|
quickAddCallback={quickAddCallback}
|
|
viewId={viewId}
|
|
readOnly={readOnly}
|
|
isMonthLayout={isMonthLayout}
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Mobile view content */}
|
|
<div
|
|
onClick={() => setSelectedDate(date.date)}
|
|
className={cn(
|
|
"text-sm py-2.5 h-full w-full font-medium mx-auto flex flex-col justify-start items-center md:hidden cursor-pointer opacity-80",
|
|
{
|
|
"bg-custom-background-100": !isWeekend,
|
|
}
|
|
)}
|
|
>
|
|
<div
|
|
className={cn("size-6 flex items-center justify-center rounded-full", {
|
|
"bg-custom-primary-100 text-white": isSelectedDate,
|
|
"bg-custom-primary-100/10 text-custom-primary-100 ": isToday && !isSelectedDate,
|
|
})}
|
|
>
|
|
{date.date.getDate()}
|
|
</div>
|
|
|
|
{totalIssues > 0 && <div className="mt-1 size-1.5 flex flex-shrink-0 rounded bg-custom-primary-100" />}
|
|
</div>
|
|
</div>
|
|
</>
|
|
);
|
|
});
|