Skip to content

Commit

Permalink
feat: add option for enforcing label's htmlFor matches control's id
Browse files Browse the repository at this point in the history
This change adds an option to the `label-has-associated-control` rule, enforcing that the label's htmlFor attribute matches the associated control's id attribute.  Previously, the only validation done on htmlFor was that it was on the label component and had text.  There was no attempt to cross-check that value against any attribute on the associated control.  Not, when the option is enabled, cases where they don't match will report.

I also took the opportunity to update the error messages so that each assert type gets an error message with verbiage specific to the assertion. (not sure if this should be called out as a separate feature in the changelog?).

Note: the current implementation only checks the first instance it finds of child component that matches each control component type.  It assumes that there won't be any acceptable cases where a label would have multiple inputs nested beneath it.  Let me know if that assumption doesn't hold.

Closes:
  • Loading branch information
michaelfaith committed Dec 24, 2024
1 parent bfb0e9e commit 4a4d2ff
Show file tree
Hide file tree
Showing 6 changed files with 381 additions and 62 deletions.
33 changes: 33 additions & 0 deletions __tests__/src/rules/label-has-associated-control-test.js
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@ const errorMessages = {
nesting: 'A form label must have an associated control as a descendant.',
either: 'A form label must either have a valid htmlFor attribute or a control as a descendant.',
both: 'A form label must have a valid htmlFor attribute and a control as a descendant.',
htmlForShouldMatchId: 'A form label must have a htmlFor attribute that matches the id of the associated control.',
};
const expectedErrors = {};
Object.keys(errorMessages).forEach((key) => {
Expand Down Expand Up @@ -58,6 +59,7 @@ const htmlForValid = [
{ code: '<label htmlFor="js_id" aria-label="A label" />' },
{ code: '<label htmlFor="js_id" aria-labelledby="A label" />' },
{ code: '<div><label htmlFor="js_id">A label</label><input id="js_id" /></div>' },
{ code: '<div><label htmlFor={inputId}>A label</label><input id={inputId} /></div>' },
{ code: '<label for="js_id"><span><span><span>A label</span></span></span></label>', options: [{ depth: 4 }], settings: attributesSettings },
{ code: '<label for="js_id" aria-label="A label" />', settings: attributesSettings },
{ code: '<label for="js_id" aria-labelledby="A label" />', settings: attributesSettings },
Expand Down Expand Up @@ -128,6 +130,12 @@ const alwaysValid = [
{ code: '<input type="hidden" />' },
];

const shouldHtmlForMatchIdValid = [
{ code: '<label htmlFor="js_id" aria-label="A label"><span><span><input id="js_id" /></span></span></label>', options: [{ depth: 4, shouldHtmlForMatchId: true }] },
{ code: '<div><label htmlFor="js_id">A label</label><input id="js_id" /></div>', options: [{ shouldHtmlForMatchId: true }] },
{ code: '<div><label htmlFor={inputId} aria-label="A label" /><input id={inputId} /></div>', options: [{ shouldHtmlForMatchId: true }] },
];

const htmlForInvalid = (assertType) => {
const expectedError = expectedErrors[assertType];
return [
Expand Down Expand Up @@ -164,6 +172,13 @@ const nestingInvalid = (assertType) => {
{ code: '<CustomLabel><span>A label<CustomInput /></span></CustomLabel>', settings: componentsSettings, errors: [expectedError] },
];
};
const shouldHtmlForMatchIdInvalid = [
{ code: '<label htmlFor="js_id" aria-label="A label"><span><span><input /></span></span></label>', options: [{ depth: 4, shouldHtmlForMatchId: true }], errors: [expectedErrors.htmlForShouldMatchId] },
{ code: '<label htmlFor="js_id" aria-label="A label"><span><span><input name="js_id" /></span></span></label>', options: [{ depth: 4, shouldHtmlForMatchId: true }], errors: [expectedErrors.htmlForShouldMatchId] },
{ code: '<div><label htmlFor="js_id">A label</label><input /></div>', options: [{ shouldHtmlForMatchId: true }], errors: [expectedErrors.htmlForShouldMatchId] },
{ code: '<div><label htmlFor="js_id">A label</label><input name="js_id" /></div>', options: [{ shouldHtmlForMatchId: true }], errors: [expectedErrors.htmlForShouldMatchId] },
{ code: '<div><label htmlFor={inputId} aria-label="A label" /><input name={inputId} /></div>', options: [{ shouldHtmlForMatchId: true }], errors: [expectedErrors.htmlForShouldMatchId] },
];

const neverValid = (assertType) => {
const expectedError = expectedErrors[assertType];
Expand Down Expand Up @@ -266,3 +281,21 @@ ruleTester.run(ruleName, rule, {
assert: 'both',
})).map(parserOptionsMapper),
});

// shouldHtmlForMatchId
ruleTester.run(ruleName, rule, {
valid: parsers.all([].concat(
...shouldHtmlForMatchIdValid,
))
.map(ruleOptionsMapperFactory({
assert: 'htmlFor',
}))
.map(parserOptionsMapper),
invalid: parsers.all([].concat(
...shouldHtmlForMatchIdInvalid,
))
.map(ruleOptionsMapperFactory({
assert: 'htmlFor',
}))
.map(parserOptionsMapper),
});
190 changes: 190 additions & 0 deletions __tests__/src/util/getChildComponent-test.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,190 @@
import expect from 'expect';
import getChildComponent from '../../../src/util/getChildComponent';
import JSXAttributeMock from '../../../__mocks__/JSXAttributeMock';
import JSXElementMock from '../../../__mocks__/JSXElementMock';
import JSXExpressionContainerMock from '../../../__mocks__/JSXExpressionContainerMock';

describe('getChildComponent', () => {
describe('no FancyComponent', () => {
it('should return undefined', () => {
expect(getChildComponent(
JSXElementMock('div', [], [
JSXElementMock('div', [], [
JSXElementMock('span', [], []),
JSXElementMock('span', [], [
JSXElementMock('span', [], []),
JSXElementMock('span', [], [
JSXElementMock('span', [], []),
]),
]),
]),
JSXElementMock('span', [], []),
JSXElementMock('img', [
JSXAttributeMock('src', 'some/path'),
]),
]),
'FancyComponent',
5,
)).toBeUndefined();
});
});
describe('contains an indicated component', () => {
it('should return input', () => {
const MockInput = JSXElementMock('input');

expect(getChildComponent(
JSXElementMock('div', [], [
MockInput,
]),
'input',
)).toEqual(MockInput);
});
it('should return FancyComponent', () => {
const MockFancyComponent = JSXElementMock('FancyComponent');

expect(getChildComponent(
JSXElementMock('div', [], [
MockFancyComponent,
]),
'FancyComponent',
)).toEqual(MockFancyComponent);
});
it('FancyComponent is outside of default depth, should return undefined', () => {
expect(getChildComponent(
JSXElementMock('div', [], [
JSXElementMock('div', [], [
JSXElementMock('FancyComponent'),
]),
]),
'FancyComponent',
)).toBeUndefined();
});
it('FancyComponent is inside of custom depth, should return component', () => {
const MockFancyComponent = JSXElementMock('FancyComponent');
expect(getChildComponent(
JSXElementMock('div', [], [
JSXElementMock('div', [], [
MockFancyComponent,
]),
]),
'FancyComponent',
2,
)).toEqual(MockFancyComponent);
});
it('deep nesting, should return component', () => {
const MockFancyComponent = JSXElementMock('FancyComponent');
expect(getChildComponent(
JSXElementMock('div', [], [
JSXElementMock('div', [], [
JSXElementMock('span', [], []),
JSXElementMock('span', [], [
JSXElementMock('span', [], []),
JSXElementMock('span', [], [
JSXElementMock('span', [], [
JSXElementMock('span', [], [
MockFancyComponent,
]),
]),
]),
]),
]),
JSXElementMock('span', [], []),
JSXElementMock('img', [
JSXAttributeMock('src', 'some/path'),
]),
]),
'FancyComponent',
6,
)).toEqual(MockFancyComponent);
});
});
describe('Indeterminate situations', () => {
describe('expression container children', () => {
it('should return component', () => {
const MysteryComponent = JSXExpressionContainerMock('mysteryBox');
expect(getChildComponent(
JSXElementMock('div', [], [
MysteryComponent,
]),
'FancyComponent',
)).toEqual(MysteryComponent);
});
});
});

describe('Glob name matching', () => {
describe('component name contains question mark ? - match any single character', () => {
it('should return component', () => {
const MockFancyComponent = JSXElementMock('FancyComponent');
expect(getChildComponent(
JSXElementMock('div', [], [
MockFancyComponent,
]),
'Fanc?Co??onent',
)).toEqual(MockFancyComponent);
});
it('should return undefined', () => {
expect(getChildComponent(
JSXElementMock('div', [], [
JSXElementMock('FancyComponent'),
]),
'FancyComponent?',
)).toBeUndefined();
});
});

describe('component name contains asterisk * - match zero or more characters', () => {
it('should return component (suffix wildcard)', () => {
const MockFancyComponent = JSXElementMock('FancyComponent');
expect(getChildComponent(
JSXElementMock('div', [], [
MockFancyComponent,
]),
'Fancy*',
)).toEqual(MockFancyComponent);
});
it('should return component (prefix wildcard)', () => {
const MockFancyComponent = JSXElementMock('FancyComponent');
expect(getChildComponent(
JSXElementMock('div', [], [
MockFancyComponent,
]),
'*Component',
)).toEqual(MockFancyComponent);
});
it('should return component (mixed wildcard)', () => {
const MockFancyComponent = JSXElementMock('FancyComponent');
expect(getChildComponent(
JSXElementMock('div', [], [
MockFancyComponent,
]),
'Fancy*C*t',
)).toEqual(MockFancyComponent);
});
});
});

describe('using a custom elementType function', () => {
it('should return the component when the custom elementType returns the proper name', () => {
const MockCustomInput = JSXElementMock('CustomInput');
expect(getChildComponent(
JSXElementMock('div', [], [
MockCustomInput,
]),
'input',
2,
() => 'input',
)).toEqual(MockCustomInput);
});
it('should return undefined when the custom elementType returns a wrong name', () => {
expect(getChildComponent(
JSXElementMock('div', [], [
JSXElementMock('CustomInput'),
]),
'input',
2,
() => 'button',
)).toBeUndefined();
});
});
});
42 changes: 25 additions & 17 deletions docs/rules/label-has-associated-control.md
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,7 @@ The simplest way to achieve an association between a label and an input is to wr

All modern browsers and assistive technology will associate the label with the control.

### Case: The label is a sibling of the control.
### Case: The label is a sibling of the control

In this case, use `htmlFor` and an ID to associate the controls.

Expand All @@ -37,7 +37,7 @@ In this case, use `htmlFor` and an ID to associate the controls.
<input type="text" id={domId} />
```

### Case: My label and input components are custom components.
### Case: My label and input components are custom components

You can configure the rule to be aware of your custom components.

Expand All @@ -49,14 +49,14 @@ You can configure the rule to be aware of your custom components.

And the configuration:

```json
```js
{
"rules": {
"jsx-a11y/label-has-associated-control": [ 2, {
"labelComponents": ["CustomInputLabel"],
"labelAttributes": ["label"],
"controlComponents": ["CustomInput"],
"depth": 3,
rules: {
'jsx-a11y/label-has-associated-control': [ 2, {
labelComponents: ['CustomInputLabel'],
labelAttributes: ['label'],
controlComponents: ['CustomInput'],
depth: 3,
}],
}
}
Expand Down Expand Up @@ -89,15 +89,16 @@ To restate, **every ID needs to be deterministic**, on the server and the client

This rule takes one optional object argument of type object:

```json
```js
{
"rules": {
"jsx-a11y/label-has-associated-control": [ 2, {
"labelComponents": ["CustomLabel"],
"labelAttributes": ["inputLabel"],
"controlComponents": ["CustomInput"],
"assert": "both",
"depth": 3,
rules: {
'jsx-a11y/label-has-associated-control': [ 2, {
labelComponents: ['CustomLabel'],
labelAttributes: ['inputLabel'],
controlComponents: ['CustomInput'],
assert: 'both',
depth: 3,
shouldHtmlForMatchId: true,
}],
}
}
Expand All @@ -113,14 +114,18 @@ This rule takes one optional object argument of type object:

`depth` (default 2, max 25) is an integer that determines how deep within a `JSXElement` label the rule should look for text content or an element with a label to determine if the `label` element will have an accessible label.

`shouldHtmlForMatchId` (default `false`) is a boolean dictating whether the `htmlFor` attribute on a label element should be validated as matching the `id` attributed on an associated control (nested or sibling as described in the cases above).

### Fail

```jsx
function Foo(props) {
return <label {...props} />
}
```

### Succeed

```jsx
function Foo(props) {
const {
Expand All @@ -133,12 +138,14 @@ function Foo(props) {
```

### Fail

```jsx
<input type="text" />
<label>Surname</label>
```

### Succeed

```jsx
<label>
<input type="text" />
Expand All @@ -147,6 +154,7 @@ function Foo(props) {
```

## Accessibility guidelines

- [WCAG 1.3.1](https://www.w3.org/WAI/WCAG21/Understanding/info-and-relationships)
- [WCAG 3.3.2](https://www.w3.org/WAI/WCAG21/Understanding/labels-or-instructions)
- [WCAG 4.1.2](https://www.w3.org/WAI/WCAG21/Understanding/name-role-value)
Loading

0 comments on commit 4a4d2ff

Please sign in to comment.