Skip to content

Commit

Permalink
Merge pull request Expensify#33864 from tienifr/fix/33820
Browse files Browse the repository at this point in the history
Fix: Drop file here modal does not dismiss when file is no longer dragged over the area
  • Loading branch information
tgolen authored Jan 3, 2024
2 parents 21894f4 + 1bae03b commit 3fa11a1
Showing 1 changed file with 6 additions and 13 deletions.
19 changes: 6 additions & 13 deletions src/hooks/useDragAndDrop.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import {useIsFocused} from '@react-navigation/native';
import React, {useCallback, useContext, useEffect, useRef, useState} from 'react';
import React, {useCallback, useContext, useEffect, useState} from 'react';
import {View} from 'react-native';
import {PopoverContext} from '@components/PopoverProvider';

Expand Down Expand Up @@ -30,18 +30,10 @@ export default function useDragAndDrop({dropZone, onDrop = () => {}, shouldAllow
const [isDraggingOver, setIsDraggingOver] = useState(false);
const {close: closePopover} = useContext(PopoverContext);

// This solution is borrowed from this SO: https://stackoverflow.com/questions/7110353/html5-dragleave-fired-when-hovering-a-child-element
// This is necessary because dragging over children will cause dragleave to execute on the parent.
// You can think of this counter as a stack. When a child is hovered over we push an element onto the stack.
// Then we only process the dragleave event if the count is 0, because it means that the last element (the parent) has been popped off the stack.
const dragCounter = useRef(0);

// If this component is out of focus or disabled, reset the drag state back to the default
useEffect(() => {
if (isFocused && !isDisabled) {
return;
}
dragCounter.current = 0;
setIsDraggingOver(false);
}, [isFocused, isDisabled]);

Expand Down Expand Up @@ -82,23 +74,24 @@ export default function useDragAndDrop({dropZone, onDrop = () => {}, shouldAllow
handleDragEvent(event);
break;
case DRAG_ENTER_EVENT:
dragCounter.current++;
handleDragEvent(event);
if (isDraggingOver) {
return;
}
setIsDraggingOver(true);
break;
case DRAG_LEAVE_EVENT:
dragCounter.current--;
if (!isDraggingOver || dragCounter.current > 0) {
if (!isDraggingOver) {
return;
}
// This is necessary because dragging over children will cause dragleave to execute on the parent.
if ((event.currentTarget as HTMLElement | null)?.contains(event.relatedTarget as HTMLElement | null)) {
return;
}

setIsDraggingOver(false);
break;
case DROP_EVENT:
dragCounter.current = 0;
setIsDraggingOver(false);
onDrop(event);
break;
Expand Down

0 comments on commit 3fa11a1

Please sign in to comment.