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 noNoninteractiveElementInteractions rule #4358

Merged
Merged
Show file tree
Hide file tree
Changes from 13 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
10 changes: 10 additions & 0 deletions crates/biome_cli/src/execute/migrate/eslint_any_rule_to_biome.rs

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

152 changes: 86 additions & 66 deletions crates/biome_configuration/src/analyzer/linter/rules.rs

Large diffs are not rendered by default.

1 change: 1 addition & 0 deletions crates/biome_diagnostics_categories/src/categories.rs
Original file line number Diff line number Diff line change
Expand Up @@ -159,6 +159,7 @@ define_categories! {
"lint/nursery/noMissingGenericFamilyKeyword": "https://biomejs.dev/linter/rules/no-missing-generic-family-keyword",
"lint/nursery/noMissingVarFunction": "https://biomejs.dev/linter/rules/no-missing-var-function",
"lint/nursery/noNestedTernary": "https://biomejs.dev/linter/rules/no-nested-ternary",
"lint/nursery/noNoninteractiveElementInteractions": "https://biomejs.dev/linter/rules/no-noninteractive-element-interactions",
"lint/nursery/noOctalEscape": "https://biomejs.dev/linter/rules/no-octal-escape",
"lint/nursery/noProcessEnv": "https://biomejs.dev/linter/rules/no-process-env",
"lint/nursery/noReactSpecificProps": "https://biomejs.dev/linter/rules/no-react-specific-props",
Expand Down
2 changes: 2 additions & 0 deletions crates/biome_js_analyze/src/lint/nursery.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,7 @@ pub mod no_head_import_in_document;
pub mod no_img_element;
pub mod no_irregular_whitespace;
pub mod no_nested_ternary;
pub mod no_noninteractive_element_interactions;
pub mod no_octal_escape;
pub mod no_process_env;
pub mod no_restricted_imports;
Expand Down Expand Up @@ -57,6 +58,7 @@ declare_lint_group! {
self :: no_img_element :: NoImgElement ,
self :: no_irregular_whitespace :: NoIrregularWhitespace ,
self :: no_nested_ternary :: NoNestedTernary ,
self :: no_noninteractive_element_interactions :: NoNoninteractiveElementInteractions ,
self :: no_octal_escape :: NoOctalEscape ,
self :: no_process_env :: NoProcessEnv ,
self :: no_restricted_imports :: NoRestrictedImports ,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,302 @@
use crate::services::aria::Aria;
use biome_analyze::{context::RuleContext, declare_lint_rule, Rule, RuleDiagnostic, RuleSource};
use biome_console::markup;
use biome_js_syntax::jsx_ext::AnyJsxElement;
use biome_rowan::AstNode;
use rustc_hash::FxHashMap;

declare_lint_rule! {
/// Disallow use event handlers on non-interactive elements.
///
/// Non-interactive HTML elements indicate _content_ and _containers_ in the user interface.
/// Non-interactive elements include `<main>`, `<area>`, `<h1>` (,`<h2>`, etc), `<img>`, `<li>`, `<ul>` and `<ol>`.
///
/// A Non-interactive element does not support event handlers(mouse and key handlers).
///
///
/// ## Examples
///
/// ### Invalid
///
/// ```jsx,expect_diagnostic
/// <div onClick={() => {}}>button</div>
/// ```
///
/// ### Valid
///
/// ```jsx
/// <button onClick={() => { }}>button</button>
/// ```
///
/// ```jsx
/// // Adding a role to element does not add behavior.
/// // If not used semantic HTML elements like `button`, developers need to implement the expected behavior for role(like focusability and key press support)
/// // See https://www.w3.org/WAI/ARIA/apg/
/// <div role="button" onClick={() => { }}>button</div>
/// ```
///
/// ```jsx
/// // The role="presentation" attribute removes the semantic meaning of an element, indicating that it should be ignored by assistive technologies.
/// // Therefore, it's acceptable to add event handlers to elements with role="presentation" for visual effects or other purposes,
/// // but users relying on assistive technologies may not be able to interact with these elements.
/// <div role="presentation" onClick={() => { }}>button</div>
/// ```
///
/// ```jsx
/// // Hidden from screen reader.
/// <div onClick={() => {}} aria-hidden />
/// ```
///
/// ```jsx
/// // Custom component is not checked.
/// <SomeComponent onClick={() => {}}>button</SomeComponent>
/// ```
///
/// ```jsx
/// // Spread attributes is not supported.
/// <div {...{"onClick":() => {}}}>button</div>
/// ```
///
/// ## Accessibility guidelines
///
/// - [WCAG 4.1.2](https://www.w3.org/WAI/WCAG21/Understanding/name-role-value)
///
/// ### Resources
///
/// - [WAI-ARIA roles](https://www.w3.org/TR/wai-aria-1.1/#usage_intro)
/// - [WAI-ARIA Authoring Practices Guide - Design Patterns and Widgets](https://www.w3.org/TR/wai-aria-practices-1.1/#aria_ex)
/// - [Fundamental Keyboard Navigation Conventions](https://www.w3.org/TR/wai-aria-practices-1.1/#kbd_generalnav)
/// - [Mozilla Developer Network - ARIA Techniques](https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/ARIA_Techniques/Using_the_button_role#Keyboard_and_focus)
///
pub NoNoninteractiveElementInteractions {
version: "next",
name: "noNoninteractiveElementInteractions",
language: "jsx",
sources: &[RuleSource::EslintJsxA11y("no-noninteractive-element-interactions")],
recommended: false,
}
}

/// Ref: https://github.com/jsx-eslint/jsx-ast-utils/blob/v3.3.5/src/eventHandlers.js
const INTERACTIVE_HANDLERS: &[&str] = &[
"onClick",
"onContextMenu",
"onDblClick",
"onDoubleClick",
"onDrag",
"onDragEnd",
"onDragEnter",
"onDragExit",
"onDragLeave",
"onDragOver",
"onDragStart",
"onDrop",
"onMouseDown",
"onMouseEnter",
"onMouseLeave",
"onMouseMove",
"onMouseOut",
"onMouseOver",
"onKeyDown",
"onKeyPress",
"onKeyUp",
"onFocus",
"onBlur",
"onLoad",
"onError",
];

type AttributesRef<'a> = Option<&'a FxHashMap<String, Vec<String>>>;

/// Check if the element contains event handler
fn has_handler_props(attributes: AttributesRef) -> bool {
attributes.is_some_and(|m| {
INTERACTIVE_HANDLERS
.iter()
.any(|handler| m.contains_key(*handler))
})
}

/// Check if the element's implicit ARIA semantics have been removed.
/// See https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Roles/presentation_role
///
/// Ref: https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/v6.10.0/src/util/isPresentationRole.js
fn has_presentation_role(attributes: AttributesRef) -> bool {
const PRESENTATION_ROLE: &[&str] = &["presentation", "none"];

if let Some(attributes) = attributes {
if let Some(values) = attributes.get("role") {
let is_presentation = values
.iter()
.any(|v| PRESENTATION_ROLE.contains(&v.as_str()));
return is_presentation;
}
}
false
}

/// Check the element is hidden from screen reader.
/// See
/// - https://developer.mozilla.org/en-US/docs/Web/Accessibility/ARIA/Attributes/aria-hidden
/// - https://developer.mozilla.org/en-US/docs/Web/HTML/Element/input/hidden
///
/// Ref: https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/v6.10.0/src/util/isHiddenFromScreenReader.js
fn is_hidden_from_screen_reader(element_name: &str, attributes: AttributesRef) -> bool {
if let Some(attributes) = attributes {
let aria_hidden = attributes
.get("aria-hidden")
.map_or(false, |attr| attr.contains(&"true".to_string()));

let input_hidden = element_name == "input"
&& attributes
.get("type")
.map_or(false, |attr| attr.contains(&"hidden".to_string()));

return aria_hidden || input_hidden;
}

false
}

/// Check if the element is `contentEditable`
/// See https://developer.mozilla.org/en-US/docs/Web/HTML/Global_attributes/contenteditable
///
/// Ref: https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/v6.10.0/src/util/isContentEditable.js
fn is_content_editable(attributes: AttributesRef) -> bool {
if let Some(attributes) = attributes {
if let Some(values) = attributes.get("contentEditable") {
return values.contains(&"true".to_string());
}
}
false
}

/// List of `widget` role
/// https://www.w3.org/TR/wai-aria-1.2/#widget
const INTERACTIVE_ROLES: &[&str] = &[
ematipico marked this conversation as resolved.
Show resolved Hide resolved
// https://www.w3.org/TR/wai-aria-1.2/#command
"button",
"link",
// https://www.w3.org/TR/wai-aria-1.2/#menuitem
"menuitem",
"menuitemcheckbox",
"menuitemradio",
// https://www.w3.org/TR/wai-aria-1.2/#composite
"spinbutton",
"tablist",
// https://www.w3.org/TR/wai-aria-1.2/#grid
"grid",
"treegrid",
// https://www.w3.org/TR/wai-aria-1.2/#select
"listbox",
"radiogroup",
// https://www.w3.org/TR/wai-aria-1.2/#menu
"menu",
"menubar",
// https://www.w3.org/TR/wai-aria-1.2/#tree
"tree",
"treegrid",
// https://www.w3.org/TR/wai-aria-1.2/#gridcell
"gridcell",
"columnheader",
"rowheader",
// https://www.w3.org/TR/wai-aria-1.2/#input
"combobox",
"radio",
"slider",
// https://www.w3.org/TR/wai-aria-1.2/#option
"option",
"treeitem",
// https://www.w3.org/TR/wai-aria-1.2/#checkbox
"checkbox",
"switch",
// https://www.w3.org/TR/wai-aria-1.2/#textbox
"textbox",
"searchbox",
// https://www.w3.org/TR/wai-aria-1.2/#progressbar
"progressbar",
// https://www.w3.org/TR/wai-aria-1.2/#row
"row",
// https://www.w3.org/TR/wai-aria-1.2/#scrollbar
"scrollbar",
// https://www.w3.org/TR/wai-aria-1.2/#separator
// `separator` is not included because it is a `widget` only when it is focusable
// "separator",
// https://www.w3.org/TR/wai-aria-1.2/#tab
"tab",
// This does not descend from widget, but support `aria-activedescendant`.
// So we treat them as widget.
// See https://www.w3.org/TR/wai-aria-1.2/#aria-activedescendant
// Ref: https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/v6.10.0/src/util/isInteractiveRole.js
"toolbar",
];

/// Check if the element has interactive role
///
/// Ref: https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/v6.10.0/src/util/isInteractiveRole.js
fn is_interactive_role(attributes: AttributesRef) -> bool {
if let Some(attributes) = attributes {
if let Some(roles) = attributes.get("role") {
return roles
.iter()
.any(|role| INTERACTIVE_ROLES.contains(&role.as_str()));
}
}
false
}

impl Rule for NoNoninteractiveElementInteractions {
type Query = Aria<AnyJsxElement>;
type State = ();
type Signals = Option<Self::State>;
type Options = ();

fn run(ctx: &RuleContext<Self>) -> Self::Signals {
let element = ctx.query();

// Custom components are not checked because we do not know what DOM will be used.
if element.is_custom_component() {
return None;
}

let attributes = ctx.extract_attributes(&element.attributes());
let attributes = ctx.convert_all_attribute_values(attributes);
let attributes_ref = attributes.as_ref();
let element_name = element.name().ok()?.as_jsx_name()?.value_token().ok()?;
let element_name = element_name.text_trimmed();

if !has_handler_props(attributes_ref)
|| is_content_editable(attributes_ref)
|| has_presentation_role(attributes_ref)
|| is_hidden_from_screen_reader(element_name, attributes_ref)
|| is_interactive_role(attributes_ref)
{
return None;
}

let aria_roles = ctx.aria_roles();

// Non-interactive elements what contains event handler should be reported.
if aria_roles.is_not_interactive_element(element_name, attributes) {
return Some(());
};

None
}

fn diagnostic(ctx: &RuleContext<Self>, _state: &Self::State) -> Option<RuleDiagnostic> {
let node = ctx.query();
Some(
RuleDiagnostic::new(
rule_category!(),
node.range(),
markup! {
"Non-interactive element should not have event handler."
},
)
.note(markup! {
"Consider replace semantically interactive element like "<Emphasis>"<button/>"</Emphasis>" or "<Emphasis>"<a href/>"</Emphasis>"."
})
)
}
}
1 change: 1 addition & 0 deletions crates/biome_js_analyze/src/options.rs

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading