PromucFlow_constructor/app/client/src/components/editorComponents/DragLayerComponent.tsx

174 lines
4.8 KiB
TypeScript
Raw Normal View History

import React, {
useContext,
useEffect,
useLayoutEffect,
RefObject,
useRef,
} from "react";
2019-09-21 01:52:38 +00:00
import styled from "styled-components";
import { useDragLayer, XYCoord } from "react-dnd";
import DropZone from "./Dropzone";
2020-01-16 11:46:21 +00:00
import { noCollision, currentDropRow } from "utils/WidgetPropsUtils";
import { OccupiedSpace } from "constants/editorConstants";
import { CONTAINER_GRID_PADDING } from "constants/WidgetConstants";
2020-01-16 11:46:21 +00:00
import { DropTargetContext } from "./DropTargetComponent";
import { scrollElementIntoParentCanvasView } from "utils/helpers";
import { getNearestParentCanvas } from "utils/generators";
const WrappedDragLayer = styled.div<{
columnWidth: number;
rowHeight: number;
ref: RefObject<HTMLDivElement>;
}>`
2019-09-21 01:52:38 +00:00
position: absolute;
pointer-events: none;
left: 0;
top: 0;
left: ${CONTAINER_GRID_PADDING}px;
top: ${CONTAINER_GRID_PADDING}px;
height: calc(100% - ${CONTAINER_GRID_PADDING}px);
width: calc(100% - ${CONTAINER_GRID_PADDING}px);
background-image: radial-gradient(
circle,
${props => props.theme.colors.grid} 2px,
transparent 0
);
background-size: ${props => props.columnWidth}px ${props => props.rowHeight}px;
background-position: -${props => props.columnWidth / 2}px -${props =>
props.rowHeight / 2}px;
2019-09-21 01:52:38 +00:00
`;
type DragLayerProps = {
parentRowHeight: number;
canDropTargetExtend: boolean;
parentColumnWidth: number;
2019-09-21 01:52:38 +00:00
visible: boolean;
occupiedSpaces?: OccupiedSpace[];
onBoundsUpdate: Function;
isOver: boolean;
2019-10-08 06:19:10 +00:00
parentRows?: number;
parentCols?: number;
2019-10-08 12:31:58 +00:00
isResizing?: boolean;
2020-01-16 11:46:21 +00:00
parentWidgetId: string;
2020-05-14 06:06:20 +00:00
force: boolean;
2019-09-21 01:52:38 +00:00
};
const DragLayerComponent = (props: DragLayerProps) => {
2020-01-16 11:46:21 +00:00
const { updateDropTargetRows } = useContext(DropTargetContext);
const dropTargetMask: RefObject<HTMLDivElement> = React.useRef(null);
const dropZoneRef = React.useRef<HTMLDivElement>(null);
useEffect(() => {
const el = dropZoneRef.current;
const scrollParent: Element | null = getNearestParentCanvas(
dropTargetMask.current,
);
if (dropTargetMask.current) {
if (el && props.canDropTargetExtend) {
scrollElementIntoParentCanvasView(el, scrollParent);
}
}
});
const dropTargetOffset = useRef({
x: 0,
y: 0,
});
2019-09-25 21:21:04 +00:00
const { isDragging, currentOffset, widget, canDrop } = useDragLayer(
monitor => ({
isDragging: monitor.isDragging(),
currentOffset: monitor.getSourceClientOffset(),
2019-09-25 21:21:04 +00:00
widget: monitor.getItem(),
canDrop: noCollision(
monitor.getSourceClientOffset() as XYCoord,
2019-09-25 21:21:04 +00:00
props.parentColumnWidth,
props.parentRowHeight,
monitor.getItem(),
dropTargetOffset.current,
2019-09-25 21:21:04 +00:00
props.occupiedSpaces,
2019-10-08 06:19:10 +00:00
props.parentRows,
props.parentCols,
2019-09-25 21:21:04 +00:00
),
}),
);
2020-01-16 11:46:21 +00:00
if (
currentOffset &&
props.isOver &&
props.canDropTargetExtend &&
isDragging
2020-01-16 11:46:21 +00:00
) {
const row = currentDropRow(
props.parentRowHeight,
dropTargetOffset.current.y,
2020-01-16 11:46:21 +00:00
currentOffset.y,
widget,
);
updateDropTargetRows && updateDropTargetRows(widget.widgetId, row);
2020-01-16 11:46:21 +00:00
}
let widgetWidth = 0;
let widgetHeight = 0;
if (widget) {
widgetWidth = widget.columns
? widget.columns
: widget.rightColumn - widget.leftColumn;
widgetHeight = widget.rows ? widget.rows : widget.bottomRow - widget.topRow;
}
useLayoutEffect(() => {
const el = dropTargetMask.current;
if (el) {
const rect = el.getBoundingClientRect();
if (
rect.x !== dropTargetOffset.current.x ||
rect.y !== dropTargetOffset.current.y
) {
dropTargetOffset.current = { x: rect.x, y: rect.y };
props.onBoundsUpdate && props.onBoundsUpdate(rect);
}
}
});
2020-05-14 06:06:20 +00:00
if (
(!isDragging || !props.visible || !props.isOver) &&
!props.force &&
!props.isResizing
) {
2019-09-21 01:52:38 +00:00
return null;
}
2020-01-16 11:46:21 +00:00
2020-02-11 09:56:21 +00:00
/*
When the parent offsets are not updated, we don't need to show the dropzone, as the dropzone
will be rendered at an incorrect coordinates.
We can be sure that the parent offset has been calculated
when the coordiantes are not [0,0].
*/
const isParentOffsetCalculated = dropTargetOffset.current.x !== 0;
2020-02-11 09:56:21 +00:00
2019-09-21 01:52:38 +00:00
return (
<WrappedDragLayer
columnWidth={props.parentColumnWidth}
rowHeight={props.parentRowHeight}
ref={dropTargetMask}
>
2020-02-11 09:56:21 +00:00
{props.visible &&
props.isOver &&
currentOffset &&
isParentOffsetCalculated && (
<DropZone
parentOffset={dropTargetOffset.current}
2020-02-11 09:56:21 +00:00
parentRowHeight={props.parentRowHeight}
parentColumnWidth={props.parentColumnWidth}
width={widgetWidth}
height={widgetHeight}
currentOffset={currentOffset as XYCoord}
canDrop={canDrop}
ref={dropZoneRef}
2020-02-11 09:56:21 +00:00
/>
)}
2019-09-21 01:52:38 +00:00
</WrappedDragLayer>
);
};
export default DragLayerComponent;