Add useDraggedFiles hook
This commit is contained in:
parent
7ea9f0ad86
commit
28c8f1dbd6
|
@ -6,6 +6,7 @@ export { useBackend } from './useBackend';
|
||||||
export { useClickOutside } from './useClickOutside';
|
export { useClickOutside } from './useClickOutside';
|
||||||
export { useCompose } from './useCompose';
|
export { useCompose } from './useCompose';
|
||||||
export { useDebounce } from './useDebounce';
|
export { useDebounce } from './useDebounce';
|
||||||
|
export { useDraggedFiles } from './useDraggedFiles';
|
||||||
export { useGetState } from './useGetState';
|
export { useGetState } from './useGetState';
|
||||||
export { useGroupsPath } from './useGroupsPath';
|
export { useGroupsPath } from './useGroupsPath';
|
||||||
export { useDimensions } from './useDimensions';
|
export { useDimensions } from './useDimensions';
|
||||||
|
|
|
@ -0,0 +1,70 @@
|
||||||
|
import React, { useCallback, useEffect, useState } from 'react';
|
||||||
|
|
||||||
|
/** Controls the state of files being dragged over a node. */
|
||||||
|
function useDraggedFiles<R extends HTMLElement>(node: React.RefObject<R>, onDrop?: (files: FileList) => void) {
|
||||||
|
const [isDragging, setIsDragging] = useState(false);
|
||||||
|
|
||||||
|
const handleDocumentDragEnter = useCallback((e: DragEvent) => {
|
||||||
|
if (isDraggingFiles(e)) {
|
||||||
|
setIsDragging(true);
|
||||||
|
}
|
||||||
|
}, [setIsDragging]);
|
||||||
|
|
||||||
|
const handleDocumentDragLeave = useCallback((e: DragEvent) => {
|
||||||
|
if (isOffscreen(e)) {
|
||||||
|
setIsDragging(false);
|
||||||
|
}
|
||||||
|
}, [setIsDragging]);
|
||||||
|
|
||||||
|
const handleDocumentDrop = useCallback((e: DragEvent) => {
|
||||||
|
setIsDragging(false);
|
||||||
|
}, [setIsDragging]);
|
||||||
|
|
||||||
|
const handleDrop = useCallback((e: DragEvent) => {
|
||||||
|
if (isDraggingFiles(e) && onDrop) {
|
||||||
|
onDrop(e.dataTransfer.files);
|
||||||
|
}
|
||||||
|
setIsDragging(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('drop', handleDrop);
|
||||||
|
return () => {
|
||||||
|
node.current?.removeEventListener('drop', handleDrop);
|
||||||
|
};
|
||||||
|
}, [node.current]);
|
||||||
|
|
||||||
|
return {
|
||||||
|
/** Whether the document is being dragged over. */
|
||||||
|
isDragging,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Ensure only files are being dragged, and not eg highlighted text. */
|
||||||
|
function 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. */
|
||||||
|
function isOffscreen(e: DragEvent): boolean {
|
||||||
|
return e.screenX === 0 && e.screenY === 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
export { useDraggedFiles };
|
Loading…
Reference in New Issue