From 640b6715685c3393bf2468d3eac1fd3197c09d76 Mon Sep 17 00:00:00 2001 From: Michael Peters Date: Thu, 30 Dec 2021 20:40:45 -0600 Subject: [PATCH] escape closes context menus --- .../webapp/elements/components/overlay.tsx | 2 +- .../contexts/components/context-menu.tsx | 2 +- .../webapp/elements/require/react-helper.tsx | 18 ++++++++++++++++-- 3 files changed, 18 insertions(+), 4 deletions(-) diff --git a/src/client/webapp/elements/components/overlay.tsx b/src/client/webapp/elements/components/overlay.tsx index a3e1700..e3aa0ce 100644 --- a/src/client/webapp/elements/components/overlay.tsx +++ b/src/client/webapp/elements/components/overlay.tsx @@ -15,7 +15,7 @@ const Overlay: FC = (props: OverlayProps) => { const { childRootRef, close, children } = props; if (childRootRef) { - ReactHelper.useCloseWhenClickedOrContextOutsideEffect(childRootRef, close); + ReactHelper.useCloseWhenEscapeOrClickedOrContextOutsideEffect(childRootRef, close); } const keyDownHandler = useCallback((e: KeyboardEvent) => { diff --git a/src/client/webapp/elements/contexts/components/context-menu.tsx b/src/client/webapp/elements/contexts/components/context-menu.tsx index 91fc403..9b808b9 100644 --- a/src/client/webapp/elements/contexts/components/context-menu.tsx +++ b/src/client/webapp/elements/contexts/components/context-menu.tsx @@ -17,7 +17,7 @@ const ContextMenu: FC = (props: ContextMenuProps) => { const rootRef = useRef(null); - ReactHelper.useCloseWhenClickedOrContextOutsideEffect(rootRef, close); + ReactHelper.useCloseWhenEscapeOrClickedOrContextOutsideEffect(rootRef, close); return ( diff --git a/src/client/webapp/elements/require/react-helper.tsx b/src/client/webapp/elements/require/react-helper.tsx index 8d29cbd..88e170d 100644 --- a/src/client/webapp/elements/require/react-helper.tsx +++ b/src/client/webapp/elements/require/react-helper.tsx @@ -342,7 +342,7 @@ export default class ReactHelper { } // Makes sure to also allow you to fly-out a click starting inside of the ref'd element but was dragged outside - static useCloseWhenClickedOrContextOutsideEffect(ref: RefObject, close: () => void) { + static useCloseWhenEscapeOrClickedOrContextOutsideEffect(ref: RefObject, close: () => void) { // Have to use a ref here and not states since we can't re-assign state between mouseup and click const mouseRef = useRef<{ mouseDownTarget: Node | null, mouseUpTarget: Node | null}>({ mouseDownTarget: null, mouseUpTarget: null }); @@ -384,7 +384,14 @@ export default class ReactHelper { if (ref.current.contains(event.target as Node)) return; // Context menu is fired on mouse-down so no need to do special checks. close(); - }, [ ref ]) + }, [ ref ]); + + const handleKeyDown = useCallback((event: KeyboardEvent) => { + if (!ref.current) return; + if (event.key !== 'Escape') return; + + close(); + }, [ ref ]); useEffect(() => { document.addEventListener('click', handleClickOutside); @@ -413,6 +420,13 @@ export default class ReactHelper { document.removeEventListener('contextmenu', handleContextMenu); } }, [ handleClickOutside ]); + + useEffect(() => { + document.addEventListener('keydown', handleKeyDown); + return () => { + document.removeEventListener('keydown', handleKeyDown); + } + }, [ handleKeyDown ]); } static useAlignment(