Focus management in React with Redux

Intro Controlling focus in a React app isn’t straightforward, but very important for accessibility and usability in general. Keyboard users rely heavily on predictable focus management. There are several possible approaches. Personally, I’ve come to like the 3rd option. autoFocus attribute eslint has an opinion and suggests you don’t use it (jsx-a11y/no-autofocus), but if you know what you’re doing, there are definitely situations where this is viable. relies on elements entering the DOM, doesn’t work when toggling visibility with css. forwardRef and useImperativeHandle lets a child expose a method (e.g. focusButton()) that a parent call on demand very explicit, lots of boilerplate and quickly become hard to follow redux or other state managers requires a “focus” state prop, ideally an object so it’s new every time works well with a custom hook to listen for changes and apply the focus AutoFocus The simplest method in terms of amount of code. Simply use to make the element gain focus when it mounts. Works well when… You have a conditionally rendered component (or element inside that component) that, when shown, always needs to gain focus. Example use case For a simple example, imagine a form that requires an extra confirmation before it’s sent to the server. So pressing Enter in any field (or clicking the 1st submit button) would ask “Are you sure?”. There would be Confirm and Cancel buttons. The form looks something like this: // TheForm.tsx import { useState } from 'react' import { ConfirmControls, Fields, SubmitButton } from './components' // details omitted for brevity const TheForm = () => { const [atConfirmStep, setAtConfirmStep] = useState(false) // logic to toggle atConfirmStep omitted for brevity const submitBtnRef = useRef(null) const handleCancelConfirmation = useCallback(() => { submitBtnRef.current?.myCustomFocus() }, []) return ( {!atConfirmStep && } {atConfirmStep && } ) } export default TheForm Upon triggering the SubmitButton, a screen reader would read “Are you sure?” and focus would automatically move to the ConfirmButton. Since the ConfirmButton is only rendered when atConfirmStep is true, the autoFocus attribute works. The button gains focus when the component mounts. Tab would focus the CancelButton. Canceling would reset atConfirmStep to false, replacing the ConfirmControls with the SubmitButton. For easier keyboard navigation, it makes sense to let users hit Escape to cancel out of the confirm step. When canceled, since the ConfirmControls are removed from DOM, the document body would gain focus. That’s not ideal, so let’s go the extra mile and focus the SubmitButton instead. Keyboard / Assistive Tech users will appreciate that. React.forwardRef with React.useImperativeHandle To move focus to the SubmitButton, we can’t use autoFocus, because that would focus the button immediately after page load. That’s a terrible experience. So how do we focus the SubmitButton when the user cancels out of the ConfirmControls? One option is to make Form call a focus() method on the SubmitButton. Yes, that’s a parent calling a method on a child component. The flow is this: ConfirmControl is canceled, invokes onCancel (received as prop from Form). Form has a ref pointing to SubmitButton. When handleCancelConfirmation is invoked, run submitButtonRef.myCustomFocus(). “myCustomFocus()” is a method of SubmitButton, made available through useImperativeHandle (details omitted, see useImperativeHandle). The myCustomFocus() method now does something like buttonRef.focus(), where buttonRef points to the actual html element. This works, but it has downsides: It requires a lot of boilerplate with the forwardRef and useImperativeHandle. The flow isn’t straightforward to follow. It becomes increasingly difficult to manage when additional components are involved, e.g. if SomeComponent is sat between Form and SubmitButton. In this case, Form would trigger a method on SomeComponent, which in turn would trigger the myCustomFocus() method on SubmitButton. Now, both SomeComponent and SubmitButton need use forwardRef and useImperativeHandle. That’s a lot of hard to read code just for setting focus. There’s a simpler approach: Using a state manager. Redux to set focus Using redux, or any other state manager, we can simplify the approach and make it much easier to read and maintain. Note: One might argue that redux is for managing state declaratively, and using it imperatively to achieve something such as setting focus isn’t what it’s meant for. I like to be pragmatic about it. As you’ll see, managing focus becomes super easy and readable. That’s ultimately what matters. For this approach, you need: A slice of state where you add a “focus” field. An action+reducer to set the focus. In your component with focusable element(s): A ref to any fo

Feb 13, 2025 - 16:36
 0
Focus management in React with Redux

Intro

Controlling focus in a React app isn’t straightforward, but very important for accessibility and usability in general. Keyboard users rely heavily on predictable focus management.

There are several possible approaches. Personally, I’ve come to like the 3rd option.

  • autoFocus attribute
    • eslint has an opinion and suggests you don’t use it (jsx-a11y/no-autofocus), but if you know what you’re doing, there are definitely situations where this is viable.
    • relies on elements entering the DOM, doesn’t work when toggling visibility with css.
  • forwardRef and useImperativeHandle
    • lets a child expose a method (e.g. focusButton()) that a parent call on demand
    • very explicit, lots of boilerplate and quickly become hard to follow
  • redux or other state managers
    • requires a “focus” state prop, ideally an object so it’s new every time
    • works well with a custom hook to listen for changes and apply the focus

AutoFocus

The simplest method in terms of amount of code. Simply use