bigbuffet-rw/packages/pl-fe/src/hooks/useDraggedFiles.ts
marcin mikołajczak 4d5690d0c1 Switch to workspace
Signed-off-by: marcin mikołajczak <git@mkljczk.pl>
2024-08-28 13:01:08 +02:00

93 lines
3.1 KiB
TypeScript

import React, { useCallback, useEffect, useState } from 'react';
/** Controls the state of files being dragged over a node. */
const useDraggedFiles = <R extends HTMLElement>(node: React.RefObject<R>, onDrop?: (files: FileList) => void) => {
const [isDragging, setIsDragging] = useState(false);
const [isDraggedOver, setIsDraggedOver] = useState(false);
const handleDocumentDragEnter = useCallback((e: DragEvent) => {
if (isDraggingFiles(e)) {
setIsDragging(true);
}
}, [setIsDragging]);
const handleDocumentDragLeave = useCallback((e: DragEvent) => {
if (isDraggedOffscreen(e)) {
setIsDragging(false);
}
}, [setIsDragging]);
const handleDocumentDrop = useCallback((e: DragEvent) => {
setIsDragging(false);
setIsDraggedOver(false);
}, [setIsDragging]);
const handleDragEnter = useCallback((e: DragEvent) => {
if (isDraggingFiles(e)) {
setIsDraggedOver(true);
}
}, [setIsDraggedOver]);
const handleDragLeave = useCallback((e: DragEvent) => {
if (!node.current || isDraggedOutOfNode(e, node.current)) {
setIsDraggedOver(false);
}
}, [setIsDraggedOver]);
const handleDrop = useCallback((e: DragEvent) => {
if (isDraggingFiles(e) && onDrop) {
onDrop(e.dataTransfer.files);
}
setIsDragging(false);
setIsDraggedOver(false);
e.preventDefault();
}, [onDrop]);
useEffect(() => {
document.addEventListener('dragenter', handleDocumentDragEnter);
document.addEventListener('dragleave', handleDocumentDragLeave);
document.addEventListener('drop', handleDocumentDrop);
return () => {
document.removeEventListener('dragenter', handleDocumentDragEnter);
document.removeEventListener('dragleave', handleDocumentDragLeave);
document.removeEventListener('drop', handleDocumentDrop);
};
}, []);
useEffect(() => {
node.current?.addEventListener('dragenter', handleDragEnter);
node.current?.addEventListener('dragleave', handleDragLeave);
node.current?.addEventListener('drop', handleDrop);
return () => {
node.current?.removeEventListener('dragenter', handleDragEnter);
node.current?.removeEventListener('dragleave', handleDragLeave);
node.current?.removeEventListener('drop', handleDrop);
};
}, [node.current]);
return {
/** Whether the document is being dragged over. */
isDragging,
/** Whether the node is being dragged over. */
isDraggedOver,
};
};
/** Ensure only files are being dragged, and not eg highlighted text. */
const isDraggingFiles = (e: DragEvent): e is DragEvent & { dataTransfer: DataTransfer } => {
if (e.dataTransfer) {
const { types } = e.dataTransfer;
return types.length === 1 && types[0] === 'Files';
} else {
return false;
}
};
/** Check whether the cursor is in the screen. Mostly useful for dragleave events. */
const isDraggedOffscreen = (e: DragEvent): boolean => e.screenX === 0 && e.screenY === 0;
/** Check whether the cursor is dragged out of the node. */
const isDraggedOutOfNode = (e: DragEvent, node: Node): boolean =>
!node.contains(document.elementFromPoint(e.clientX, e.clientY));
export { useDraggedFiles };