useFocusZone
Designates a container where focus can be moved using keys other than 'Tab'.
This hook is useful for implementing many of the patterns described in Section 6 of the WAI-ARIA Authoring Practices document. The most common use case of this behavior is to allow arrow keys (up and down or left and right) to move focus between related elements, such as items in a menu.
At a high level, useFocusZone
works by adjusting the tabindex
attribute on focusable elements and setting up event listeners on the container that respond to the relevant key presses.
Focusability: which elements participate in the Focus Zone?
Focusable elements are those that either are normally focusable via the Tab key OR have a valid tabindex
attribute (including "-1"
). The easiest way to ensure an element participates in the focus zone is by applying the attribute tabindex="-1"
.
Entering the focus zone
By default, when focus enters a focus zone, the element that receives focus will be the most recently-focused element within that focus zone. If no element had previously been focused, or if that previously-focused element was removed, focus will revert to the first focusable element within the focus zone, regardless of the direction of focus movement.
Using the focusInStrategy
option, you can change this behavior.
For more information on choosing the right focus in behavior, see 6.6 Keyboard Navigation Inside Components from the ARIA Authoring Practices document.
DOM focus vs. active descendant
The useFocusZone
hook supports two modes of operation: DOM Focus and Active descendant.
- DOM Focus is the default mode and by far the most commonly needed. When a key is used to move focus, we call
.focus()
directly on the element to receive focus. This results indocument.activeElement
getting set to this new element, and it will receive any necessary styles via:focus
and:focus-within
. - Active descendant mode does not move DOM focus. Instead, focus remains on the control element, and its
aria-activedescendant
attribute is set to the ID of the relevant element. Because there are no:focus
styles applied and nofocus
events fired, you can supply anonActivedescendantChanged
callback to handle any necessary styles or other logic as the active descendant changes. For more information on the Active descendant focus pattern, see 6.6.2 Managing Focus in Composites Usingaria-activedescendant
from the ARIA Authoring Practices document.
Examples
Default
Could not load examples for Primer component library: product
Custom key bindings
The bindKeys
option is used to set which of the following keys can be used to move focus.
This example shows how to use the bindKeys
option to allow focus to be moved using the J
and K
keys.
Could not load examples for Primer component library: product
Custom first focused element
Setting focusInStrategy
to "closest"
will cause either the first or last focusable element in the container to be focused depending on the direction of focus movement. For example, a shift+tab that brings focus to the container will cause the last focusable element to be focused, whereas a regular tab would cause the first focusable element to be focused.
Otherwise, you may provide a callback to choose a custom element to receive initial focus. One scenario where this would be useful is if you wanted to focus an item that is "selected" in a list.
Could not load examples for Primer component library: product
Custom focus out behavior
Focus cannot be moved beyond the last element of the container (other than using the Tab key). The focusOutBehavior
option can be used to allow focus to wrap around from last to first element (or vice-versa).
For a more customized focus movement behavior, the consumer has the ability to supply a custom callback that identifies the next element to focus.
Could not load examples for Primer component library: product
Custom focus movement
Could not load examples for Primer component library: product
Prevent element(s) from being focused
If you need to prevent a focusable element from participating in the focus zone, you can provide a focusableElementFilter
.
Could not load examples for Primer component library: product
Prevent container from scrolling while moving focus
Could not load examples for Primer component library: product
Disabling the focus zone
Could not load examples for Primer component library: product
Active descendant behavior
This is the kind of behavior that comboboxes use, where the focus is on the control element, but the active descendant is set to the ID of the relevant element.
Could not load examples for Primer component library: product
Nested focus zones
Could not load examples for Primer component library: product