Interactive elements, including form controls, links, and buttons, are by default focusable and tabbable. Tabbable elements are part of the document's sequential focus navigation order. Other elements are inert, meaning they are not interactive. With HTML attributes, it's possible to make interactive elements inert and to make inert elements interactive.
By default, the navigation focus order is the same as the visual order, which is the source code order. There are HTML attributes that can alter this order and CSS properties that can alter the visual order of content. Changing the tabbing order with HTML or visual rendering order with CSS can harm user experience.
Don't alter the perceived and actual tabbing order with CSS and HTML. As the following two examples demonstrate, tab orders that differ from the visually expected order are confusing to users and bad for user experience.
In this example, the value of the tabindex attribute
has made the tab order chaotic:
In this example, CSS has created a divergence between the tabbing order and the visual order of the content:
The flex-flow: row-reverse; declaration has reversed the visual order.
In addition, the CSS order property was applied to the sixth word, "This", which visually moved that
one word. The tabbing sequence is the order of the code, which no longer matches the visual order, creating a disconnect
for keyboard users.
Make inert elements interactive
The contenteditable and tabindex attributes, being global attributes, can be added to any element, making them focusable
in the process. Focusable elements can also be focused with a mouse or pointer, by having the autofocus
attribute set, or by script, such as with element.focus().
The tabindex attribute
The global tabindex attribute, introduced in attributes, enables
elements that otherwise wouldn't be able to receive focus to get
focus, usually with Tab, hence the name.
The tabindex attribute takes as its value an integer. A negative value makes
an element focusable but not tabbable. A tabindex value of 0 makes the
element focusable and tabbable, adding the element on which it's applied to
the sequential focus navigation order in source code order. A value of 1 or
greater makes the element focusable and tabbable, but adds it to a prioritized
tabbing sequence and should be avoided.
On this page, the share button, <share-action>, is a
custom element. The tabindex="0" adds this not-normally focusable
element into the keyboard default tabbing order:
<share-action authors="@front-end.social/@estellevw" data-action="click" data-category="web.dev" data-icon="share" data-label="share, mastodon" role="button" tabindex="0">
<svg aria-label="share" role="img" xmlns="http://www.w3.org/2000/svg">
<use href="#shareIcon" />
</svg>
<span>Share</span>
</share-action>
There's another custom element on this page: the local navigation has a custom
element with a negative tabindex value:
<web-navigation-drawer type="standard" tabindex="-1">
A tabindex attribute with a negative value makes the element focusable but not tabbable. The element is capable of receiving
focus, such as using HTMLElement.focus(), but it's not part
of the sequential focus navigation order. The convention for non-tabbable, focusable elements is to use tabindex="-1". If you add tabindex="-1" to an interactive element, it will no longer be tabbable.
The element.focus()
method can be used to set focus to focusable elements. Browsers scroll focused
elements into view. For this reason, avoid the use of
element.focus({preventScroll:true}), as focusing on a non-visible element is a
bad user experience.
If you want to query the document to find out which element has focus, use the
read-only Document.activeElement property.
Elements with a tabindex of 1 or greater are included in a separate tab sequence. As you'll notice in the Codepen,
tabbing begins in a separate sequence, in order of lowest value to highest value, before going through those in the regular sequence
(no tabindex set, or tabindex="0") in source order:
tabindex with a positive value puts the element into a prioritized focus sequence, which can lead to focus order chaos.
Avoid modifying the DOM order with tabindex. Not only can altered tabbing orders create bad user
experiences, they are difficult for developers to manage and maintain.
The contenteditable attribute
The contenteditable attribute was discussed earlier. Setting contenteditable="true" on any element makes it editable,
focusable, and part of the tab order. The focus behavior is similar to setting tabindex="0", but not the same. Nested
contenteditable elements are focusable but not tabbable. To make a nested contenteditable element tabbable, add tabindex="0",
which add it to the sequential focus navigation order.
Give autofocus to interactive elements
While the boolean autofocus is a global attribute
that can be set on any element, it doesn't make an inert element interactive. When the page loads, the first focusable element
with the autofocus attributes receives focus, as long as that element is displayed and not nested in a <dialog>.
Automatically setting focus on content can be confusing. Setting autofocus on a form control means that the form control
scrolls into view on page load. All your users, including screen reader users and users with small viewports, may not
"see" the instructions for the form, possibly even scrolling past the form control's normally visible label. The autofocus
attribute doesn't alter the document's sequential focus navigation order. The elements in the sequence coming before the
auto-focused element are skipped. For these reasons, it's not advised to include the autofocus attribute.
The exception to the "don't use autofocus" recommendation is including the
autofocus attribute within <dialog> elements. When a dialog is opened,
the browser automatically focuses on the first focusable interactive element within the <dialog>, which means it's unnecessary to add autofocus to an element. If
you want to be sure a specific interactive element within the dialog receives
focus when the dialog opens, add the autofocus attribute to that element.
<dialog open>
<form method="dialog">
<button type="submit" autofocus>close</button>
</form>
</dialog>
The autofocus attribute set on the close <button> allow it to receive focus
when the dialog is opened. As the first element in the dialog, it would have
received focus in any case. By default, when a dialog is opened, the first
focusable element within the dialog receives focus unless a different element
within the dialog has the autofocus attribute set.
Make interactive elements inert
There are also HTML attributes that can remove interactive elements from the tabbing sequence. Including a negative tabindex
to focusable elements, adding the disabled attribute to supporting form controls, and adding the global inert attribute
to a container all make elements un-tabbable. These three attributes are NOT interchangeable.
Negative tabindex value
A tabindex attribute with a negative value makes an element focusable, but not
tabbable. While adding tabindex="0" to a focusable-by-default element,
including links, buttons, form controls, and elements that are contenteditable
is not necessary; including a tabindex with a negative value removes normally
tabbable elements from the sequential focus navigation order.
A negative tabindex value prevents keyboard users from focusing on interactive
elements, but doesn't disable the element. Pointer users can still focus on the
element. To disable an element, use the disabled attribute.
Disabled
The boolean disabled attribute makes the form controls on which it's applied and their descendants, if any, unfocusable. Disabled form controls can't be focused, don't get click events, and are not submitted upon form submission.
disabled is not a global attribute. It applies to <button>, <input>,
<optgroup>, <option>, <select>, <textarea>, form-associated custom
elements, and <fieldset>.
When set on <optgroup> or <fieldset>, all the child form controls are
disabled, except for the contents of the <fieldset>'s first
<legend>.
The same elements that support disabled are also targetable with the
:disabled
and :enabled
pseudo-classes. Elements that are disabled with the disabled attribute are
generally styled as light gray with the user-agent stylesheet, even if an
accent-color
is set.
Being a boolean attribute, the presence of the attribute disables the otherwise
enabled element; you can't set it to false. To re-enable a disabled element,
the attribute has to be removed, generally with
Element.removeAttribute('disabled').
The HTMLInputElement.disabled
property lets you check if an input is disabled. As disabled is not a global
attribute, it's not inherited from the HTMLElement, but every supporting
element interface, like HTMLSelectElement,
HTMLTextareaElement,
has the same read-only property.
The disabled attribute doesn't apply to normally inert elements that are
made focusable with tabindex or contenteditable, nor does it apply to the
<form> element. To disable these elements, the global inert attribute can
be used.
The inert attribute
When the inert global boolean attribute is added to an element, that element
and all nested content become disabled, which means they cannot be clicked or
tabbed to. They are also removed from the accessibility tree. While inert
can be applied to any element, it's generally used for sections of content,
such as offscreen or otherwise hidden content.
When applying disabled to form controls, the browser provides default styling
and can be styled using the :disabled
pseudo class. The inert attribute provides no visual indicators and has no
matching pseudoclass (though the [inert]
attribute selector matches).
Using inert on visible content without styles indicating the inertness can
lead to poor user experience. As inert content is not available to screen reader
users, it can lead to confusion when sighted screen reader users see content on
screen that is not available to the accessibility tools. Make inertness very
apparent with CSS.
Make sure that the focus never moves to non-visible content. Anything rendered off-screen that does not automatically come into view when focused should be made inert. If content is hidden, but comes into view when focused, like a skip to content link, it does not need to be made inert.
Check your understanding
Test your knowledge of focus.
If an element cannot be focused it is described as what?
What will be true if the element has a disabled attribute?