no-noninteractive-tabindex

Tab key navigation should be limited to elements on the page that can be interacted with. Thus it is not necessary to add a tabindex to items in an unordered list, for example, to make them navigable through assistive technology. These applications already afford page traversal mechanisms based on the HTML of the page. Generally, we should try to reduce the size of the page's tab ring rather than increasing it.

How do I resolve this error?

Case: I am using an <a> tag. Isn't that interactive?

The <a> tag is tricky. Consider the following:

<a>Edit</a>
<a href="#">Edit</a>
<a role="button">Edit</a>

The bare <a> tag is an anchor. It has no semantic AX API mapping in either ARIA or the AXObject model. It's as meaningful as <div>, which is to say it has no meaning. An <a> tag with an href attribute has an inherent role of link. An <a> tag with an explicit role obtains the designated role. In the example above, this role is button.

Case: I am using “semantic” HTML. Isn't that interactive?

If we take a step back into the field of linguistics for a moment, let's consider what it means for something to be “semantic”. Nothing, in and of itself, has meaning. Meaning is constructed through dialogue. A speaker intends a meaning and a listener/observer interprets a meaning. Each participant constructs their own meaning through dialogue. There is no intrinsic or isolated meaning outside of interaction. Thus, we must ask, given that we have a “speaker” who communicates via “semantic” HTML, who is listening/observing?

In our case, the observer is the Accessibility (AX) API. Browsers interpret HTML (inflected at times by ARIA) to construct a meaning (AX Tree) of the page. Whatever the semantic HTML intends has only the force of suggestion to the AX API. Therefore, we have inconsistencies. For example, there is not yet an ARIA role for text or label and thus no way to change a <label> into plain text or a <span> into a label via ARIA. <div> has an AXObject correpondant DivRole, but no such object maps to <span>.

What this lint rule endeavors to do is apply the AX API understanding of the semantics of an HTML document back onto your code. The concept of interactivity boils down to whether a user can do something with the indicated or focused component.

Common interactive roles include:

  1. button
  2. link
  3. checkbox
  4. menuitem
  5. menuitemcheckbox
  6. menuitemradio
  7. option
  8. radio
  9. searchbox
  10. switch
  11. textbox

Endeavor to limit tabbable elements to those that a user can act upon.

Case: Shouldn't I add a tabindex so that users can navigate to this item?

It is not necessary to put a tabindex on an <article>, for instance or on <li> items; assistive technologies provide affordances to users to find and traverse these containers. Most elements that require a tabindex -- <a href>, <button>, <input>, <textarea> -- have it already.

Your application might require an exception to this rule in the case of an element that captures incoming tab traversal for a composite widget. In that case, turn off this rule on a per instance basis. This is an uncommon case.

References

  1. Fundamental Keyboard Navigation Conventions

Rule details

The recommended options for this rule allow tabIndex on elements with the noninteractive tabpanel role. Adding tabIndex to a tabpanel is a recommended practice in some instances.

'jsx-a11y/no-noninteractive-tabindex': [
  'error',
  {
    tags: [],
    roles: ['tabpanel'],
  },
]

Succeed

<div />
<MyButton tabIndex={0} />
<button />
<button tabIndex="0" />
<button tabIndex={0} />
<div />
<div tabIndex="-1" />
<div role="button" tabIndex="0" />
<div role="article" tabIndex="-1" />
<article tabIndex="-1" />

Fail

<div tabIndex="0" />
<div role="article" tabIndex="0" />
<article tabIndex="0" />
<article tabIndex={0} />