Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(linter): add jsx-a11y/no-noninteractive-tabindex rule #8167

Merged
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions crates/oxc_linter/src/rules.rs
Original file line number Diff line number Diff line change
Expand Up @@ -412,6 +412,7 @@ mod jsx_a11y {
pub mod no_aria_hidden_on_focusable;
pub mod no_autofocus;
pub mod no_distracting_elements;
pub mod no_noninteractive_tabindex;
pub mod no_redundant_roles;
pub mod prefer_tag_over_role;
pub mod role_has_required_aria_props;
Expand Down Expand Up @@ -749,6 +750,7 @@ oxc_macros::declare_all_lint_rules! {
jsx_a11y::lang,
jsx_a11y::media_has_caption,
jsx_a11y::mouse_events_have_key_events,
jsx_a11y::no_noninteractive_tabindex,
jsx_a11y::no_access_key,
jsx_a11y::no_aria_hidden_on_focusable,
jsx_a11y::no_autofocus,
Expand Down
217 changes: 217 additions & 0 deletions crates/oxc_linter/src/rules/jsx_a11y/no_noninteractive_tabindex.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,217 @@
use oxc_ast::{
ast::{JSXAttributeItem, JSXAttributeValue},
AstKind,
};
use oxc_diagnostics::OxcDiagnostic;
use oxc_macros::declare_oxc_lint;
use oxc_span::{CompactStr, Span};
use phf::phf_set;

use crate::{
context::LintContext,
rule::Rule,
utils::{get_element_type, has_jsx_prop_ignore_case},
AstNode,
};

fn no_noninteractive_tabindex_diagnostic(span: Span) -> OxcDiagnostic {
OxcDiagnostic::warn("tabIndex should only be declared on interactive elements")
.with_help("tabIndex attribute should be removed")
.with_label(span)
}

#[derive(Debug, Clone)]
pub struct NoNoninteractiveTabindex(Box<NoNoninteractiveTabindexConfig>);

#[derive(Debug, Clone)]
struct NoNoninteractiveTabindexConfig {
tags: Vec<CompactStr>,
roles: Vec<CompactStr>,
allow_expression_values: bool,
}

impl Default for NoNoninteractiveTabindex {
fn default() -> Self {
Self(Box::new(NoNoninteractiveTabindexConfig {
roles: vec![CompactStr::new("tabpanel")],
allow_expression_values: true,
tags: vec![],
}))
}
}

declare_oxc_lint!(
/// ### What it does
/// This rule checks that non-interactive elements don't have a tabIndex which would make them interactive via keyboard navigation.
///
/// ### Why is this bad?
///
/// 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.
///
/// ### Examples
///
/// Examples of **incorrect** code for this rule:
/// ```jsx
/// <div tabIndex="0" />
/// <div role="article" tabIndex="0" />
/// <article tabIndex="0" />
/// <article tabIndex={0} />
/// ```
///
/// Examples of **correct** code for this rule:
/// ```jsx
/// <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" />
/// ```
NoNoninteractiveTabindex,
correctness,
);

// https://html.spec.whatwg.org/multipage/dom.html#interactive-content
const INTERACTIVE_HTML_ELEMENTS: phf::set::Set<&'static str> = phf_set! {
"a", "audio", "button", "details", "embed", "iframe", "img", "input", "label", "select", "textarea", "video"
};

// https://www.w3.org/TR/wai-aria/#widget_roles
// NOTE: "tabpanel" is not included here because it's technically a section role. It can optionally be considered interactive within the context of a tablist, because its visibility is dynamically controlled by an element with the "tab" aria role. It's included in the recommended jsx-a11y config for this reason.
const INTERACTIVE_HTML_ROLES: phf::set::Set<&'static str> = phf_set! {
"button", "checkbox", "gridcell", "link", "menuitem", "menuitemcheckbox", "menuitemradio", "option", "progressbar", "radio", "scrollbar", "searchbox", "separator", "slider", "spinbutton", "switch", "tab", "textbox", "treeitem"
};

impl Rule for NoNoninteractiveTabindex {
fn run<'a>(&self, node: &AstNode<'a>, ctx: &LintContext<'a>) {
let AstKind::JSXOpeningElement(jsx_el) = node.kind() else {
return;
};

let Some(JSXAttributeItem::Attribute(tabindex_attr)) =
has_jsx_prop_ignore_case(jsx_el, "tabIndex")
else {
return;
};

let Some(JSXAttributeValue::StringLiteral(tabindex)) = &tabindex_attr.value else {
return;
};

if tabindex.value == "-1" {
return;
}

let component = &get_element_type(ctx, jsx_el);

if INTERACTIVE_HTML_ELEMENTS.contains(component) {
return;
}

let Some(JSXAttributeItem::Attribute(role_attr)) = has_jsx_prop_ignore_case(jsx_el, "role")
else {
// if the component is not an interactive element and has no role, the tabindex is invalid.
ctx.diagnostic(no_noninteractive_tabindex_diagnostic(tabindex_attr.span));
return;
};

if self.0.allow_expression_values {
return;
}

let Some(JSXAttributeValue::StringLiteral(role)) = &role_attr.value else {
ctx.diagnostic(no_noninteractive_tabindex_diagnostic(tabindex_attr.span));
return;
};

if !INTERACTIVE_HTML_ROLES.contains(role.value.as_str())
&& !self.0.roles.contains(&CompactStr::new(role.value.as_str()))
{
ctx.diagnostic(no_noninteractive_tabindex_diagnostic(tabindex_attr.span));
}
}

fn from_configuration(value: serde_json::Value) -> Self {
let default = Self::default();

let Some(config) = value.get(0) else {
return default;
};

Self(Box::new(NoNoninteractiveTabindexConfig {
roles: config
.get("roles")
.and_then(serde_json::Value::as_array)
.map_or(default.0.roles, |v| {
v.iter().map(|v| CompactStr::new(v.as_str().unwrap())).collect()
}),
tags: config
.get("tags")
.and_then(serde_json::Value::as_array)
.map_or(default.0.tags, |v| {
v.iter().map(|v| CompactStr::new(v.as_str().unwrap())).collect()
}),
allow_expression_values: config
.get("allowExpressionValues")
.and_then(serde_json::Value::as_bool)
.unwrap_or(default.0.allow_expression_values),
}))
}
}

#[test]
fn test() {
use crate::tester::Tester;

let pass = vec![
(r#"<div role="tabpanel" tabIndex="0" />"#, None),
(r#"<div role={ROLE_BUTTON} onClick={() => {}} tabIndex="0" />;"#, None),
(
r#"<div role={BUTTON} onClick={() => {}} tabIndex="0" />;"#,
Some(serde_json::json!([{ "allowExpressionValues": true }])),
),
(
r#"<div role={isButton ? "button" : "link"} onClick={() => {}} tabIndex="0" />;"#,
Some(serde_json::json!([{ "allowExpressionValues": true }])),
),
(
r#"<div role={isButton ? "button" : LINK} onClick={() => {}} tabIndex="0" />;"#,
Some(serde_json::json!([{ "allowExpressionValues": true }])),
),
(
r#"<div role={isButton ? BUTTON : LINK} onClick={() => {}} tabIndex="0"/>;"#,
Some(serde_json::json!([{ "allowExpressionValues": true }])),
),
];

let fail = vec![
(
r#"<div role="tabpanel" tabIndex="0" />"#,
Some(serde_json::json!([{ "roles": [], "allowExpressionValues": false }])),
),
(
r#"<div role={ROLE_BUTTON} onClick={() => {}} tabIndex="0" />;"#,
Some(serde_json::json!([{ "roles": [], "allowExpressionValues": false }])),
),
(
r#"<div role={BUTTON} onClick={() => {}} tabIndex="0" />;"#,
Some(serde_json::json!([{ "allowExpressionValues": false }])),
),
(
r#"<div role={isButton ? "button" : "link"} onClick={() => {}} tabIndex="0" />;"#,
Some(serde_json::json!([{ "allowExpressionValues": false }])),
),
];

Tester::new(NoNoninteractiveTabindex::NAME, NoNoninteractiveTabindex::CATEGORY, pass, fail)
.test_and_snapshot();
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
---
source: crates/oxc_linter/src/tester.rs
snapshot_kind: text
---
⚠ eslint-plugin-jsx-a11y(no-noninteractive-tabindex): tabIndex should only be declared on interactive elements
╭─[no_noninteractive_tabindex.tsx:1:22]
1 │ <div role="tabpanel" tabIndex="0" />
· ────────────
╰────
help: tabIndex attribute should be removed

⚠ eslint-plugin-jsx-a11y(no-noninteractive-tabindex): tabIndex should only be declared on interactive elements
╭─[no_noninteractive_tabindex.tsx:1:44]
1 │ <div role={ROLE_BUTTON} onClick={() => {}} tabIndex="0" />;
· ────────────
╰────
help: tabIndex attribute should be removed

⚠ eslint-plugin-jsx-a11y(no-noninteractive-tabindex): tabIndex should only be declared on interactive elements
╭─[no_noninteractive_tabindex.tsx:1:39]
1 │ <div role={BUTTON} onClick={() => {}} tabIndex="0" />;
· ────────────
╰────
help: tabIndex attribute should be removed

⚠ eslint-plugin-jsx-a11y(no-noninteractive-tabindex): tabIndex should only be declared on interactive elements
╭─[no_noninteractive_tabindex.tsx:1:61]
1 │ <div role={isButton ? "button" : "link"} onClick={() => {}} tabIndex="0" />;
· ────────────
╰────
help: tabIndex attribute should be removed
Loading