From 483f6054521d5a37de6fd0135c7f2c4d80aa5872 Mon Sep 17 00:00:00 2001 From: Andrea Fercia Date: Wed, 12 Oct 2022 16:09:16 +0200 Subject: [PATCH] Refactor to make focusOnMount=true work as expected. --- .../src/hooks/use-focus-on-mount/index.js | 52 +++++++++++++------ 1 file changed, 36 insertions(+), 16 deletions(-) diff --git a/packages/compose/src/hooks/use-focus-on-mount/index.js b/packages/compose/src/hooks/use-focus-on-mount/index.js index 2703e57b599c54..c5176b17e9ceab 100644 --- a/packages/compose/src/hooks/use-focus-on-mount/index.js +++ b/packages/compose/src/hooks/use-focus-on-mount/index.js @@ -27,10 +27,37 @@ import { focus } from '@wordpress/dom'; */ export default function useFocusOnMount( focusOnMount = 'firstElement' ) { const focusOnMountRef = useRef( focusOnMount ); + + /** + * Sets focus on a DOM element. + * + * @param {HTMLElement} target The DOM element to set focus to. + * @return {void} + */ + const setFocus = ( target ) => { + target.focus( { + // When focusing newly mounted dialogs, + // the position of the popover is often not right on the first render + // This prevents the layout shifts when focusing the dialogs. + preventScroll: true, + } ); + }; + + /** @type {import('react').MutableRefObject | undefined>} */ + const timerId = useRef(); + useEffect( () => { focusOnMountRef.current = focusOnMount; }, [ focusOnMount ] ); + useEffect( () => { + return () => { + if ( timerId.current ) { + clearTimeout( timerId.current ); + } + }; + }, [] ); + return useCallback( ( node ) => { if ( ! node || focusOnMountRef.current === false ) { return; @@ -40,25 +67,18 @@ export default function useFocusOnMount( focusOnMount = 'firstElement' ) { return; } - if ( focusOnMountRef.current !== 'firstElement' ) { - node.focus( { - // When focusing newly mounted dialogs, - // the position of the popover is often not right on the first render - // This prevents the layout shifts when focusing the dialogs. - preventScroll: true, - } ); + if ( focusOnMountRef.current === 'firstElement' ) { + timerId.current = setTimeout( () => { + const firstTabbable = focus.tabbable.find( node )[ 0 ]; + + if ( firstTabbable ) { + setFocus( /** @type {HTMLElement} */ ( firstTabbable ) ); + } + }, 0 ); return; } - const focusTimeout = setTimeout( () => { - const firstTabbable = focus.tabbable.find( node )[ 0 ]; - - if ( firstTabbable ) { - /** @type {HTMLElement} */ ( firstTabbable ).focus(); - } - }, 0 ); - - return () => clearTimeout( focusTimeout ); + setFocus( node ); }, [] ); }