@arcaauth/eslint-plugin-jsx-a11y 6.10.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (229) hide show
  1. package/.babelrc +17 -0
  2. package/.eslintrc +44 -0
  3. package/CHANGELOG.md +774 -0
  4. package/LICENSE.md +8 -0
  5. package/README.md +423 -0
  6. package/__mocks__/IdentifierMock.js +15 -0
  7. package/__mocks__/JSXAttributeMock.js +39 -0
  8. package/__mocks__/JSXElementMock.js +37 -0
  9. package/__mocks__/JSXExpressionContainerMock.js +15 -0
  10. package/__mocks__/JSXSpreadAttributeMock.js +18 -0
  11. package/__mocks__/JSXTextMock.js +17 -0
  12. package/__mocks__/LiteralMock.js +17 -0
  13. package/__mocks__/genInteractives.js +218 -0
  14. package/__tests__/__util__/axeMapping.js +6 -0
  15. package/__tests__/__util__/helpers/getESLintCoreRule.js +9 -0
  16. package/__tests__/__util__/helpers/parsers.js +186 -0
  17. package/__tests__/__util__/parserOptionsMapper.js +53 -0
  18. package/__tests__/__util__/ruleOptionsMapperFactory.js +33 -0
  19. package/__tests__/index-test.js +40 -0
  20. package/__tests__/src/rules/accessible-emoji-test.js +66 -0
  21. package/__tests__/src/rules/alt-text-test.js +291 -0
  22. package/__tests__/src/rules/anchor-ambiguous-text-test.js +117 -0
  23. package/__tests__/src/rules/anchor-has-content-test.js +54 -0
  24. package/__tests__/src/rules/anchor-is-valid-test.js +532 -0
  25. package/__tests__/src/rules/aria-activedescendant-has-tabindex-test.js +95 -0
  26. package/__tests__/src/rules/aria-props-test.js +69 -0
  27. package/__tests__/src/rules/aria-proptypes-test.js +311 -0
  28. package/__tests__/src/rules/aria-role-test.js +118 -0
  29. package/__tests__/src/rules/aria-unsupported-elements-test.js +75 -0
  30. package/__tests__/src/rules/autocomplete-valid-test.js +77 -0
  31. package/__tests__/src/rules/click-events-have-key-events-test.js +77 -0
  32. package/__tests__/src/rules/control-has-associated-label-test.js +327 -0
  33. package/__tests__/src/rules/heading-has-content-test.js +85 -0
  34. package/__tests__/src/rules/html-has-lang-test.js +42 -0
  35. package/__tests__/src/rules/iframe-has-title-test.js +55 -0
  36. package/__tests__/src/rules/img-redundant-alt-test.js +137 -0
  37. package/__tests__/src/rules/interactive-supports-focus-test.js +267 -0
  38. package/__tests__/src/rules/label-has-associated-control-test.js +243 -0
  39. package/__tests__/src/rules/label-has-for-test.js +235 -0
  40. package/__tests__/src/rules/lang-test.js +59 -0
  41. package/__tests__/src/rules/media-has-caption-test.js +220 -0
  42. package/__tests__/src/rules/mouse-events-have-key-events-test.js +154 -0
  43. package/__tests__/src/rules/no-access-key-test.js +48 -0
  44. package/__tests__/src/rules/no-aria-hidden-on-focusable-test.js +44 -0
  45. package/__tests__/src/rules/no-autofocus-test.js +68 -0
  46. package/__tests__/src/rules/no-distracting-elements-test.js +51 -0
  47. package/__tests__/src/rules/no-interactive-element-to-noninteractive-role-test.js +405 -0
  48. package/__tests__/src/rules/no-noninteractive-element-interactions-test.js +502 -0
  49. package/__tests__/src/rules/no-noninteractive-element-to-interactive-role-test.js +500 -0
  50. package/__tests__/src/rules/no-noninteractive-tabindex-test.js +123 -0
  51. package/__tests__/src/rules/no-onchange-test.js +57 -0
  52. package/__tests__/src/rules/no-redundant-roles-test.js +98 -0
  53. package/__tests__/src/rules/no-static-element-interactions-test.js +501 -0
  54. package/__tests__/src/rules/prefer-tag-over-role-test.js +63 -0
  55. package/__tests__/src/rules/role-has-required-aria-props-test.js +134 -0
  56. package/__tests__/src/rules/role-supports-aria-props-test.js +570 -0
  57. package/__tests__/src/rules/scope-test.js +50 -0
  58. package/__tests__/src/rules/tabindex-no-positive-test.js +55 -0
  59. package/__tests__/src/util/attributesComparator-test.js +91 -0
  60. package/__tests__/src/util/getAccessibleChildText-test.js +174 -0
  61. package/__tests__/src/util/getComputedRole-test.js +71 -0
  62. package/__tests__/src/util/getElementType-test.js +154 -0
  63. package/__tests__/src/util/getExplicitRole-test.js +35 -0
  64. package/__tests__/src/util/getImplicitRole-test.js +25 -0
  65. package/__tests__/src/util/getSuggestion-test.js +33 -0
  66. package/__tests__/src/util/getTabIndex-test.js +85 -0
  67. package/__tests__/src/util/hasAccessibleChild-test.js +157 -0
  68. package/__tests__/src/util/implicitRoles/input-test.js +87 -0
  69. package/__tests__/src/util/implicitRoles/menu-test.js +20 -0
  70. package/__tests__/src/util/implicitRoles/menuitem-test.js +38 -0
  71. package/__tests__/src/util/isAbstractRole-test.js +51 -0
  72. package/__tests__/src/util/isContentEditable-test.js +52 -0
  73. package/__tests__/src/util/isDOMElement-test.js +30 -0
  74. package/__tests__/src/util/isDisabledElement-test.js +88 -0
  75. package/__tests__/src/util/isFocusable-test.js +111 -0
  76. package/__tests__/src/util/isInteractiveElement-test.js +104 -0
  77. package/__tests__/src/util/isInteractiveRole-test.js +59 -0
  78. package/__tests__/src/util/isNonInteractiveElement-test.js +97 -0
  79. package/__tests__/src/util/isNonInteractiveRole-test.js +59 -0
  80. package/__tests__/src/util/isNonLiteralProperty-test.js +52 -0
  81. package/__tests__/src/util/isSemanticRoleElement-test.js +72 -0
  82. package/__tests__/src/util/mayContainChildComponent-test.js +219 -0
  83. package/__tests__/src/util/mayHaveAccessibleLabel-test.js +256 -0
  84. package/__tests__/src/util/parserOptionsMapper-test.js +93 -0
  85. package/__tests__/src/util/schemas-test.js +35 -0
  86. package/docs/rules/accessible-emoji.md +30 -0
  87. package/docs/rules/alt-text.md +168 -0
  88. package/docs/rules/anchor-ambiguous-text.md +91 -0
  89. package/docs/rules/anchor-has-content.md +64 -0
  90. package/docs/rules/anchor-is-valid.md +270 -0
  91. package/docs/rules/aria-activedescendant-has-tabindex.md +52 -0
  92. package/docs/rules/aria-props.md +29 -0
  93. package/docs/rules/aria-proptypes.md +30 -0
  94. package/docs/rules/aria-role.md +51 -0
  95. package/docs/rules/aria-unsupported-elements.md +30 -0
  96. package/docs/rules/autocomplete-valid.md +49 -0
  97. package/docs/rules/click-events-have-key-events.md +28 -0
  98. package/docs/rules/control-has-associated-label.md +113 -0
  99. package/docs/rules/heading-has-content.md +67 -0
  100. package/docs/rules/html-has-lang.md +31 -0
  101. package/docs/rules/iframe-has-title.md +37 -0
  102. package/docs/rules/img-redundant-alt.md +48 -0
  103. package/docs/rules/interactive-supports-focus.md +156 -0
  104. package/docs/rules/label-has-associated-control.md +152 -0
  105. package/docs/rules/label-has-for.md +130 -0
  106. package/docs/rules/lang.md +31 -0
  107. package/docs/rules/media-has-caption.md +48 -0
  108. package/docs/rules/mouse-events-have-key-events.md +58 -0
  109. package/docs/rules/no-access-key.md +30 -0
  110. package/docs/rules/no-aria-hidden-on-focusable.md +37 -0
  111. package/docs/rules/no-autofocus.md +43 -0
  112. package/docs/rules/no-distracting-elements.md +41 -0
  113. package/docs/rules/no-interactive-element-to-noninteractive-role.md +73 -0
  114. package/docs/rules/no-noninteractive-element-interactions.md +145 -0
  115. package/docs/rules/no-noninteractive-element-to-interactive-role.md +76 -0
  116. package/docs/rules/no-noninteractive-tabindex.md +115 -0
  117. package/docs/rules/no-onchange.md +36 -0
  118. package/docs/rules/no-redundant-roles.md +46 -0
  119. package/docs/rules/no-static-element-interactions.md +114 -0
  120. package/docs/rules/prefer-tag-over-role.md +32 -0
  121. package/docs/rules/role-has-required-aria-props.md +31 -0
  122. package/docs/rules/role-supports-aria-props.md +39 -0
  123. package/docs/rules/scope.md +30 -0
  124. package/docs/rules/tabindex-no-positive.md +32 -0
  125. package/lib/configs/flat-config-base.js +11 -0
  126. package/lib/configs/legacy-config-base.js +9 -0
  127. package/lib/index.js +209 -0
  128. package/lib/rules/accessible-emoji.js +63 -0
  129. package/lib/rules/alt-text.js +218 -0
  130. package/lib/rules/anchor-ambiguous-text.js +64 -0
  131. package/lib/rules/anchor-has-content.js +60 -0
  132. package/lib/rules/anchor-is-valid.js +122 -0
  133. package/lib/rules/aria-activedescendant-has-tabindex.js +66 -0
  134. package/lib/rules/aria-props.js +59 -0
  135. package/lib/rules/aria-proptypes.js +114 -0
  136. package/lib/rules/aria-role.js +89 -0
  137. package/lib/rules/aria-unsupported-elements.js +64 -0
  138. package/lib/rules/autocomplete-valid.js +67 -0
  139. package/lib/rules/click-events-have-key-events.js +68 -0
  140. package/lib/rules/control-has-associated-label.js +103 -0
  141. package/lib/rules/heading-has-content.js +61 -0
  142. package/lib/rules/html-has-lang.js +50 -0
  143. package/lib/rules/iframe-has-title.js +50 -0
  144. package/lib/rules/img-redundant-alt.js +88 -0
  145. package/lib/rules/interactive-supports-focus.js +87 -0
  146. package/lib/rules/label-has-associated-control.js +127 -0
  147. package/lib/rules/label-has-for.js +150 -0
  148. package/lib/rules/lang.js +68 -0
  149. package/lib/rules/media-has-caption.js +96 -0
  150. package/lib/rules/mouse-events-have-key-events.js +94 -0
  151. package/lib/rules/no-access-key.js +43 -0
  152. package/lib/rules/no-aria-hidden-on-focusable.js +47 -0
  153. package/lib/rules/no-autofocus.js +62 -0
  154. package/lib/rules/no-distracting-elements.js +54 -0
  155. package/lib/rules/no-interactive-element-to-noninteractive-role.js +81 -0
  156. package/lib/rules/no-noninteractive-element-interactions.js +95 -0
  157. package/lib/rules/no-noninteractive-element-to-interactive-role.js +80 -0
  158. package/lib/rules/no-noninteractive-tabindex.js +109 -0
  159. package/lib/rules/no-onchange.js +52 -0
  160. package/lib/rules/no-redundant-roles.js +86 -0
  161. package/lib/rules/no-static-element-interactions.js +102 -0
  162. package/lib/rules/prefer-tag-over-role.js +75 -0
  163. package/lib/rules/role-has-required-aria-props.js +88 -0
  164. package/lib/rules/role-supports-aria-props.js +78 -0
  165. package/lib/rules/scope.js +58 -0
  166. package/lib/rules/tabindex-no-positive.js +53 -0
  167. package/lib/util/attributesComparator.js +34 -0
  168. package/lib/util/getAccessibleChildText.js +55 -0
  169. package/lib/util/getComputedRole.js +19 -0
  170. package/lib/util/getElementType.js +30 -0
  171. package/lib/util/getExplicitRole.js +27 -0
  172. package/lib/util/getImplicitRole.js +24 -0
  173. package/lib/util/getSuggestion.js +32 -0
  174. package/lib/util/getTabIndex.js +34 -0
  175. package/lib/util/hasAccessibleChild.js +30 -0
  176. package/lib/util/implicitRoles/a.js +17 -0
  177. package/lib/util/implicitRoles/area.js +17 -0
  178. package/lib/util/implicitRoles/article.js +13 -0
  179. package/lib/util/implicitRoles/aside.js +13 -0
  180. package/lib/util/implicitRoles/body.js +13 -0
  181. package/lib/util/implicitRoles/button.js +13 -0
  182. package/lib/util/implicitRoles/datalist.js +13 -0
  183. package/lib/util/implicitRoles/details.js +13 -0
  184. package/lib/util/implicitRoles/dialog.js +13 -0
  185. package/lib/util/implicitRoles/form.js +13 -0
  186. package/lib/util/implicitRoles/h1.js +13 -0
  187. package/lib/util/implicitRoles/h2.js +13 -0
  188. package/lib/util/implicitRoles/h3.js +13 -0
  189. package/lib/util/implicitRoles/h4.js +13 -0
  190. package/lib/util/implicitRoles/h5.js +13 -0
  191. package/lib/util/implicitRoles/h6.js +13 -0
  192. package/lib/util/implicitRoles/hr.js +13 -0
  193. package/lib/util/implicitRoles/img.js +31 -0
  194. package/lib/util/implicitRoles/index.js +82 -0
  195. package/lib/util/implicitRoles/input.js +38 -0
  196. package/lib/util/implicitRoles/li.js +13 -0
  197. package/lib/util/implicitRoles/link.js +17 -0
  198. package/lib/util/implicitRoles/menu.js +19 -0
  199. package/lib/util/implicitRoles/menuitem.js +28 -0
  200. package/lib/util/implicitRoles/meter.js +13 -0
  201. package/lib/util/implicitRoles/nav.js +13 -0
  202. package/lib/util/implicitRoles/ol.js +13 -0
  203. package/lib/util/implicitRoles/option.js +13 -0
  204. package/lib/util/implicitRoles/output.js +13 -0
  205. package/lib/util/implicitRoles/progress.js +13 -0
  206. package/lib/util/implicitRoles/section.js +13 -0
  207. package/lib/util/implicitRoles/select.js +13 -0
  208. package/lib/util/implicitRoles/tbody.js +13 -0
  209. package/lib/util/implicitRoles/textarea.js +13 -0
  210. package/lib/util/implicitRoles/tfoot.js +13 -0
  211. package/lib/util/implicitRoles/thead.js +13 -0
  212. package/lib/util/implicitRoles/ul.js +13 -0
  213. package/lib/util/isAbstractRole.js +23 -0
  214. package/lib/util/isContentEditable.js +13 -0
  215. package/lib/util/isDOMElement.js +15 -0
  216. package/lib/util/isDisabledElement.js +23 -0
  217. package/lib/util/isFocusable.js +23 -0
  218. package/lib/util/isHiddenFromScreenReader.js +26 -0
  219. package/lib/util/isInteractiveElement.js +116 -0
  220. package/lib/util/isInteractiveRole.js +54 -0
  221. package/lib/util/isNonInteractiveElement.js +131 -0
  222. package/lib/util/isNonInteractiveRole.js +55 -0
  223. package/lib/util/isNonLiteralProperty.js +29 -0
  224. package/lib/util/isPresentationRole.js +13 -0
  225. package/lib/util/isSemanticRoleElement.js +54 -0
  226. package/lib/util/mayContainChildComponent.js +50 -0
  227. package/lib/util/mayHaveAccessibleLabel.js +95 -0
  228. package/lib/util/schemas.js +52 -0
  229. package/package.json +120 -0
@@ -0,0 +1,63 @@
1
+ import { RuleTester } from 'eslint';
2
+ import parserOptionsMapper from '../../__util__/parserOptionsMapper';
3
+ import parsers from '../../__util__/helpers/parsers';
4
+ import rule from '../../../src/rules/prefer-tag-over-role';
5
+
6
+ const ruleTester = new RuleTester();
7
+
8
+ const expectedError = (role, tag) => ({
9
+ message: `Use ${tag} instead of the "${role}" role to ensure accessibility across all devices.`,
10
+ type: 'JSXOpeningElement',
11
+ });
12
+
13
+ ruleTester.run('prefer-tag-over-role', rule, {
14
+ valid: parsers.all([].concat(
15
+ { code: '<div />;' },
16
+ { code: '<div role="unknown" />;' },
17
+ { code: '<div role="also unknown" />;' },
18
+ { code: '<other />' },
19
+ { code: '<img role="img" />' },
20
+ { code: '<input role="checkbox" />' },
21
+ )).map(parserOptionsMapper),
22
+ invalid: parsers.all([].concat(
23
+ {
24
+ code: '<div role="checkbox" />',
25
+ errors: [expectedError('checkbox', '<input type="checkbox">')],
26
+ },
27
+ {
28
+ code: '<div role="button checkbox" />',
29
+ errors: [expectedError('checkbox', '<input type="checkbox">')],
30
+ },
31
+ {
32
+ code: '<div role="heading" />',
33
+ errors: [
34
+ expectedError('heading', '<h1>, <h2>, <h3>, <h4>, <h5>, or <h6>'),
35
+ ],
36
+ },
37
+ {
38
+ code: '<div role="link" />',
39
+ errors: [
40
+ expectedError(
41
+ 'link',
42
+ '<a href=...>, or <area href=...>',
43
+ ),
44
+ ],
45
+ },
46
+ {
47
+ code: '<div role="rowgroup" />',
48
+ errors: [expectedError('rowgroup', '<tbody>, <tfoot>, or <thead>')],
49
+ },
50
+ {
51
+ code: '<span role="checkbox" />',
52
+ errors: [expectedError('checkbox', '<input type="checkbox">')],
53
+ },
54
+ {
55
+ code: '<other role="checkbox" />',
56
+ errors: [expectedError('checkbox', '<input type="checkbox">')],
57
+ },
58
+ {
59
+ code: '<div role="banner" />',
60
+ errors: [expectedError('banner', '<header>')],
61
+ },
62
+ )).map(parserOptionsMapper),
63
+ });
@@ -0,0 +1,134 @@
1
+ /**
2
+ * @fileoverview Enforce that elements with ARIA roles must
3
+ * have all required attributes for that role.
4
+ * @author Ethan Cohen
5
+ */
6
+
7
+ // -----------------------------------------------------------------------------
8
+ // Requirements
9
+ // -----------------------------------------------------------------------------
10
+
11
+ import { roles } from 'aria-query';
12
+ import { RuleTester } from 'eslint';
13
+
14
+ import parserOptionsMapper from '../../__util__/parserOptionsMapper';
15
+ import parsers from '../../__util__/helpers/parsers';
16
+ import rule from '../../../src/rules/role-has-required-aria-props';
17
+
18
+ // -----------------------------------------------------------------------------
19
+ // Tests
20
+ // -----------------------------------------------------------------------------
21
+
22
+ const ruleTester = new RuleTester();
23
+
24
+ const errorMessage = (role) => {
25
+ const requiredProps = Object.keys(roles.get(role).requiredProps);
26
+
27
+ return {
28
+ message: `Elements with the ARIA role "${role}" must have the following attributes defined: ${requiredProps}`,
29
+ type: 'JSXAttribute',
30
+ };
31
+ };
32
+
33
+ const componentsSettings = {
34
+ 'jsx-a11y': {
35
+ components: {
36
+ MyComponent: 'div',
37
+ },
38
+ },
39
+ };
40
+
41
+ // Create basic test cases using all valid role types.
42
+ const basicValidityTests = roles.keys().map((role) => {
43
+ const {
44
+ requiredProps: requiredPropKeyValues,
45
+ } = roles.get(role);
46
+ const requiredProps = Object.keys(requiredPropKeyValues);
47
+ const propChain = requiredProps.join(' ');
48
+
49
+ return {
50
+ code: `<div role="${role.toLowerCase()}" ${propChain} />`,
51
+ };
52
+ });
53
+
54
+ ruleTester.run('role-has-required-aria-props', rule, {
55
+ valid: parsers.all([].concat(
56
+ { code: '<Bar baz />' },
57
+ { code: '<MyComponent role="combobox" />' },
58
+ // Variables should pass, as we are only testing literals.
59
+ { code: '<div />' },
60
+ { code: '<div></div>' },
61
+ { code: '<div role={role} />' },
62
+ { code: '<div role={role || "button"} />' },
63
+ { code: '<div role={role || "foobar"} />' },
64
+ { code: '<div role="row" />' },
65
+ { code: '<span role="checkbox" aria-checked="false" aria-labelledby="foo" tabindex="0"></span>' },
66
+ { code: '<input role="checkbox" aria-checked="false" aria-labelledby="foo" tabindex="0" {...props} type="checkbox" />' },
67
+ { code: '<input type="checkbox" role="switch" />' },
68
+ { code: '<MyComponent role="checkbox" aria-checked="false" aria-labelledby="foo" tabindex="0" />', settings: componentsSettings },
69
+ { code: '<div role="heading" aria-level={2} />' },
70
+ { code: '<div role="heading" aria-level="3" />' },
71
+ )).concat(basicValidityTests).map(parserOptionsMapper),
72
+
73
+ invalid: parsers.all([].concat(
74
+ // SLIDER
75
+ { code: '<div role="slider" />', errors: [errorMessage('slider')] },
76
+ {
77
+ code: '<div role="slider" aria-valuemax />',
78
+ errors: [errorMessage('slider')],
79
+ },
80
+ {
81
+ code: '<div role="slider" aria-valuemax aria-valuemin />',
82
+ errors: [errorMessage('slider')],
83
+ },
84
+
85
+ // CHECKBOX
86
+ { code: '<div role="checkbox" />', errors: [errorMessage('checkbox')] },
87
+ { code: '<div role="checkbox" checked />', errors: [errorMessage('checkbox')] },
88
+ {
89
+ code: '<div role="checkbox" aria-chcked />',
90
+ errors: [errorMessage('checkbox')],
91
+ },
92
+ {
93
+ code: '<span role="checkbox" aria-labelledby="foo" tabindex="0"></span>',
94
+ errors: [errorMessage('checkbox')],
95
+ },
96
+
97
+ // COMBOBOX
98
+ { code: '<div role="combobox" />', errors: [errorMessage('combobox')] },
99
+ { code: '<div role="combobox" expanded />', errors: [errorMessage('combobox')] },
100
+ {
101
+ code: '<div role="combobox" aria-expandd />',
102
+ errors: [errorMessage('combobox')],
103
+ },
104
+
105
+ // SCROLLBAR
106
+ { code: '<div role="scrollbar" />', errors: [errorMessage('scrollbar')] },
107
+ {
108
+ code: '<div role="scrollbar" aria-valuemax />',
109
+ errors: [errorMessage('scrollbar')],
110
+ },
111
+ {
112
+ code: '<div role="scrollbar" aria-valuemax aria-valuemin />',
113
+ errors: [errorMessage('scrollbar')],
114
+ },
115
+ {
116
+ code: '<div role="scrollbar" aria-valuemax aria-valuenow />',
117
+ errors: [errorMessage('scrollbar')],
118
+ },
119
+ {
120
+ code: '<div role="scrollbar" aria-valuemin aria-valuenow />',
121
+ errors: [errorMessage('scrollbar')],
122
+ },
123
+ {
124
+ code: '<div role="heading" />',
125
+ errors: [errorMessage('heading')],
126
+ },
127
+ {
128
+ code: '<div role="option" />',
129
+ errors: [errorMessage('option')],
130
+ },
131
+ // Custom element
132
+ { code: '<MyComponent role="combobox" />', settings: componentsSettings, errors: [errorMessage('combobox')] },
133
+ )).map(parserOptionsMapper),
134
+ });