@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,532 @@
1
+ /**
2
+ * @fileoverview Performs validity check on anchor hrefs. Warns when anchors are used as buttons.
3
+ * @author Almero Steyn
4
+ */
5
+
6
+ // -----------------------------------------------------------------------------
7
+ // Requirements
8
+ // -----------------------------------------------------------------------------
9
+
10
+ import { RuleTester } from 'eslint';
11
+ import parserOptionsMapper from '../../__util__/parserOptionsMapper';
12
+ import parsers from '../../__util__/helpers/parsers';
13
+ import rule from '../../../src/rules/anchor-is-valid';
14
+
15
+ // -----------------------------------------------------------------------------
16
+ // Tests
17
+ // -----------------------------------------------------------------------------
18
+
19
+ const ruleTester = new RuleTester();
20
+
21
+ const preferButtonErrorMessage = 'Anchor used as a button. Anchors are primarily expected to navigate. Use the button element instead. Learn more: https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/anchor-is-valid.md';
22
+
23
+ const noHrefErrorMessage = 'The href attribute is required for an anchor to be keyboard accessible. Provide a valid, navigable address as the href value. If you cannot provide an href, but still need the element to resemble a link, use a button and change it with appropriate styles. Learn more: https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/anchor-is-valid.md';
24
+
25
+ const invalidHrefErrorMessage = 'The href attribute requires a valid value to be accessible. Provide a valid, navigable address as the href value. If you cannot provide a valid href, but still need the element to resemble a link, use a button and change it with appropriate styles. Learn more: https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/HEAD/docs/rules/anchor-is-valid.md';
26
+
27
+ const preferButtonexpectedError = {
28
+ message: preferButtonErrorMessage,
29
+ type: 'JSXOpeningElement',
30
+ };
31
+ const noHrefexpectedError = {
32
+ message: noHrefErrorMessage,
33
+ type: 'JSXOpeningElement',
34
+ };
35
+ const invalidHrefexpectedError = {
36
+ message: invalidHrefErrorMessage,
37
+ type: 'JSXOpeningElement',
38
+ };
39
+
40
+ const components = [{
41
+ components: ['Anchor', 'Link'],
42
+ }];
43
+ const specialLink = [{
44
+ specialLink: ['hrefLeft', 'hrefRight'],
45
+ }];
46
+ const noHrefAspect = [{
47
+ aspects: ['noHref'],
48
+ }];
49
+ const invalidHrefAspect = [{
50
+ aspects: ['invalidHref'],
51
+ }];
52
+ const preferButtonAspect = [{
53
+ aspects: ['preferButton'],
54
+ }];
55
+ const noHrefInvalidHrefAspect = [{
56
+ aspects: ['noHref', 'invalidHref'],
57
+ }];
58
+ const noHrefPreferButtonAspect = [{
59
+ aspects: ['noHref', 'preferButton'],
60
+ }];
61
+ const preferButtonInvalidHrefAspect = [{
62
+ aspects: ['preferButton', 'invalidHref'],
63
+ }];
64
+
65
+ const componentsAndSpecialLink = [{
66
+ components: ['Anchor'],
67
+ specialLink: ['hrefLeft'],
68
+ }];
69
+
70
+ const componentsAndSpecialLinkAndInvalidHrefAspect = [{
71
+ components: ['Anchor'],
72
+ specialLink: ['hrefLeft'],
73
+ aspects: ['invalidHref'],
74
+ }];
75
+
76
+ const componentsAndSpecialLinkAndNoHrefAspect = [{
77
+ components: ['Anchor'],
78
+ specialLink: ['hrefLeft'],
79
+ aspects: ['noHref'],
80
+ }];
81
+
82
+ const componentsSettings = {
83
+ 'jsx-a11y': {
84
+ components: {
85
+ Anchor: 'a',
86
+ Link: 'a',
87
+ },
88
+ },
89
+ };
90
+
91
+ ruleTester.run('anchor-is-valid', rule, {
92
+ valid: parsers.all([].concat(
93
+ // DEFAULT ELEMENT 'a' TESTS
94
+ { code: '<Anchor />' },
95
+ { code: '<a {...props} />' },
96
+ { code: '<a href="foo" />' },
97
+ { code: '<a href={foo} />' },
98
+ { code: '<a href="/foo" />' },
99
+ { code: '<a href="https://foo.bar.com" />' },
100
+ { code: '<div href="foo" />' },
101
+ { code: '<a href="javascript" />' },
102
+ { code: '<a href="javascriptFoo" />' },
103
+ { code: '<a href={`#foo`}/>' },
104
+ { code: '<a href={"foo"}/>' },
105
+ { code: '<a href={"javascript"}/>' },
106
+ { code: '<a href={`#javascript`}/>' },
107
+ { code: '<a href="#foo" />' },
108
+ { code: '<a href="#javascript" />' },
109
+ { code: '<a href="#javascriptFoo" />' },
110
+ { code: '<UX.Layout>test</UX.Layout>' },
111
+ { code: '<a href={this} />' },
112
+
113
+ // CUSTOM ELEMENT TEST FOR ARRAY OPTION
114
+ { code: '<Anchor {...props} />', options: components },
115
+ { code: '<Anchor href="foo" />', options: components },
116
+ { code: '<Anchor href={foo} />', options: components },
117
+ { code: '<Anchor href="/foo" />', options: components },
118
+ { code: '<Anchor href="https://foo.bar.com" />', options: components },
119
+ { code: '<div href="foo" />', options: components },
120
+ { code: '<Anchor href={`#foo`}/>', options: components },
121
+ { code: '<Anchor href={"foo"}/>', options: components },
122
+ { code: '<Anchor href="#foo" />', options: components },
123
+ { code: '<Link {...props} />', options: components },
124
+ { code: '<Link href="foo" />', options: components },
125
+ { code: '<Link href={foo} />', options: components },
126
+ { code: '<Link href="/foo" />', options: components },
127
+ { code: '<Link href="https://foo.bar.com" />', options: components },
128
+ { code: '<div href="foo" />', options: components },
129
+ { code: '<Link href={`#foo`}/>', options: components },
130
+ { code: '<Link href={"foo"}/>', options: components },
131
+ { code: '<Link href="#foo" />', options: components },
132
+ { code: '<Link href="#foo" />', settings: componentsSettings },
133
+
134
+ // CUSTOM PROP TESTS
135
+ { code: '<a {...props} />', options: specialLink },
136
+ { code: '<a hrefLeft="foo" />', options: specialLink },
137
+ { code: '<a hrefLeft={foo} />', options: specialLink },
138
+ { code: '<a hrefLeft="/foo" />', options: specialLink },
139
+ { code: '<a hrefLeft="https://foo.bar.com" />', options: specialLink },
140
+ { code: '<div hrefLeft="foo" />', options: specialLink },
141
+ { code: '<a hrefLeft={`#foo`}/>', options: specialLink },
142
+ { code: '<a hrefLeft={"foo"}/>', options: specialLink },
143
+ { code: '<a hrefLeft="#foo" />', options: specialLink },
144
+ { code: '<UX.Layout>test</UX.Layout>', options: specialLink },
145
+ { code: '<a hrefRight={this} />', options: specialLink },
146
+ { code: '<a {...props} />', options: specialLink },
147
+ { code: '<a hrefRight="foo" />', options: specialLink },
148
+ { code: '<a hrefRight={foo} />', options: specialLink },
149
+ { code: '<a hrefRight="/foo" />', options: specialLink },
150
+ { code: '<a hrefRight="https://foo.bar.com" />', options: specialLink },
151
+ { code: '<div hrefRight="foo" />', options: specialLink },
152
+ { code: '<a hrefRight={`#foo`}/>', options: specialLink },
153
+ { code: '<a hrefRight={"foo"}/>', options: specialLink },
154
+ { code: '<a hrefRight="#foo" />', options: specialLink },
155
+ { code: '<UX.Layout>test</UX.Layout>', options: specialLink },
156
+ { code: '<a hrefRight={this} />', options: specialLink },
157
+
158
+ // CUSTOM BOTH COMPONENTS AND SPECIALLINK TESTS
159
+ { code: '<Anchor {...props} />', options: componentsAndSpecialLink },
160
+ { code: '<Anchor hrefLeft="foo" />', options: componentsAndSpecialLink },
161
+ { code: '<Anchor hrefLeft={foo} />', options: componentsAndSpecialLink },
162
+ { code: '<Anchor hrefLeft="/foo" />', options: componentsAndSpecialLink },
163
+ { code: '<Anchor hrefLeft="https://foo.bar.com" />', options: componentsAndSpecialLink },
164
+ { code: '<div hrefLeft="foo" />', options: componentsAndSpecialLink },
165
+ { code: '<Anchor hrefLeft={`#foo`}/>', options: componentsAndSpecialLink },
166
+ { code: '<Anchor hrefLeft={"foo"}/>', options: componentsAndSpecialLink },
167
+ { code: '<Anchor hrefLeft="#foo" />', options: componentsAndSpecialLink },
168
+ { code: '<UX.Layout>test</UX.Layout>', options: componentsAndSpecialLink },
169
+
170
+ // WITH ONCLICK
171
+ // DEFAULT ELEMENT 'a' TESTS
172
+ { code: '<a {...props} onClick={() => void 0} />' },
173
+ { code: '<a href="foo" onClick={() => void 0} />' },
174
+ { code: '<a href={foo} onClick={() => void 0} />' },
175
+ { code: '<a href="/foo" onClick={() => void 0} />' },
176
+ { code: '<a href="https://foo.bar.com" onClick={() => void 0} />' },
177
+ { code: '<div href="foo" onClick={() => void 0} />' },
178
+ { code: '<a href={`#foo`} onClick={() => void 0} />' },
179
+ { code: '<a href={"foo"} onClick={() => void 0} />' },
180
+ { code: '<a href="#foo" onClick={() => void 0} />' },
181
+ { code: '<a href={this} onClick={() => void 0} />' },
182
+
183
+ // CUSTOM ELEMENT TEST FOR ARRAY OPTION
184
+ { code: '<Anchor {...props} onClick={() => void 0} />', options: components },
185
+ { code: '<Anchor href="foo" onClick={() => void 0} />', options: components },
186
+ { code: '<Anchor href={foo} onClick={() => void 0} />', options: components },
187
+ { code: '<Anchor href="/foo" onClick={() => void 0} />', options: components },
188
+ { code: '<Anchor href="https://foo.bar.com" onClick={() => void 0} />', options: components },
189
+ { code: '<Anchor href={`#foo`} onClick={() => void 0} />', options: components },
190
+ { code: '<Anchor href={"foo"} onClick={() => void 0} />', options: components },
191
+ { code: '<Anchor href="#foo" onClick={() => void 0} />', options: components },
192
+ { code: '<Link {...props} onClick={() => void 0} />', options: components },
193
+ { code: '<Link href="foo" onClick={() => void 0} />', options: components },
194
+ { code: '<Link href={foo} onClick={() => void 0} />', options: components },
195
+ { code: '<Link href="/foo" onClick={() => void 0} />', options: components },
196
+ { code: '<Link href="https://foo.bar.com" onClick={() => void 0} />', options: components },
197
+ { code: '<div href="foo" onClick={() => void 0} />', options: components },
198
+ { code: '<Link href={`#foo`} onClick={() => void 0} />', options: components },
199
+ { code: '<Link href={"foo"} onClick={() => void 0} />', options: components },
200
+ { code: '<Link href="#foo" onClick={() => void 0} />', options: components },
201
+
202
+ // CUSTOM PROP TESTS
203
+ { code: '<a {...props} onClick={() => void 0} />', options: specialLink },
204
+ { code: '<a hrefLeft="foo" onClick={() => void 0} />', options: specialLink },
205
+ { code: '<a hrefLeft={foo} onClick={() => void 0} />', options: specialLink },
206
+ { code: '<a hrefLeft="/foo" onClick={() => void 0} />', options: specialLink },
207
+ { code: '<a hrefLeft href="https://foo.bar.com" onClick={() => void 0} />', options: specialLink },
208
+ { code: '<div hrefLeft="foo" onClick={() => void 0} />', options: specialLink },
209
+ { code: '<a hrefLeft={`#foo`} onClick={() => void 0} />', options: specialLink },
210
+ { code: '<a hrefLeft={"foo"} onClick={() => void 0} />', options: specialLink },
211
+ { code: '<a hrefLeft="#foo" onClick={() => void 0} />', options: specialLink },
212
+ { code: '<a hrefRight={this} onClick={() => void 0} />', options: specialLink },
213
+ { code: '<a {...props} onClick={() => void 0} />', options: specialLink },
214
+ { code: '<a hrefRight="foo" onClick={() => void 0} />', options: specialLink },
215
+ { code: '<a hrefRight={foo} onClick={() => void 0} />', options: specialLink },
216
+ { code: '<a hrefRight="/foo" onClick={() => void 0} />', options: specialLink },
217
+ { code: '<a hrefRight href="https://foo.bar.com" onClick={() => void 0} />', options: specialLink },
218
+ { code: '<div hrefRight="foo" onClick={() => void 0} />', options: specialLink },
219
+ { code: '<a hrefRight={`#foo`} onClick={() => void 0} />', options: specialLink },
220
+ { code: '<a hrefRight={"foo"} onClick={() => void 0} />', options: specialLink },
221
+ { code: '<a hrefRight="#foo" onClick={() => void 0} />', options: specialLink },
222
+ { code: '<a hrefRight={this} onClick={() => void 0} />', options: specialLink },
223
+
224
+ // CUSTOM BOTH COMPONENTS AND SPECIALLINK TESTS
225
+ { code: '<Anchor {...props} onClick={() => void 0} />', options: componentsAndSpecialLink },
226
+ { code: '<Anchor hrefLeft="foo" onClick={() => void 0} />', options: componentsAndSpecialLink },
227
+ { code: '<Anchor hrefLeft={foo} onClick={() => void 0} />', options: componentsAndSpecialLink },
228
+ { code: '<Anchor hrefLeft="/foo" onClick={() => void 0} />', options: componentsAndSpecialLink },
229
+ {
230
+ code: '<Anchor hrefLeft href="https://foo.bar.com" onClick={() => void 0} />',
231
+ options: componentsAndSpecialLink,
232
+ },
233
+ { code: '<Anchor hrefLeft={`#foo`} onClick={() => void 0} />', options: componentsAndSpecialLink },
234
+ { code: '<Anchor hrefLeft={"foo"} onClick={() => void 0} />', options: componentsAndSpecialLink },
235
+ { code: '<Anchor hrefLeft="#foo" onClick={() => void 0} />', options: componentsAndSpecialLink },
236
+
237
+ // WITH ASPECTS TESTS
238
+ // NO HREF
239
+ { code: '<a />', options: invalidHrefAspect },
240
+ { code: '<a href={undefined} />', options: invalidHrefAspect },
241
+ { code: '<a href={null} />', options: invalidHrefAspect },
242
+ { code: '<a />', options: preferButtonAspect },
243
+ { code: '<a href={undefined} />', options: preferButtonAspect },
244
+ { code: '<a href={null} />', options: preferButtonAspect },
245
+ { code: '<a />', options: preferButtonInvalidHrefAspect },
246
+ { code: '<a href={undefined} />', options: preferButtonInvalidHrefAspect },
247
+ { code: '<a href={null} />', options: preferButtonInvalidHrefAspect },
248
+
249
+ // INVALID HREF
250
+ { code: '<a href="" />;', options: preferButtonAspect },
251
+ { code: '<a href="#" />', options: preferButtonAspect },
252
+ { code: '<a href={"#"} />', options: preferButtonAspect },
253
+ { code: '<a href="javascript:void(0)" />', options: preferButtonAspect },
254
+ { code: '<a href={"javascript:void(0)"} />', options: preferButtonAspect },
255
+ { code: '<a href="" />;', options: noHrefAspect },
256
+ { code: '<a href="#" />', options: noHrefAspect },
257
+ { code: '<a href={"#"} />', options: noHrefAspect },
258
+ { code: '<a href="javascript:void(0)" />', options: noHrefAspect },
259
+ { code: '<a href={"javascript:void(0)"} />', options: noHrefAspect },
260
+ { code: '<a href="" />;', options: noHrefPreferButtonAspect },
261
+ { code: '<a href="#" />', options: noHrefPreferButtonAspect },
262
+ { code: '<a href={"#"} />', options: noHrefPreferButtonAspect },
263
+ { code: '<a href="javascript:void(0)" />', options: noHrefPreferButtonAspect },
264
+ { code: '<a href={"javascript:void(0)"} />', options: noHrefPreferButtonAspect },
265
+
266
+ // SHOULD BE BUTTON
267
+ { code: '<a onClick={() => void 0} />', options: invalidHrefAspect },
268
+ { code: '<a href="#" onClick={() => void 0} />', options: noHrefAspect },
269
+ { code: '<a href="javascript:void(0)" onClick={() => void 0} />', options: noHrefAspect },
270
+ {
271
+ code: '<a href={"javascript:void(0)"} onClick={() => void 0} />',
272
+ options: noHrefAspect,
273
+ },
274
+
275
+ // CUSTOM COMPONENTS AND SPECIAL LINK AND ASPECT
276
+ { code: '<Anchor hrefLeft={undefined} />', options: componentsAndSpecialLinkAndInvalidHrefAspect },
277
+ { code: '<Anchor hrefLeft={null} />', options: componentsAndSpecialLinkAndInvalidHrefAspect },
278
+ )).map(parserOptionsMapper),
279
+ invalid: parsers.all([].concat(
280
+ // DEFAULT ELEMENT 'a' TESTS
281
+ // NO HREF
282
+ { code: '<a />', errors: [noHrefexpectedError] },
283
+ { code: '<a href={undefined} />', errors: [noHrefexpectedError] },
284
+ { code: '<a href={null} />', errors: [noHrefexpectedError] },
285
+ // INVALID HREF
286
+ { code: '<a href="" />;', errors: [invalidHrefexpectedError] },
287
+ { code: '<a href="#" />', errors: [invalidHrefexpectedError] },
288
+ { code: '<a href={"#"} />', errors: [invalidHrefexpectedError] },
289
+ { code: '<a href="javascript:void(0)" />', errors: [invalidHrefexpectedError] },
290
+ { code: '<a href={"javascript:void(0)"} />', errors: [invalidHrefexpectedError] },
291
+ // SHOULD BE BUTTON
292
+ { code: '<a onClick={() => void 0} />', errors: [preferButtonexpectedError] },
293
+ { code: '<a href="#" onClick={() => void 0} />', errors: [preferButtonexpectedError] },
294
+ { code: '<a href="javascript:void(0)" onClick={() => void 0} />', errors: [preferButtonexpectedError] },
295
+ {
296
+ code: '<a href={"javascript:void(0)"} onClick={() => void 0} />',
297
+ errors: [preferButtonexpectedError],
298
+ },
299
+
300
+ // CUSTOM ELEMENT TEST FOR ARRAY OPTION
301
+ // NO HREF
302
+ { code: '<Link />', errors: [noHrefexpectedError], options: components },
303
+ { code: '<Link href={undefined} />', errors: [noHrefexpectedError], options: components },
304
+ { code: '<Link href={null} />', errors: [noHrefexpectedError], options: components },
305
+ // INVALID HREF
306
+ { code: '<Link href="" />', errors: [invalidHrefexpectedError], options: components },
307
+ { code: '<Link href="#" />', errors: [invalidHrefexpectedError], options: components },
308
+ { code: '<Link href={"#"} />', errors: [invalidHrefexpectedError], options: components },
309
+ { code: '<Link href="javascript:void(0)" />', errors: [invalidHrefexpectedError], options: components },
310
+ { code: '<Link href={"javascript:void(0)"} />', errors: [invalidHrefexpectedError], options: components },
311
+ { code: '<Anchor href="" />', errors: [invalidHrefexpectedError], options: components },
312
+ { code: '<Anchor href="#" />', errors: [invalidHrefexpectedError], options: components },
313
+ { code: '<Anchor href={"#"} />', errors: [invalidHrefexpectedError], options: components },
314
+ { code: '<Anchor href="javascript:void(0)" />', errors: [invalidHrefexpectedError], options: components },
315
+ { code: '<Anchor href={"javascript:void(0)"} />', errors: [invalidHrefexpectedError], options: components },
316
+ // SHOULD BE BUTTON
317
+ { code: '<Link onClick={() => void 0} />', errors: [preferButtonexpectedError], options: components },
318
+ { code: '<Link href="#" onClick={() => void 0} />', errors: [preferButtonexpectedError], options: components },
319
+ {
320
+ code: '<Link href="javascript:void(0)" onClick={() => void 0} />',
321
+ errors: [preferButtonexpectedError],
322
+ options: components,
323
+ },
324
+ {
325
+ code: '<Link href={"javascript:void(0)"} onClick={() => void 0} />',
326
+ errors: [preferButtonexpectedError],
327
+ options: components,
328
+ },
329
+ { code: '<Anchor onClick={() => void 0} />', errors: [preferButtonexpectedError], options: components },
330
+ { code: '<Anchor href="#" onClick={() => void 0} />', errors: [preferButtonexpectedError], options: components },
331
+ {
332
+ code: '<Anchor href="javascript:void(0)" onClick={() => void 0} />',
333
+ errors: [preferButtonexpectedError],
334
+ options: components,
335
+ },
336
+ {
337
+ code: '<Anchor href={"javascript:void(0)"} onClick={() => void 0} />',
338
+ errors: [preferButtonexpectedError],
339
+ options: components,
340
+ },
341
+ {
342
+ code: '<Link href="#" onClick={() => void 0} />',
343
+ errors: [preferButtonexpectedError],
344
+ settings: componentsSettings,
345
+ },
346
+
347
+ // CUSTOM PROP TESTS
348
+ // NO HREF
349
+ { code: '<a hrefLeft={undefined} />', errors: [noHrefexpectedError], options: specialLink },
350
+ { code: '<a hrefLeft={null} />', errors: [noHrefexpectedError], options: specialLink },
351
+ // INVALID HREF
352
+ { code: '<a hrefLeft="" />;', errors: [invalidHrefexpectedError], options: specialLink },
353
+ { code: '<a hrefLeft="#" />', errors: [invalidHrefexpectedError], options: specialLink },
354
+ { code: '<a hrefLeft={"#"} />', errors: [invalidHrefexpectedError], options: specialLink },
355
+ { code: '<a hrefLeft="javascript:void(0)" />', errors: [invalidHrefexpectedError], options: specialLink },
356
+ { code: '<a hrefLeft={"javascript:void(0)"} />', errors: [invalidHrefexpectedError], options: specialLink },
357
+ // SHOULD BE BUTTON
358
+ { code: '<a hrefLeft="#" onClick={() => void 0} />', errors: [preferButtonexpectedError], options: specialLink },
359
+ {
360
+ code: '<a hrefLeft="javascript:void(0)" onClick={() => void 0} />',
361
+ errors: [preferButtonexpectedError],
362
+ options: specialLink,
363
+ },
364
+ {
365
+ code: '<a hrefLeft={"javascript:void(0)"} onClick={() => void 0} />',
366
+ errors: [preferButtonexpectedError],
367
+ options: specialLink,
368
+ },
369
+
370
+ // CUSTOM BOTH COMPONENTS AND SPECIAL LINK TESTS
371
+ // NO HREF
372
+ { code: '<Anchor Anchor={undefined} />', errors: [noHrefexpectedError], options: componentsAndSpecialLink },
373
+ { code: '<Anchor hrefLeft={null} />', errors: [noHrefexpectedError], options: componentsAndSpecialLink },
374
+ // INVALID HREF
375
+ { code: '<Anchor hrefLeft="" />;', errors: [invalidHrefexpectedError], options: componentsAndSpecialLink },
376
+ { code: '<Anchor hrefLeft="#" />', errors: [invalidHrefexpectedError], options: componentsAndSpecialLink },
377
+ { code: '<Anchor hrefLeft={"#"} />', errors: [invalidHrefexpectedError], options: componentsAndSpecialLink },
378
+ {
379
+ code: '<Anchor hrefLeft="javascript:void(0)" />',
380
+ errors: [invalidHrefexpectedError],
381
+ options: componentsAndSpecialLink,
382
+ },
383
+ {
384
+ code: '<Anchor hrefLeft={"javascript:void(0)"} />',
385
+ errors: [invalidHrefexpectedError],
386
+ options: componentsAndSpecialLink,
387
+ },
388
+ // SHOULD BE BUTTON
389
+ {
390
+ code: '<Anchor hrefLeft="#" onClick={() => void 0} />',
391
+ errors: [preferButtonexpectedError],
392
+ options: componentsAndSpecialLink,
393
+ },
394
+ {
395
+ code: '<Anchor hrefLeft="javascript:void(0)" onClick={() => void 0} />',
396
+ errors: [preferButtonexpectedError],
397
+ options: componentsAndSpecialLink,
398
+ },
399
+ {
400
+ code: '<Anchor hrefLeft={"javascript:void(0)"} onClick={() => void 0} />',
401
+ errors: [preferButtonexpectedError],
402
+ options: componentsAndSpecialLink,
403
+ },
404
+
405
+ // WITH ASPECTS TESTS
406
+ // NO HREF
407
+ { code: '<a />', options: noHrefAspect, errors: [noHrefexpectedError] },
408
+ { code: '<a />', options: noHrefPreferButtonAspect, errors: [noHrefexpectedError] },
409
+ { code: '<a />', options: noHrefInvalidHrefAspect, errors: [noHrefexpectedError] },
410
+ { code: '<a href={undefined} />', options: noHrefAspect, errors: [noHrefexpectedError] },
411
+ { code: '<a href={undefined} />', options: noHrefPreferButtonAspect, errors: [noHrefexpectedError] },
412
+ { code: '<a href={undefined} />', options: noHrefInvalidHrefAspect, errors: [noHrefexpectedError] },
413
+ { code: '<a href={null} />', options: noHrefAspect, errors: [noHrefexpectedError] },
414
+ { code: '<a href={null} />', options: noHrefPreferButtonAspect, errors: [noHrefexpectedError] },
415
+ { code: '<a href={null} />', options: noHrefInvalidHrefAspect, errors: [noHrefexpectedError] },
416
+
417
+ // INVALID HREF
418
+ { code: '<a href="" />;', options: invalidHrefAspect, errors: [invalidHrefexpectedError] },
419
+ { code: '<a href="" />;', options: noHrefInvalidHrefAspect, errors: [invalidHrefexpectedError] },
420
+ { code: '<a href="" />;', options: preferButtonInvalidHrefAspect, errors: [invalidHrefexpectedError] },
421
+ { code: '<a href="#" />;', options: invalidHrefAspect, errors: [invalidHrefexpectedError] },
422
+ { code: '<a href="#" />;', options: noHrefInvalidHrefAspect, errors: [invalidHrefexpectedError] },
423
+ { code: '<a href="#" />;', options: preferButtonInvalidHrefAspect, errors: [invalidHrefexpectedError] },
424
+ { code: '<a href={"#"} />;', options: invalidHrefAspect, errors: [invalidHrefexpectedError] },
425
+ { code: '<a href={"#"} />;', options: noHrefInvalidHrefAspect, errors: [invalidHrefexpectedError] },
426
+ { code: '<a href={"#"} />;', options: preferButtonInvalidHrefAspect, errors: [invalidHrefexpectedError] },
427
+ { code: '<a href="javascript:void(0)" />;', options: invalidHrefAspect, errors: [invalidHrefexpectedError] },
428
+ { code: '<a href="javascript:void(0)" />;', options: noHrefInvalidHrefAspect, errors: [invalidHrefexpectedError] },
429
+ {
430
+ code: '<a href="javascript:void(0)" />;',
431
+ options: preferButtonInvalidHrefAspect,
432
+ errors: [invalidHrefexpectedError],
433
+ },
434
+ { code: '<a href={"javascript:void(0)"} />;', options: invalidHrefAspect, errors: [invalidHrefexpectedError] },
435
+ { code: '<a href={"javascript:void(0)"} />;', options: noHrefInvalidHrefAspect, errors: [invalidHrefexpectedError] },
436
+ {
437
+ code: '<a href={"javascript:void(0)"} />;',
438
+ options: preferButtonInvalidHrefAspect,
439
+ errors: [invalidHrefexpectedError],
440
+ },
441
+
442
+ // SHOULD BE BUTTON
443
+ { code: '<a onClick={() => void 0} />', options: preferButtonAspect, errors: [preferButtonexpectedError] },
444
+ {
445
+ code: '<a onClick={() => void 0} />',
446
+ options: preferButtonInvalidHrefAspect,
447
+ errors: [preferButtonexpectedError],
448
+ },
449
+ { code: '<a onClick={() => void 0} />', options: noHrefPreferButtonAspect, errors: [preferButtonexpectedError] },
450
+ { code: '<a onClick={() => void 0} />', options: noHrefAspect, errors: [noHrefexpectedError] },
451
+ { code: '<a onClick={() => void 0} />', options: noHrefInvalidHrefAspect, errors: [noHrefexpectedError] },
452
+ { code: '<a href="#" onClick={() => void 0} />', options: preferButtonAspect, errors: [preferButtonexpectedError] },
453
+ {
454
+ code: '<a href="#" onClick={() => void 0} />',
455
+ options: noHrefPreferButtonAspect,
456
+ errors: [preferButtonexpectedError],
457
+ },
458
+ {
459
+ code: '<a href="#" onClick={() => void 0} />',
460
+ options: preferButtonInvalidHrefAspect,
461
+ errors: [preferButtonexpectedError],
462
+ },
463
+ { code: '<a href="#" onClick={() => void 0} />', options: invalidHrefAspect, errors: [invalidHrefexpectedError] },
464
+ {
465
+ code: '<a href="#" onClick={() => void 0} />',
466
+ options: noHrefInvalidHrefAspect,
467
+ errors: [invalidHrefexpectedError],
468
+ },
469
+ {
470
+ code: '<a href="javascript:void(0)" onClick={() => void 0} />',
471
+ options: preferButtonAspect,
472
+ errors: [preferButtonexpectedError],
473
+ },
474
+ {
475
+ code: '<a href="javascript:void(0)" onClick={() => void 0} />',
476
+ options: noHrefPreferButtonAspect,
477
+ errors: [preferButtonexpectedError],
478
+ },
479
+ {
480
+ code: '<a href="javascript:void(0)" onClick={() => void 0} />',
481
+ options: preferButtonInvalidHrefAspect,
482
+ errors: [preferButtonexpectedError],
483
+ },
484
+ {
485
+ code: '<a href="javascript:void(0)" onClick={() => void 0} />',
486
+ options: invalidHrefAspect,
487
+ errors: [invalidHrefexpectedError],
488
+ },
489
+ {
490
+ code: '<a href="javascript:void(0)" onClick={() => void 0} />',
491
+ options: noHrefInvalidHrefAspect,
492
+ errors: [invalidHrefexpectedError],
493
+ },
494
+ {
495
+ code: '<a href={"javascript:void(0)"} onClick={() => void 0} />',
496
+ options: preferButtonAspect,
497
+ errors: [preferButtonexpectedError],
498
+ },
499
+ {
500
+ code: '<a href={"javascript:void(0)"} onClick={() => void 0} />',
501
+ options: noHrefPreferButtonAspect,
502
+ errors: [preferButtonexpectedError],
503
+ },
504
+ {
505
+ code: '<a href={"javascript:void(0)"} onClick={() => void 0} />',
506
+ options: preferButtonInvalidHrefAspect,
507
+ errors: [preferButtonexpectedError],
508
+ },
509
+ {
510
+ code: '<a href={"javascript:void(0)"} onClick={() => void 0} />',
511
+ options: invalidHrefAspect,
512
+ errors: [invalidHrefexpectedError],
513
+ },
514
+ {
515
+ code: '<a href={"javascript:void(0)"} onClick={() => void 0} />',
516
+ options: noHrefInvalidHrefAspect,
517
+ errors: [invalidHrefexpectedError],
518
+ },
519
+
520
+ // CUSTOM COMPONENTS AND SPECIAL LINK AND ASPECT
521
+ {
522
+ code: '<Anchor hrefLeft={undefined} />',
523
+ options: componentsAndSpecialLinkAndNoHrefAspect,
524
+ errors: [noHrefexpectedError],
525
+ },
526
+ {
527
+ code: '<Anchor hrefLeft={null} />',
528
+ options: componentsAndSpecialLinkAndNoHrefAspect,
529
+ errors: [noHrefexpectedError],
530
+ },
531
+ )).map(parserOptionsMapper),
532
+ });
@@ -0,0 +1,95 @@
1
+ /**
2
+ * @fileoverview Enforce elements with aria-activedescendant are tabbable.
3
+ * @author Jesse Beach <@jessebeach>
4
+ */
5
+
6
+ // -----------------------------------------------------------------------------
7
+ // Requirements
8
+ // -----------------------------------------------------------------------------
9
+
10
+ import { RuleTester } from 'eslint';
11
+ import parserOptionsMapper from '../../__util__/parserOptionsMapper';
12
+ import parsers from '../../__util__/helpers/parsers';
13
+ import rule from '../../../src/rules/aria-activedescendant-has-tabindex';
14
+
15
+ // -----------------------------------------------------------------------------
16
+ // Tests
17
+ // -----------------------------------------------------------------------------
18
+
19
+ const ruleTester = new RuleTester();
20
+
21
+ const expectedError = {
22
+ message: 'An element that manages focus with `aria-activedescendant` must have a tabindex',
23
+ type: 'JSXOpeningElement',
24
+ };
25
+
26
+ ruleTester.run('aria-activedescendant-has-tabindex', rule, {
27
+ valid: parsers.all([].concat(
28
+ {
29
+ code: '<CustomComponent />;',
30
+ },
31
+ {
32
+ code: '<CustomComponent aria-activedescendant={someID} />;',
33
+ },
34
+ {
35
+ code: '<CustomComponent aria-activedescendant={someID} tabIndex={0} />;',
36
+ },
37
+ {
38
+ code: '<CustomComponent aria-activedescendant={someID} tabIndex={-1} />;',
39
+ },
40
+ {
41
+ code: '<CustomComponent aria-activedescendant={someID} tabIndex={0} />;',
42
+ settings: { 'jsx-a11y': { components: { CustomComponent: 'div' } } },
43
+ },
44
+ {
45
+ code: '<div />;',
46
+ },
47
+ {
48
+ code: '<input />;',
49
+ },
50
+ {
51
+ code: '<div tabIndex={0} />;',
52
+ },
53
+ {
54
+ code: '<div aria-activedescendant={someID} tabIndex={0} />;',
55
+ },
56
+ {
57
+ code: '<div aria-activedescendant={someID} tabIndex="0" />;',
58
+ },
59
+ {
60
+ code: '<div aria-activedescendant={someID} tabIndex={1} />;',
61
+ },
62
+ {
63
+ code: '<input aria-activedescendant={someID} />;',
64
+ },
65
+ {
66
+ code: '<input aria-activedescendant={someID} tabIndex={1} />;',
67
+ },
68
+ {
69
+ code: '<input aria-activedescendant={someID} tabIndex={0} />;',
70
+ },
71
+ {
72
+ code: '<input aria-activedescendant={someID} tabIndex={-1} />;',
73
+ },
74
+ {
75
+ code: '<div aria-activedescendant={someID} tabIndex={-1} />;',
76
+ },
77
+ {
78
+ code: '<div aria-activedescendant={someID} tabIndex="-1" />;',
79
+ },
80
+ {
81
+ code: '<input aria-activedescendant={someID} tabIndex={-1} />;',
82
+ },
83
+ )).map(parserOptionsMapper),
84
+ invalid: parsers.all([].concat(
85
+ {
86
+ code: '<div aria-activedescendant={someID} />;',
87
+ errors: [expectedError],
88
+ },
89
+ {
90
+ code: '<CustomComponent aria-activedescendant={someID} />;',
91
+ errors: [expectedError],
92
+ settings: { 'jsx-a11y': { components: { CustomComponent: 'div' } } },
93
+ },
94
+ )).map(parserOptionsMapper),
95
+ });