Skip to content

Commit

Permalink
[Select][base] Add attributes to conform with ARIA 1.2 (mui#35182)
Browse files Browse the repository at this point in the history
  • Loading branch information
michaldudak authored Nov 18, 2022
1 parent 08db62a commit d59f395
Show file tree
Hide file tree
Showing 6 changed files with 274 additions and 94 deletions.
138 changes: 113 additions & 25 deletions packages/mui-base/src/MultiSelectUnstyled/MultiSelectUnstyled.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import {
userEvent,
act,
fireEvent,
screen,
} from 'test/utils';

describe('MultiSelectUnstyled', () => {
Expand Down Expand Up @@ -53,13 +54,13 @@ describe('MultiSelectUnstyled', () => {
it(`opens the dropdown when the "${key}" key is down on the button`, () => {
// can't use the default native `button` as it doesn't treat enter or space press as a click
const { getByRole } = render(<MultiSelectUnstyled slots={{ root: 'div' }} />);
const button = getByRole('button');
const select = getByRole('combobox');
act(() => {
button.focus();
select.focus();
});
fireEvent.keyDown(button, { key });
fireEvent.keyDown(select, { key });

expect(button).to.have.attribute('aria-expanded', 'true');
expect(select).to.have.attribute('aria-expanded', 'true');
expect(getByRole('listbox')).not.to.equal(null);
expect(document.activeElement).to.equal(getByRole('listbox'));
});
Expand All @@ -68,13 +69,13 @@ describe('MultiSelectUnstyled', () => {
it(`opens the dropdown when the " " key is let go on the button`, () => {
// can't use the default native `button` as it doesn't treat enter or space press as a click
const { getByRole } = render(<MultiSelectUnstyled slots={{ root: 'div' }} />);
const button = getByRole('button');
const select = getByRole('combobox');
act(() => {
button.focus();
select.focus();
});
fireEvent.keyUp(button, { key: ' ' });
fireEvent.keyUp(select, { key: ' ' });

expect(button).to.have.attribute('aria-expanded', 'true');
expect(select).to.have.attribute('aria-expanded', 'true');
expect(getByRole('listbox')).not.to.equal(null);
expect(document.activeElement).to.equal(getByRole('listbox'));
});
Expand All @@ -89,9 +90,9 @@ describe('MultiSelectUnstyled', () => {
</MultiSelectUnstyled>,
);

const button = getByRole('button');
const select = getByRole('combobox');
act(() => {
button.click();
select.click();
});

const listbox = getByRole('listbox');
Expand All @@ -100,7 +101,7 @@ describe('MultiSelectUnstyled', () => {
userEvent.keyPress(listbox, { key: 'ArrowDown' }); // highlights '2'
userEvent.keyPress(listbox, { key });

expect(button).to.have.text('2');
expect(select).to.have.text('2');
}),
);
});
Expand All @@ -113,18 +114,18 @@ describe('MultiSelectUnstyled', () => {
</MultiSelectUnstyled>,
);

const button = getByRole('button');
const select = getByRole('combobox');

act(() => {
button.click();
select.click();
});

const listbox = getByRole('listbox');
userEvent.keyPress(listbox, { key: 'ArrowDown' }); // highlights '2'
userEvent.keyPress(listbox, { key: 'Escape' });

expect(button).to.have.attribute('aria-expanded', 'false');
expect(button).to.have.text('1');
expect(select).to.have.attribute('aria-expanded', 'false');
expect(select).to.have.text('1');
expect(queryByRole('listbox')).to.equal(null);
});
});
Expand Down Expand Up @@ -243,9 +244,9 @@ describe('MultiSelectUnstyled', () => {
</MultiSelectUnstyled>,
);

const button = getByRole('button');
const select = getByRole('combobox');
act(() => {
button.click();
select.click();
});

const optionTwo = getByText('Two');
Expand Down Expand Up @@ -299,7 +300,7 @@ describe('MultiSelectUnstyled', () => {
</MultiSelectUnstyled>,
);

expect(getByRole('button')).to.have.text('One (1), Two (2)');
expect(getByRole('combobox')).to.have.text('One (1), Two (2)');
});

it('renders the selected values as comma-separated list of labels if renderValue is not provided', () => {
Expand All @@ -310,7 +311,94 @@ describe('MultiSelectUnstyled', () => {
</MultiSelectUnstyled>,
);

expect(getByRole('button')).to.have.text('One, Two');
expect(getByRole('combobox')).to.have.text('One, Two');
});
});

// according to WAI-ARIA 1.2 (https://www.w3.org/TR/wai-aria-1.2/#combobox)
describe('a11y attributes', () => {
it('should have the `combobox` role', () => {
render(
<MultiSelectUnstyled>
<OptionUnstyled value={1}>One</OptionUnstyled>
</MultiSelectUnstyled>,
);

expect(screen.queryByRole('combobox')).not.to.equal(null);
});

it('should have the aria-haspopup listbox', () => {
render(
<MultiSelectUnstyled>
<OptionUnstyled value={1}>One</OptionUnstyled>
</MultiSelectUnstyled>,
);

expect(screen.getByRole('combobox')).to.have.attribute('aria-haspopup', 'listbox');
});

it('should have the aria-expanded attribute', () => {
render(
<MultiSelectUnstyled>
<OptionUnstyled value={1}>One</OptionUnstyled>
</MultiSelectUnstyled>,
);

expect(screen.getByRole('combobox')).to.have.attribute('aria-expanded', 'false');
});

it('should have the aria-expanded attribute set to true when the listbox is open', () => {
render(
<MultiSelectUnstyled>
<OptionUnstyled value={1}>One</OptionUnstyled>
</MultiSelectUnstyled>,
);

const select = screen.getByRole('combobox');
act(() => {
select.click();
});

expect(select).to.have.attribute('aria-expanded', 'true');
});

it('should have the aria-controls attribute', () => {
render(
<MultiSelectUnstyled>
<OptionUnstyled value={1}>One</OptionUnstyled>
</MultiSelectUnstyled>,
);

const select = screen.getByRole('combobox');

act(() => {
select.click();
});

const listbox = screen.getByRole('listbox');
const listboxId = listbox.getAttribute('id');
expect(listboxId).not.to.equal(null);

expect(select).to.have.attribute('aria-controls', listboxId!);
});

it('should have the aria-activedescendant attribute', () => {
render(
<MultiSelectUnstyled>
<OptionUnstyled value={1}>One</OptionUnstyled>
</MultiSelectUnstyled>,
);

const select = screen.getByRole('combobox');
act(() => {
select.click();
});

const listbox = screen.getByRole('listbox');
fireEvent.keyDown(listbox, { key: 'ArrowDown' });

const options = screen.getAllByRole('option');
expect(listbox).to.have.attribute('aria-activedescendant', options[0].getAttribute('id')!);
});
});

Expand Down Expand Up @@ -365,10 +453,10 @@ describe('MultiSelectUnstyled', () => {
</div>,
);

const button = getByRole('button');
const select = getByRole('combobox');

act(() => {
button.click();
select.click();
});

const listbox = getByRole('listbox');
Expand All @@ -379,8 +467,8 @@ describe('MultiSelectUnstyled', () => {
focusTarget.focus();
});

expect(button).to.have.attribute('aria-expanded', 'false');
expect(button).to.have.text('1');
expect(select).to.have.attribute('aria-expanded', 'false');
expect(select).to.have.text('1');
});

it('focuses the listbox after it is opened', () => {
Expand All @@ -390,9 +478,9 @@ describe('MultiSelectUnstyled', () => {
</MultiSelectUnstyled>,
);

const button = getByRole('button');
const select = getByRole('combobox');
act(() => {
button.click();
select.click();
});

expect(document.activeElement).to.equal(getByRole('listbox'));
Expand Down
Loading

0 comments on commit d59f395

Please sign in to comment.