diff --git a/packages/ui-react/src/components/Modal/Modal.tsx b/packages/ui-react/src/components/Modal/Modal.tsx index e135c52b6..a8eb651c5 100644 --- a/packages/ui-react/src/components/Modal/Modal.tsx +++ b/packages/ui-react/src/components/Modal/Modal.tsx @@ -40,6 +40,7 @@ const Modal: React.FC = ({ ...ariaAttributes }: Props) => { const modalRef = useRef() as React.MutableRefObject; + const mouseEventTargetsPairRef = useRef>({ downTarget: null, upTarget: null }); const { handleOpen, handleClose } = useModal(); @@ -89,6 +90,15 @@ const Modal: React.FC = ({ }, [openModalEvent, closeModalEvent, open]); const clickHandler: ReactEventHandler = (event) => { + const { + current: { downTarget, upTarget }, + } = mouseEventTargetsPairRef; + + // modal should only close if both mousedown and mouseup are done on the overlay + if (downTarget !== upTarget) { + return; + } + // Backdrop click (the dialog itself) will close the modal if (event.target === modalRef.current) { onClose?.(); @@ -102,6 +112,12 @@ const Modal: React.FC = ({ onKeyDown={keyDownHandler} onClose={closeHandler} onClick={clickHandler} + onMouseDown={(e) => { + mouseEventTargetsPairRef.current.downTarget = e.target as HTMLElement; + }} + onMouseUp={(e) => { + mouseEventTargetsPairRef.current.upTarget = e.target as HTMLElement; + }} ref={modalRef} role={role} {...ariaAttributes}