2024-02-23 13:40:45 +00:00
|
|
|
import { observer } from "mobx-react";
|
|
|
|
// helpers
|
2024-03-19 14:38:35 +00:00
|
|
|
import { cn } from "@/helpers/common.helper";
|
2024-03-20 08:14:08 +00:00
|
|
|
import { getDate, renderFormattedPayloadDate } from "@/helpers/date-time.helper";
|
2024-06-04 05:42:24 +00:00
|
|
|
// hooks
|
2024-03-19 14:38:35 +00:00
|
|
|
import { useIssueDetail } from "@/hooks/store";
|
2024-06-04 05:42:24 +00:00
|
|
|
import { TSelectionHelper } from "@/hooks/use-multiple-select";
|
2024-02-23 13:40:45 +00:00
|
|
|
// constants
|
|
|
|
import { BLOCK_HEIGHT } from "../constants";
|
2024-06-04 05:42:24 +00:00
|
|
|
// components
|
2024-03-06 13:09:14 +00:00
|
|
|
import { ChartAddBlock, ChartDraggable } from "../helpers";
|
|
|
|
import { useGanttChart } from "../hooks";
|
2024-06-10 14:45:03 +00:00
|
|
|
import { ChartDataType, IBlockUpdateData, IGanttBlock } from "../types";
|
2024-02-23 13:40:45 +00:00
|
|
|
|
|
|
|
type Props = {
|
2024-06-10 14:45:03 +00:00
|
|
|
blockId: string;
|
|
|
|
getBlockById: (id: string, currentViewData?: ChartDataType | undefined) => IGanttBlock;
|
|
|
|
showAllBlocks: boolean;
|
2024-02-23 13:40:45 +00:00
|
|
|
blockToRender: (data: any) => React.ReactNode;
|
|
|
|
blockUpdateHandler: (block: any, payload: IBlockUpdateData) => void;
|
|
|
|
enableBlockLeftResize: boolean;
|
|
|
|
enableBlockRightResize: boolean;
|
|
|
|
enableBlockMove: boolean;
|
|
|
|
enableAddBlock: boolean;
|
|
|
|
ganttContainerRef: React.RefObject<HTMLDivElement>;
|
2024-06-04 05:42:24 +00:00
|
|
|
selectionHelpers: TSelectionHelper;
|
2024-02-23 13:40:45 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
export const GanttChartBlock: React.FC<Props> = observer((props) => {
|
|
|
|
const {
|
2024-06-10 14:45:03 +00:00
|
|
|
blockId,
|
|
|
|
getBlockById,
|
|
|
|
showAllBlocks,
|
2024-02-23 13:40:45 +00:00
|
|
|
blockToRender,
|
|
|
|
blockUpdateHandler,
|
|
|
|
enableBlockLeftResize,
|
|
|
|
enableBlockRightResize,
|
|
|
|
enableBlockMove,
|
|
|
|
enableAddBlock,
|
|
|
|
ganttContainerRef,
|
2024-06-04 05:42:24 +00:00
|
|
|
selectionHelpers,
|
2024-02-23 13:40:45 +00:00
|
|
|
} = props;
|
|
|
|
// store hooks
|
2024-06-10 14:45:03 +00:00
|
|
|
const { currentViewData, updateActiveBlockId, isBlockActive } = useGanttChart();
|
2024-04-30 11:50:02 +00:00
|
|
|
const { getIsIssuePeeked } = useIssueDetail();
|
2024-02-23 13:40:45 +00:00
|
|
|
|
2024-06-10 14:45:03 +00:00
|
|
|
const block = getBlockById(blockId, currentViewData);
|
|
|
|
|
|
|
|
// hide the block if it doesn't have start and target dates and showAllBlocks is false
|
|
|
|
if (!block || (!showAllBlocks && !(block.start_date && block.target_date))) return null;
|
|
|
|
|
2024-02-23 13:40:45 +00:00
|
|
|
const isBlockVisibleOnChart = block.start_date && block.target_date;
|
|
|
|
|
|
|
|
const handleChartBlockPosition = (
|
|
|
|
block: IGanttBlock,
|
|
|
|
totalBlockShifts: number,
|
|
|
|
dragDirection: "left" | "right" | "move"
|
|
|
|
) => {
|
2024-03-20 08:14:08 +00:00
|
|
|
const originalStartDate = getDate(block.start_date);
|
|
|
|
const originalTargetDate = getDate(block.target_date);
|
2024-02-23 13:40:45 +00:00
|
|
|
|
2024-03-20 08:14:08 +00:00
|
|
|
if (!originalStartDate || !originalTargetDate) return;
|
2024-02-23 13:40:45 +00:00
|
|
|
|
2024-03-20 08:14:08 +00:00
|
|
|
const updatedStartDate = new Date(originalStartDate);
|
2024-02-23 13:40:45 +00:00
|
|
|
const updatedTargetDate = new Date(originalTargetDate);
|
|
|
|
|
|
|
|
// update the start date on left resize
|
|
|
|
if (dragDirection === "left") updatedStartDate.setDate(originalStartDate.getDate() - totalBlockShifts);
|
|
|
|
// update the target date on right resize
|
|
|
|
else if (dragDirection === "right") updatedTargetDate.setDate(originalTargetDate.getDate() + totalBlockShifts);
|
|
|
|
// update both the dates on x-axis move
|
|
|
|
else if (dragDirection === "move") {
|
|
|
|
updatedStartDate.setDate(originalStartDate.getDate() + totalBlockShifts);
|
|
|
|
updatedTargetDate.setDate(originalTargetDate.getDate() + totalBlockShifts);
|
|
|
|
}
|
|
|
|
|
|
|
|
// call the block update handler with the updated dates
|
|
|
|
blockUpdateHandler(block.data, {
|
|
|
|
start_date: renderFormattedPayloadDate(updatedStartDate) ?? undefined,
|
|
|
|
target_date: renderFormattedPayloadDate(updatedTargetDate) ?? undefined,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2024-06-10 14:45:03 +00:00
|
|
|
if (!block.data) return null;
|
|
|
|
|
2024-06-04 05:42:24 +00:00
|
|
|
const isBlockSelected = selectionHelpers.getIsEntitySelected(block.id);
|
|
|
|
const isBlockFocused = selectionHelpers.getIsEntityActive(block.id);
|
|
|
|
const isBlockHoveredOn = isBlockActive(block.id);
|
|
|
|
|
2024-02-23 13:40:45 +00:00
|
|
|
return (
|
|
|
|
<div
|
|
|
|
className="relative min-w-full w-max"
|
|
|
|
style={{
|
|
|
|
height: `${BLOCK_HEIGHT}px`,
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
className={cn("relative h-full", {
|
2024-06-04 05:42:24 +00:00
|
|
|
"rounded-l border border-r-0 border-custom-primary-70": getIsIssuePeeked(block.data.id),
|
|
|
|
"bg-custom-background-90": isBlockHoveredOn,
|
|
|
|
"bg-custom-primary-100/5 hover:bg-custom-primary-100/10": isBlockSelected,
|
|
|
|
"bg-custom-primary-100/10": isBlockSelected && isBlockHoveredOn,
|
|
|
|
"border border-r-0 border-custom-border-400": isBlockFocused,
|
2024-02-23 13:40:45 +00:00
|
|
|
})}
|
2024-06-10 14:45:03 +00:00
|
|
|
onMouseEnter={() => updateActiveBlockId(blockId)}
|
2024-02-23 13:40:45 +00:00
|
|
|
onMouseLeave={() => updateActiveBlockId(null)}
|
|
|
|
>
|
|
|
|
{isBlockVisibleOnChart ? (
|
|
|
|
<ChartDraggable
|
|
|
|
block={block}
|
|
|
|
blockToRender={blockToRender}
|
|
|
|
handleBlock={(...args) => handleChartBlockPosition(block, ...args)}
|
|
|
|
enableBlockLeftResize={enableBlockLeftResize}
|
|
|
|
enableBlockRightResize={enableBlockRightResize}
|
|
|
|
enableBlockMove={enableBlockMove}
|
|
|
|
ganttContainerRef={ganttContainerRef}
|
|
|
|
/>
|
|
|
|
) : (
|
|
|
|
enableAddBlock && <ChartAddBlock block={block} blockUpdateHandler={blockUpdateHandler} />
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
});
|