UNPKG

18.1 kBMarkdownView Raw
1<p align="center">
2 <a href="https://travis-ci.com/github/jsx-eslint/eslint-plugin-jsx-a11y">
3 <img src="https://travis-ci.com/jsx-eslint/eslint-plugin-jsx-a11y.svg?branch=master"
4 alt="build status">
5 </a>
6 <a href="https://npmjs.org/package/eslint-plugin-jsx-a11y">
7 <img src="https://img.shields.io/npm/v/eslint-plugin-jsx-a11y.svg"
8 alt="npm version">
9 </a>
10 <a href="https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/LICENSE.md">
11 <img src="https://img.shields.io/npm/l/eslint-plugin-jsx-a11y.svg"
12 alt="license">
13 </a>
14 <a href='https://coveralls.io/github/jsx-eslint/eslint-plugin-jsx-a11y?branch=master'>
15 <img src='https://coveralls.io/repos/github/jsx-eslint/eslint-plugin-jsx-a11y/badge.svg?branch=master' alt='Coverage Status' />
16 </a>
17 <a href='https://npmjs.org/package/eslint-plugin-jsx-a11y'>
18 <img src='https://img.shields.io/npm/dt/eslint-plugin-jsx-a11y.svg'
19 alt='Total npm downloads' />
20 </a>
21</p>
22
23<a href='https://tidelift.com/subscription/pkg/npm-eslint-plugin-jsx-a11y?utm_source=npm-eslint-plugin-jsx-a11y&utm_medium=referral&utm_campaign=readme'>Get professional support for eslint-plugin-jsx-a11y on Tidelift</a>
24
25# eslint-plugin-jsx-a11y
26
27Static AST checker for accessibility rules on JSX elements.
28
29
30
31#### *Read this in [other languages](https://github.com/ari-os310/eslint-plugin-jsx-a11y/blob/HEAD/translations/Translations.md).*
32
33[Mexican Spanish🇲🇽](https://github.com/ari-os310/eslint-plugin-jsx-a11y/blob/HEAD/translations/README.mx.md)
34
35## Why?
36Ryan Florence built out this awesome runtime-analysis tool called [react-a11y](https://github.com/reactjs/react-a11y). It is super useful. However, since you're probably already using linting in your project, this plugin comes for free and closer to the actual development process. Pairing this plugin with an editor lint plugin, you can bake accessibility standards into your application in real-time.
37
38**Note**: This project does not *replace* react-a11y, but can and should be used in conjunction with it. Static analysis tools cannot determine values of variables that are being placed in props before runtime, so linting will not fail if that value is undefined and/or does not pass the lint rule.
39
40## Installation
41
42**If you are installing this plugin via `eslint-config-airbnb`, please follow [these instructions](https://github.com/airbnb/javascript/tree/HEAD/packages/eslint-config-airbnb#eslint-config-airbnb-1).**
43
44You'll first need to install [ESLint](https://eslint.org):
45
46```sh
47# npm
48npm install eslint --save-dev
49
50# yarn
51yarn add eslint --dev
52```
53
54Next, install `eslint-plugin-jsx-a11y`:
55
56```sh
57# npm
58npm install eslint-plugin-jsx-a11y --save-dev
59
60# yarn
61yarn add eslint-plugin-jsx-a11y --dev
62```
63
64**Note:** If you installed ESLint globally (using the `-g` flag in npm, or the `global` prefix in yarn) then you must also install `eslint-plugin-jsx-a11y` globally.
65
66## Usage
67
68Add `jsx-a11y` to the plugins section of your `.eslintrc` configuration file. You can omit the `eslint-plugin-` prefix:
69
70```json
71{
72 "plugins": [
73 "jsx-a11y"
74 ]
75}
76```
77
78
79Then configure the rules you want to use under the rules section.
80
81```json
82{
83 "rules": {
84 "jsx-a11y/rule-name": 2
85 }
86}
87```
88
89You can also enable all the recommended or strict rules at once.
90Add `plugin:jsx-a11y/recommended` or `plugin:jsx-a11y/strict` in `extends`:
91
92```json
93{
94 "extends": [
95 "plugin:jsx-a11y/recommended"
96 ]
97}
98```
99
100## Supported Rules
101
102- [alt-text](docs/rules/alt-text.md): Enforce all elements that require alternative text have meaningful information to relay back to end user.
103- [anchor-has-content](docs/rules/anchor-has-content.md): Enforce all anchors to contain accessible content.
104- [anchor-is-valid](docs/rules/anchor-is-valid.md): Enforce all anchors are valid, navigable elements.
105- [aria-activedescendant-has-tabindex](docs/rules/aria-activedescendant-has-tabindex.md): Enforce elements with aria-activedescendant are tabbable.
106- [aria-props](docs/rules/aria-props.md): Enforce all `aria-*` props are valid.
107- [aria-proptypes](docs/rules/aria-proptypes.md): Enforce ARIA state and property values are valid.
108- [aria-role](docs/rules/aria-role.md): Enforce that elements with ARIA roles must use a valid, non-abstract ARIA role.
109- [aria-unsupported-elements](docs/rules/aria-unsupported-elements.md): Enforce that elements that do not support ARIA roles, states, and properties do not have those attributes.
110- [autocomplete-valid](docs/rules/autocomplete-valid.md): Enforce that autocomplete attributes are used correctly.
111- [click-events-have-key-events](docs/rules/click-events-have-key-events.md): Enforce a clickable non-interactive element has at least one keyboard event listener.
112- [heading-has-content](docs/rules/heading-has-content.md): Enforce heading (`h1`, `h2`, etc) elements contain accessible content.
113- [html-has-lang](docs/rules/html-has-lang.md): Enforce `<html>` element has `lang` prop.
114- [iframe-has-title](docs/rules/iframe-has-title.md): Enforce iframe elements have a title attribute.
115- [img-redundant-alt](docs/rules/img-redundant-alt.md): Enforce `<img>` alt prop does not contain the word "image", "picture", or "photo".
116- [interactive-supports-focus](docs/rules/interactive-supports-focus.md): Enforce that elements with interactive handlers like `onClick` must be focusable.
117- [label-has-associated-control](docs/rules/label-has-associated-control.md): Enforce that a `label` tag has a text label and an associated control.
118- [lang](docs/rules/lang.md): Enforce lang attribute has a valid value.
119- [media-has-caption](docs/rules/media-has-caption.md): Enforces that `<audio>` and `<video>` elements must have a `<track>` for captions.
120- [mouse-events-have-key-events](docs/rules/mouse-events-have-key-events.md): Enforce that `onMouseOver`/`onMouseOut` are accompanied by `onFocus`/`onBlur` for keyboard-only users.
121- [no-access-key](docs/rules/no-access-key.md): Enforce that the `accessKey` prop is not used on any element to avoid complications with keyboard commands used by a screenreader.
122- [no-autofocus](docs/rules/no-autofocus.md): Enforce autoFocus prop is not used.
123- [no-distracting-elements](docs/rules/no-distracting-elements.md): Enforce distracting elements are not used.
124- [no-interactive-element-to-noninteractive-role](docs/rules/no-interactive-element-to-noninteractive-role.md): Interactive elements should not be assigned non-interactive roles.
125- [no-noninteractive-element-interactions](docs/rules/no-noninteractive-element-interactions.md): Non-interactive elements should not be assigned mouse or keyboard event listeners.
126- [no-noninteractive-element-to-interactive-role](docs/rules/no-noninteractive-element-to-interactive-role.md): Non-interactive elements should not be assigned interactive roles.
127- [no-noninteractive-tabindex](docs/rules/no-noninteractive-tabindex.md): `tabIndex` should only be declared on interactive elements.
128- [no-onchange](docs/rules/no-onchange.md): Enforce usage of `onBlur` over `onChange` on select menus for accessibility.
129- [no-redundant-roles](docs/rules/no-redundant-roles.md): Enforce explicit role property is not the same as implicit/default role property on element.
130- [no-static-element-interactions](docs/rules/no-static-element-interactions.md): Enforce that non-interactive, visible elements (such as `<div>`) that have click handlers use the role attribute.
131- [role-has-required-aria-props](docs/rules/role-has-required-aria-props.md): Enforce that elements with ARIA roles must have all required attributes for that role.
132- [role-supports-aria-props](docs/rules/role-supports-aria-props.md): Enforce that elements with explicit or implicit roles defined contain only `aria-*` properties supported by that `role`.
133- [scope](docs/rules/scope.md): Enforce `scope` prop is only used on `<th>` elements.
134- [tabindex-no-positive](docs/rules/tabindex-no-positive.md): Enforce `tabIndex` value is not greater than zero.
135
136### Rule strictness in different modes
137
138Rule | Recommended | Strict
139------------ | ------------- | -------------
140[alt-text](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/alt-text.md) | error | error
141[anchor-has-content](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/anchor-has-content.md) | error | error
142[anchor-is-valid](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/anchor-is-valid.md) | error | error
143[aria-activedescendant-has-tabindex](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/aria-activedescendant-has-tabindex.md) | error | error
144[aria-props](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/aria-props.md) | error | error
145[aria-proptypes](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/aria-proptypes.md) | error | error
146[aria-role](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/aria-role.md) | error | error
147[aria-unsupported-elements](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/aria-unsupported-elements.md) | error | error
148[autocomplete-valid](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/autocomplete-valid.md) | error | error
149[click-events-have-key-events](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/click-events-have-key-events.md) | error | error
150[heading-has-content](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/heading-has-content.md) | error | error
151[html-has-lang](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/html-has-lang.md) | error | error
152[iframe-has-title](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/iframe-has-title.md) | error | error
153[img-redundant-alt](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/img-redundant-alt.md) | error | error
154[interactive-supports-focus](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/interactive-supports-focus.md) | error | error
155[label-has-associated-control](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/label-has-associated-control.md) | error | error
156[media-has-caption](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/media-has-caption.md) | error | error
157[mouse-events-have-key-events](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/mouse-events-have-key-events.md) | error | error
158[no-access-key](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/no-access-key.md) | error | error
159[no-autofocus](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/no-autofocus.md) | error | error
160[no-distracting-elements](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/no-distracting-elements.md) | error | error
161[no-interactive-element-to-noninteractive-role](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/no-interactive-element-to-noninteractive-role.md) | error, with options | error
162[no-noninteractive-element-interactions](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/no-noninteractive-element-interactions.md) | error, with options | error
163[no-noninteractive-element-to-interactive-role](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/no-noninteractive-element-to-interactive-role.md) | error, with options | error
164[no-noninteractive-tabindex](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/no-noninteractive-tabindex.md) | error, with options | error
165[no-onchange](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/no-onchange.md) | error | error
166[no-redundant-roles](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/no-redundant-roles.md) | error | error
167[no-static-element-interactions](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/no-static-element-interactions.md) | error, with options | error
168[role-has-required-aria-props](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/role-has-required-aria-props.md) | error | error
169[role-supports-aria-props](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/role-supports-aria-props.md) | error | error
170[scope](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/scope.md) | error, with options | error
171[tabindex-no-positive](https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/tabindex-no-positive.md) | error | error
172
173
174The following rules have extra options when in *recommended* mode:
175
176#### no-interactive-element-to-noninteractive-role
177```js
178'jsx-a11y/no-interactive-element-to-noninteractive-role': [
179 'error',
180 {
181 tr: ['none', 'presentation'],
182 },
183]
184```
185
186#### no-noninteractive-element-interactions
187```js
188'jsx-a11y/no-noninteractive-element-interactions': [
189 'error',
190 {
191 handlers: [
192 'onClick',
193 'onMouseDown',
194 'onMouseUp',
195 'onKeyPress',
196 'onKeyDown',
197 'onKeyUp',
198 ],
199 },
200]
201```
202
203#### no-noninteractive-element-to-interactive-role
204```js
205'jsx-a11y/no-noninteractive-element-to-interactive-role': [
206 'error',
207 {
208 ul: [
209 'listbox',
210 'menu',
211 'menubar',
212 'radiogroup',
213 'tablist',
214 'tree',
215 'treegrid',
216 ],
217 ol: [
218 'listbox',
219 'menu',
220 'menubar',
221 'radiogroup',
222 'tablist',
223 'tree',
224 'treegrid',
225 ],
226 li: ['menuitem', 'option', 'row', 'tab', 'treeitem'],
227 table: ['grid'],
228 td: ['gridcell'],
229 },
230]
231```
232
233#### no-noninteractive-tabindex
234```js
235'jsx-a11y/no-noninteractive-tabindex': [
236 'error',
237 {
238 tags: [],
239 roles: ['tabpanel'],
240 },
241]
242```
243
244#### no-static-element-interactions
245```js
246'jsx-a11y/no-noninteractive-element-interactions': [
247 'error',
248 {
249 handlers: [
250 'onClick',
251 'onMouseDown',
252 'onMouseUp',
253 'onKeyPress',
254 'onKeyDown',
255 'onKeyUp',
256 ],
257 },
258]
259```
260
261## Creating a new rule
262
263If you are developing new rules for this project, you can use the `create-rule`
264script to scaffold the new files.
265
266```
267$ ./scripts/create-rule.js my-new-rule
268```
269
270## Some background on WAI-ARIA, the AX Tree and Browsers
271
272### Accessibility API
273An operating system will provide an accessibility API that maps application state and content onto input/output controllers such as a screen reader, braille device, keyboard, etc.
274
275These APIs were developed as computer interfaces shifted from buffers (which are text-based and inherently quite accessible) to graphical user interfaces (GUIs). The first attempts to make GUIs accessible involved raster image parsing to recognize characters, words, etc. This information was stored in a parallel buffer and made accessible to assistive technology (AT) devices.
276
277As GUIs became more complex, the raster parsing approach became untenable. Accessibility APIs were developed to replace them. Check out [NSAccessibility (AXAPI)](https://developer.apple.com/library/mac/documentation/Cocoa/Reference/ApplicationKit/Protocols/NSAccessibility_Protocol/index.html) for an example. See [Core Accessibility API Mappings 1.1](https://www.w3.org/TR/core-aam-1.1/) for more details.
278
279### Browsers
280Browsers support an Accessibility API on a per operating system basis. For instance, Firefox implements the MSAA accessibility API on Windows, but does not implement the AXAPI on OSX.
281
282### The Accessibility (AX) Tree & DOM
283From the [W3 Core Accessibility API Mappings 1.1](https://www.w3.org/TR/core-aam-1.1/#intro_treetypes)
284
285> The accessibility tree and the DOM tree are parallel structures. Roughly speaking the accessibility tree is a subset of the DOM tree. It includes the user interface objects of the user agent and the objects of the document. Accessible objects are created in the accessibility tree for every DOM element that should be exposed to assistive technology, either because it may fire an accessibility event or because it has a property, relationship or feature which needs to be exposed. Generally, if something can be trimmed out it will be, for reasons of performance and simplicity. For example, a `<span>` with just a style change and no semantics may not get its own accessible object, but the style change will be exposed by other means.
286
287Browser vendors are beginning to expose the AX Tree through inspection tools. Chrome has an experiment available to enable their inspection tool.
288
289You can also see a text-based version of the AX Tree in Chrome in the stable release version.
290
291#### Viewing the AX Tree in Chrome
292 1. Navigate to `chrome://accessibility/` in Chrome.
293 1. Toggle the `accessibility off` link for any tab that you want to inspect.
294 1. A link labeled `show accessibility tree` will appear; click this link.
295 1. Balk at the wall of text that gets displayed, but then regain your conviction.
296 1. Use the browser's find command to locate strings and values in the wall of text.
297
298### Pulling it all together
299A browser constructs an AX Tree as a subset of the DOM. ARIA heavily informs the properties of this AX Tree. This AX Tree is exposed to the system level Accessibility API which mediates assistive technology agents.
300
301We model ARIA in the [aria-query](https://github.com/a11yance/aria-query) project. We model AXObjects (that comprise the AX Tree) in the [axobject-query](https://github.com/A11yance/axobject-query) project. The goal of the WAI-ARIA specification is to be a complete declarative interface to the AXObject model. The [in-draft 1.2 version](https://github.com/w3c/aria/issues?q=is%3Aissue+is%3Aopen+label%3A%22ARIA+1.2%22) is moving towards this goal. But until then, we must consider the semantics constructs afforded by ARIA as well as those afforded by the AXObject model (AXAPI) in order to determine how HTML can be used to express user interface affordances to assistive technology users.
302
303## License
304
305eslint-plugin-jsx-a11y is licensed under the [MIT License](LICENSE.md).