forked from github/plane
d8ab3e0087
* replace Pragmatic DND for calendar * remove unnecessary check
194 lines
6.9 KiB
TypeScript
194 lines
6.9 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 { observer } from "mobx-react-lite";
|
|
// types
|
|
import { TGroupedIssues, TIssue, TIssueMap } from "@plane/types";
|
|
// 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;
|
|
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 = date.date.getDay() === 0 || date.date.getDay() === 6;
|
|
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
|
|
} ${
|
|
date.date.getDay() === 0 || date.date.getDay() === 6
|
|
? "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={`h-full w-full select-none ${
|
|
isDraggingOver ? `${draggingOverBackground} opacity-70` : normalBackground
|
|
} ${isMonthLayout ? "min-h-[5rem]" : ""}`}
|
|
>
|
|
<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("flex h-6 w-6 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 flex h-1.5 w-1.5 flex-shrink-0 rounded bg-custom-primary-100" />}
|
|
</div>
|
|
</div>
|
|
</>
|
|
);
|
|
});
|