diff --git a/collections/forms/i18n/en.pot b/collections/forms/i18n/en.pot index d03caf846..c99afb8eb 100644 --- a/collections/forms/i18n/en.pot +++ b/collections/forms/i18n/en.pot @@ -5,8 +5,8 @@ msgstr "" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" "Plural-Forms: nplurals=2; plural=(n != 1)\n" -"POT-Creation-Date: 2024-10-16T02:11:46.033Z\n" -"PO-Revision-Date: 2024-10-16T02:11:46.034Z\n" +"POT-Creation-Date: 2024-10-22T03:25:41.899Z\n" +"PO-Revision-Date: 2024-10-22T03:25:41.902Z\n" msgid "Upload file" msgstr "Upload file" diff --git a/components/select/src/single-select-a11y/features/keyboard-interactions.test.e2e.js b/components/select/src/single-select-a11y/features/keyboard-interactions.test.e2e.js new file mode 100644 index 000000000..993dd3731 --- /dev/null +++ b/components/select/src/single-select-a11y/features/keyboard-interactions.test.e2e.js @@ -0,0 +1,256 @@ +describe('', () => { + it('should highlight the first option on the currently displayed page', () => { + // Given a select with 100 options is displayed + cy.visitStory('Single Select A11y', 'Hundret Options') + + // And the menu is visible + cy.findByRole('combobox').click() + cy.findByRole('option', { selected: true }).should('be.visible') + + // And the 74th option is being highlighted + cy.findByRole('combobox').focus().type(`Select option 73`) + + // And the 70th option is the first option on the current page + let optionOffset + cy.findAllByRole('option') + .eq(70) + .then((option) => { + const { offsetTop } = option.get(0) + optionOffset = offsetTop + }) + + cy.findByRole('option', { selected: true }) + .invoke('parent') // listbox + .invoke('parent') // scrollable div + .then((listBoxParent) => { + console.log('> listBoxParent', listBoxParent.get(0)) + console.log('> optionOffset', optionOffset) + listBoxParent.get(0).scrollTop = optionOffset + }) + + cy.findAllByRole('option').eq(70).should('be.visible') + + // When the PageUp key is pressed + cy.findByRole('combobox').trigger('keydown', { + key: 'PageUp', + force: true, + }) + + // Then the first option on the currently displayed page is highlighted + cy.findByRole('option', { selected: true }) + .invoke('attr', 'aria-label') + .should('equal', 'Select option 70') + }) + + it('should highlight the first option on the previous page', () => { + // Given a select with 100 options is displayed + cy.visitStory('Single Select A11y', 'Hundret Options') + + // And the menu is visible + cy.findByRole('combobox').click() + cy.findByRole('option', { selected: true }).should('be.visible') + + // And the 70th option is being highlighted + // Will automatically scroll there and make it the first option on the page + cy.findByRole('combobox').focus().type(`Select option 70`) + + // When the PageUp key is pressed + cy.findByRole('combobox').trigger('keydown', { + key: 'PageUp', + force: true, + }) + + // Then the first option on the previous page is highlighted + cy.findByRole('option', { selected: true }) + .invoke('attr', 'aria-label') + .should('equal', 'Select option 61') + + // And the previously highlighted option is not visible + cy.findAllByRole('option').eq(70).should('not.be.visible') + }) + + it('should highlight the first option', () => { + // Given a select with 100 options is displayed + cy.visitStory('Single Select A11y', 'Hundret Options') + + // And the menu is visible + cy.findByRole('combobox').click() + cy.findByRole('option', { selected: true }).should('be.visible') + + // And the 2nd option is being highlighted + cy.findByRole('combobox').focus().type(`Select option 1`) + + // And the 2nd option is the first option on the current page + let optionOffset + cy.findAllByRole('option') + .eq(1) + .then((option) => { + const { offsetTop } = option.get(0) + optionOffset = offsetTop + }) + + cy.findByRole('option', { selected: true }) + .invoke('parent') // listbox + .invoke('parent') // scrollable div + .then((listBoxParent) => { + console.log('> listBoxParent', listBoxParent.get(0)) + console.log('> optionOffset', optionOffset) + listBoxParent.get(0).scrollTop = optionOffset + }) + + // When the PageUp key is pressed + cy.findByRole('combobox').trigger('keydown', { + key: 'PageUp', + force: true, + }) + + // Then the first option is being highlighted + cy.all( + () => cy.findAllByRole('option').first().invoke('get', 0), + () => cy.findByRole('option', { selected: true }).invoke('get', 0) + ).then(([firstOption, highlightedOption]) => { + expect(highlightedOption).to.equal(firstOption) + }) + }) + + it('should highlight the last option on the currently displayed page', () => { + // Given a select with 100 options is displayed + cy.visitStory('Single Select A11y', 'Hundret Options') + + // And the menu is visible + // first option will be highlighted automatically + cy.findByRole('combobox').click() + cy.findByRole('option', { selected: true }).should('be.visible') + + // When the PageDown key is pressed + cy.findByRole('combobox').trigger('keydown', { + key: 'PageDown', + force: true, + }) + + // Then the last option on the currently displayed page is highlighted + cy.all( + () => cy.get('[role="option"]:visible').last().invoke('get', 0), + () => cy.findByRole('option', { selected: true }).invoke('get', 0) + ).then(([lastVisibleOption, highlightedOption]) => { + expect(highlightedOption).to.equal(lastVisibleOption) + }) + }) + + it( + 'should highlight the last option on the next page', + // We don't want the options to scroll when we check whether they're + // visible or not (as that'd make them visible) + { scrollBehavior: false }, + () => { + // Given a select with 100 options is displayed + cy.visitStory('Single Select A11y', 'Hundret Options') + + // And the menu is visible + cy.findByRole('combobox').click() + cy.findByRole('option', { selected: true }).should('be.visible') + + // And the option last visible option is being highlighted + cy.get('[role="option"]:visible').then(($visibleOptions) => { + const visibleOptionsAmount = $visibleOptions.length + + for (let i = 0; i < visibleOptionsAmount - 1; ++i) { + cy.findByRole('combobox').trigger('keydown', { + key: 'ArrowDown', + force: true, + }) + + if (i === visibleOptionsAmount - 2) { + cy.wrap(i).as('lastVisibleOptionIndex') + } + } + }) + + cy.get('@lastVisibleOptionIndex').then((lastVisibleOptionIndex) => { + cy.get('[role="option"]') + .eq(lastVisibleOptionIndex + 1) // 1-based + .invoke('attr', 'aria-selected') + .should('equal', 'true') + }) + + // When the PageDown key is pressed + cy.findByRole('combobox').trigger('keydown', { + key: 'PageDown', + force: true, + }) + + // Then the next page is shown + cy.get('@lastVisibleOptionIndex').then((lastVisibleOptionIndex) => { + cy.get('[role="option"]') + .eq(lastVisibleOptionIndex + 1) + .should('not.be.visible') + cy.get('[role="option"]') + .eq(lastVisibleOptionIndex + 2) + .should('be.visible') + }) + + // And the last option on the next page is highlighted + cy.get('[role="option"]:visible') + .last() + .invoke('attr', 'aria-selected') + .should('equal', 'true') + + // And the previously highlighted option is not visible + cy.get('@lastVisibleOptionIndex').then((lastVisibleOptionIndex) => { + cy.get('[role="option"]') + .eq(lastVisibleOptionIndex + 1) + .invoke('attr', 'aria-selected') + .should('equal', 'false') + }) + } + ) + + it( + 'should highlight the last option', + // We don't want the options to scroll when we check whether they're + // visible or not (as that'd make them visible) + { scrollBehavior: false }, + () => { + // Given a select with 100 options is displayed + cy.visitStory('Single Select A11y', 'Hundret Options') + + // And the menu is visible + cy.findByRole('combobox').click() + cy.findByRole('option', { selected: true }).should('be.visible') + + // And the 2nd-last option is being highlighted and visible + for ( + let i = 0; + i < 11; // This will bring us to the second last option exactly + ++i + ) { + cy.findByRole('combobox').trigger('keydown', { + key: 'PageDown', + force: true, + }) + } + cy.findAllByRole('option').eq(98).should('be.visible') + + // And the last option is not visible + cy.findAllByRole('option').last().should('not.be.visible') + + // When the PageDown key is pressed + cy.findByRole('combobox').trigger('keydown', { + key: 'PageDown', + force: true, + }) + + // Then the last option is highlighted + cy.all( + () => cy.findAllByRole('option').last().invoke('get', 0), + () => + cy.findByRole('option', { selected: true }).invoke('get', 0) + ).then(([lastOption, highlightedOption]) => { + expect(highlightedOption).to.equal(lastOption) + }) + + // And the last option is visible + cy.findAllByRole('option').last().should('be.visible') + } + ) +}) diff --git a/components/select/src/single-select-a11y/menu/option.js b/components/select/src/single-select-a11y/menu/option.js index 4ad2a5ef8..bcc3837d6 100644 --- a/components/select/src/single-select-a11y/menu/option.js +++ b/components/select/src/single-select-a11y/menu/option.js @@ -74,7 +74,7 @@ export function Option({ data-test={dataTest} disabled={disabled} role="option" - aria-selected={highlighted || ''} + aria-selected={highlighted || 'false'} aria-disabled={disabled} aria-label={label} onClick={() => { diff --git a/components/select/src/single-select-a11y/single-select-a11y.e2e.stories.js b/components/select/src/single-select-a11y/single-select-a11y.e2e.stories.js index df78e235f..2785ae080 100644 --- a/components/select/src/single-select-a11y/single-select-a11y.e2e.stories.js +++ b/components/select/src/single-select-a11y/single-select-a11y.e2e.stories.js @@ -1,4 +1,4 @@ -import React from 'react' +import React, { useState } from 'react' import { SingleSelectA11y } from './single-select-a11y.js' export default { @@ -102,7 +102,7 @@ const fiveOptions = options.slice(0, 5) export const DefaultPosition = () => ( null} options={fiveOptions} /> @@ -112,7 +112,7 @@ export const FlippedPosition = () => ( <> null} options={options} /> @@ -136,7 +136,7 @@ export const ShiftedIntoView = () => ( <> null} options={options} /> @@ -155,3 +155,21 @@ export const ShiftedIntoView = () => ( `} ) + +const hundretOptions = Array.apply(null, Array(100)).map((x, i) => ({ + value: `${i}`, + label: `Select option ${i}`, +})) + +export const HundretOptions = () => { + const [value, setValue] = useState('0') + + return ( + + ) +} diff --git a/components/select/src/single-select-a11y/single-select-a11y.prod.stories.js b/components/select/src/single-select-a11y/single-select-a11y.prod.stories.js index 53fe672cd..a1c11677f 100644 --- a/components/select/src/single-select-a11y/single-select-a11y.prod.stories.js +++ b/components/select/src/single-select-a11y/single-select-a11y.prod.stories.js @@ -394,6 +394,7 @@ export const WithOptionsAndLoading = () => { } onChange={(nextValue) => setValue(nextValue)} options={[ + { label: 'None', value: '' }, { value: '1', label: 'Option 1' }, { value: '2', label: 'Option 2' }, { value: '3', label: 'Option 3' }, @@ -418,6 +419,7 @@ export const WithOptionsAndLoadingText = () => { } onChange={(nextValue) => setValue(nextValue)} options={[ + { label: 'None', value: '' }, { value: '1', label: 'Option 1' }, { value: '2', label: 'Option 2' }, { value: '3', label: 'Option 3' }, @@ -427,35 +429,20 @@ export const WithOptionsAndLoadingText = () => { } export const WithManyOptions = () => { - // const [value, setValue] = useState('art_entry_point:_no_pmtct') - const [value, setValue] = useState('10') - const selectOptions = Array.apply(null, Array(100)).map((x, i) => i) + const [value, setValue] = useState('art_entry_point:_no_pmtct') return ( - <> - option.value === value).label - // : '' - // } - onChange={(nextValue) => setValue(nextValue)} - options={selectOptions.map((i) => ({ - value: i.toString(), - label: `Select option ${i + 1}`, - }))} - /> - - - + option.value === value).label + : '' + } + onChange={(nextValue) => setValue(nextValue)} + options={options} + /> ) } diff --git a/components/select/src/single-select-a11y/single-select-a11y.test.js b/components/select/src/single-select-a11y/single-select-a11y.test.js index 2bb282378..15b12209d 100644 --- a/components/select/src/single-select-a11y/single-select-a11y.test.js +++ b/components/select/src/single-select-a11y/single-select-a11y.test.js @@ -425,14 +425,13 @@ describe('', () => { expect(combobox).toContainElement(withTextBar) }) - /************************** + /*************************** * * * ===================== * * Keyboard interactions * * ===================== * * * **************************/ - describe.each([ { key: ' ' }, { key: 'Enter' }, @@ -579,9 +578,163 @@ describe('', () => { expect(onChange).toHaveBeenCalledWith('foo') }) - // @TODO - it.skip('should move up an entire page', () => {}) - it.skip('should move up half a page to the first option', () => {}) - it.skip('should move down an entire page', () => {}) - it.skip('should move down half a page to the last option', () => {}) + it('should highlight the next option', () => { + const onChange = jest.fn() + + render( + + ) + + // open the menu + expect(screen.queryByRole('listbox')).toBeNull() + const comboBox = screen.getByRole('combobox') + fireEvent.click(comboBox) + expect(screen.queryByRole('listbox')).not.toBeNull() + + // the first option should be highlighted + const highlightedOptionBefore = screen.getByRole('option', { + selected: true, + }) + expect( + highlightedOptionBefore.attributes.getNamedItem('aria-label').value + ).toBe('None') + + // The second option should be highlighted + fireEvent.keyDown(comboBox, { key: 'ArrowDown' }) + const highlightedOptionAfter = screen.getByRole('option', { + selected: true, + }) + expect( + highlightedOptionAfter.attributes.getNamedItem('aria-label').value + ).toBe('Foo') + }) + + it('should highlight the previous option', () => { + const onChange = jest.fn() + + render( + + ) + + // open the menu + expect(screen.queryByRole('listbox')).toBeNull() + const comboBox = screen.getByRole('combobox') + fireEvent.click(comboBox) + expect(screen.queryByRole('listbox')).not.toBeNull() + + // the last option should be highlighted + const highlightedOptionBefore = screen.getByRole('option', { + selected: true, + }) + expect( + highlightedOptionBefore.attributes.getNamedItem('aria-label').value + ).toBe('Bar') + + // The second option should be highlighted + fireEvent.keyDown(comboBox, { key: 'ArrowUp' }) + const highlightedOptionAfter = screen.getByRole('option', { + selected: true, + }) + expect( + highlightedOptionAfter.attributes.getNamedItem('aria-label').value + ).toBe('Foo') + }) + + it('should highlight the first option', () => { + const onChange = jest.fn() + + render( + + ) + + // open the menu + expect(screen.queryByRole('listbox')).toBeNull() + const comboBox = screen.getByRole('combobox') + fireEvent.click(comboBox) + expect(screen.queryByRole('listbox')).not.toBeNull() + + // the last option should be highlighted + const highlightedOptionBefore = screen.getByRole('option', { + selected: true, + }) + expect( + highlightedOptionBefore.attributes.getNamedItem('aria-label').value + ).toBe('Bar') + + // The first option should be highlighted + fireEvent.keyDown(comboBox, { key: 'Home' }) + const highlightedOptionAfter = screen.getByRole('option', { + selected: true, + }) + expect( + highlightedOptionAfter.attributes.getNamedItem('aria-label').value + ).toBe('None') + }) + + it('should highlight the last option', () => { + const onChange = jest.fn() + + render( + + ) + + // open the menu + expect(screen.queryByRole('listbox')).toBeNull() + const comboBox = screen.getByRole('combobox') + fireEvent.click(comboBox) + expect(screen.queryByRole('listbox')).not.toBeNull() + + // the first option should be highlighted + const highlightedOptionBefore = screen.getByRole('option', { + selected: true, + }) + expect( + highlightedOptionBefore.attributes.getNamedItem('aria-label').value + ).toBe('None') + + // The last option should be highlighted + fireEvent.keyDown(comboBox, { key: 'End' }) + const highlightedOptionAfter = screen.getByRole('option', { + selected: true, + }) + expect( + highlightedOptionAfter.attributes.getNamedItem('aria-label').value + ).toBe('Bar') + }) }) diff --git a/cypress.config.js b/cypress.config.js index 17f05f725..52bb6bf9f 100644 --- a/cypress.config.js +++ b/cypress.config.js @@ -15,7 +15,7 @@ module.exports = defineConfig({ e2e: { setupNodeEvents, baseUrl: 'http://localhost:5000', - specPattern: '**/src/**/*.feature', + specPattern: '**/src/**/*{.feature,.test.e2e.js}', experimentalRunAllSpecs: true, }, })