2020-05-25 09:27:36 -05:00
|
|
|
import React from 'react';
|
|
|
|
|
|
|
|
const LISTENER = {
|
|
|
|
ADD: 'add',
|
|
|
|
REMOVE: 'remove',
|
|
|
|
};
|
|
|
|
|
|
|
|
const DRAG_TYPES = {
|
|
|
|
END: 'dragend',
|
|
|
|
START: 'dragstart',
|
|
|
|
ENTER: 'dragenter',
|
|
|
|
LEAVE: 'dragleave',
|
|
|
|
};
|
|
|
|
|
|
|
|
const DRAG_SCORE = {
|
|
|
|
[DRAG_TYPES.ENTER]: 1,
|
|
|
|
[DRAG_TYPES.LEAVE]: -1,
|
|
|
|
};
|
|
|
|
|
|
|
|
const DRAG_STATE = {
|
|
|
|
[DRAG_TYPES.END]: false,
|
|
|
|
[DRAG_TYPES.START]: true,
|
|
|
|
};
|
|
|
|
|
|
|
|
// Returns simple detection for global drag-drop
|
2022-03-31 15:55:00 -04:00
|
|
|
export default function useDragDrop() {
|
2020-05-25 09:27:36 -05:00
|
|
|
const [drag, setDrag] = React.useState(false);
|
|
|
|
const [dropData, setDropData] = React.useState(null);
|
|
|
|
|
|
|
|
React.useEffect(() => {
|
|
|
|
let dragCount = 0;
|
|
|
|
let draggingElement = false;
|
|
|
|
|
|
|
|
// Handle file drop
|
2022-03-31 15:55:00 -04:00
|
|
|
const handleDropEvent = (event) => {
|
2020-05-25 09:27:36 -05:00
|
|
|
// Ignore non file types ( html elements / text )
|
|
|
|
if (!draggingElement) {
|
|
|
|
event.stopPropagation();
|
|
|
|
event.preventDefault();
|
|
|
|
// Get files
|
|
|
|
const files = event.dataTransfer.files;
|
|
|
|
// Check for files
|
|
|
|
if (files.length > 0) {
|
2022-03-31 15:55:00 -04:00
|
|
|
setDropData(event);
|
2020-05-25 09:27:36 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
// Reset state ( hide drop zone )
|
|
|
|
dragCount = 0;
|
|
|
|
setDrag(false);
|
|
|
|
};
|
|
|
|
|
|
|
|
// Drag event for non files type ( html elements / text )
|
2022-03-31 15:55:00 -04:00
|
|
|
const handleDragElementEvent = (event) => {
|
2020-05-25 09:27:36 -05:00
|
|
|
draggingElement = DRAG_STATE[event.type];
|
|
|
|
};
|
|
|
|
|
|
|
|
// Drag events
|
2022-03-31 15:55:00 -04:00
|
|
|
const handleDragEvent = (event) => {
|
2020-05-25 09:27:36 -05:00
|
|
|
event.stopPropagation();
|
|
|
|
event.preventDefault();
|
|
|
|
// Prevent multiple drop areas
|
|
|
|
dragCount += DRAG_SCORE[event.type];
|
|
|
|
// Dragged file enters the drop area
|
|
|
|
if (dragCount === 1 && !draggingElement && event.type === DRAG_TYPES.ENTER) {
|
|
|
|
setDrag(true);
|
|
|
|
}
|
|
|
|
// Dragged file leaves the drop area
|
|
|
|
if (dragCount === 0 && event.type === DRAG_TYPES.LEAVE) {
|
|
|
|
setDrag(false);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Register / Unregister listeners
|
2022-03-31 15:55:00 -04:00
|
|
|
const handleEventListeners = (event) => {
|
2020-05-25 09:27:36 -05:00
|
|
|
const action = `${event}EventListener`;
|
|
|
|
// Handle drop event
|
|
|
|
document[action]('drop', handleDropEvent);
|
|
|
|
// Handle drag events
|
|
|
|
document[action](DRAG_TYPES.ENTER, handleDragEvent);
|
|
|
|
document[action](DRAG_TYPES.LEAVE, handleDragEvent);
|
|
|
|
// Handle non files drag events
|
|
|
|
document[action](DRAG_TYPES.END, handleDragElementEvent);
|
|
|
|
document[action](DRAG_TYPES.START, handleDragElementEvent);
|
|
|
|
};
|
2022-03-31 15:55:00 -04:00
|
|
|
|
2020-05-25 09:27:36 -05:00
|
|
|
// On component mounted:
|
|
|
|
// Register event listeners
|
|
|
|
handleEventListeners(LISTENER.ADD);
|
|
|
|
|
|
|
|
// On component unmounted:
|
|
|
|
return () => {
|
|
|
|
// Unregister event listeners
|
|
|
|
handleEventListeners(LISTENER.REMOVE);
|
|
|
|
};
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
return { drag, dropData };
|
|
|
|
}
|