From b750fa3e20a9224deb3e1d5e270a25f5853fee18 Mon Sep 17 00:00:00 2001 From: Tyler Jones Date: Tue, 19 Dec 2023 16:40:38 -0500 Subject: [PATCH 1/5] Add focus management into its own function --- src/focus-zone.ts | 77 +++++++++++++++++++++++++++-------------------- 1 file changed, 44 insertions(+), 33 deletions(-) diff --git a/src/focus-zone.ts b/src/focus-zone.ts index 157c42d..f91e7ea 100644 --- a/src/focus-zone.ts +++ b/src/focus-zone.ts @@ -1,6 +1,6 @@ import {polyfill as eventListenerSignalPolyfill} from './polyfills/event-listener-signal.js' import {isMacOS} from './utils/user-agent.js' -import {IterateFocusableElements, iterateFocusableElements} from './utils/iterate-focusable-elements.js' +import {IterateFocusableElements, isFocusable, iterateFocusableElements} from './utils/iterate-focusable-elements.js' import {uniqueId} from './utils/unique-id.js' eventListenerSignalPolyfill() @@ -686,6 +686,40 @@ export function focusZone(container: HTMLElement, settings?: FocusZoneSettings): return focusedIndex !== -1 ? focusedIndex : fallbackIndex } + function handleKeydownInteraction(direction: Direction, key: string) { + const lastFocusedIndex = getCurrentFocusedIndex() + let nextFocusedIndex = lastFocusedIndex + if (direction === 'previous') { + nextFocusedIndex -= 1 + } else if (direction === 'start') { + nextFocusedIndex = 0 + } else if (direction === 'next') { + nextFocusedIndex += 1 + } else { + // end + nextFocusedIndex = focusableElements.length - 1 + } + + if (nextFocusedIndex < 0) { + // Tab should never cause focus to wrap. Use focusTrap for that behavior. + if (focusOutBehavior === 'wrap' && key !== 'Tab') { + nextFocusedIndex = focusableElements.length - 1 + } else { + nextFocusedIndex = 0 + } + } + if (nextFocusedIndex >= focusableElements.length) { + if (focusOutBehavior === 'wrap' && key !== 'Tab') { + nextFocusedIndex = 0 + } else { + nextFocusedIndex = focusableElements.length - 1 + } + } + if (lastFocusedIndex !== nextFocusedIndex) { + return focusableElements[nextFocusedIndex] + } + } + // "keydown" is the event that triggers DOM focus change, so that is what we use here keyboardEventRecipient.addEventListener( 'keydown', @@ -708,46 +742,23 @@ export function focusZone(container: HTMLElement, settings?: FocusZoneSettings): nextElementToFocus = settings.getNextFocusable(direction, document.activeElement ?? undefined, event) } if (!nextElementToFocus) { - const lastFocusedIndex = getCurrentFocusedIndex() - let nextFocusedIndex = lastFocusedIndex - if (direction === 'previous') { - nextFocusedIndex -= 1 - } else if (direction === 'start') { - nextFocusedIndex = 0 - } else if (direction === 'next') { - nextFocusedIndex += 1 - } else { - // end - nextFocusedIndex = focusableElements.length - 1 - } - - if (nextFocusedIndex < 0) { - // Tab should never cause focus to wrap. Use focusTrap for that behavior. - if (focusOutBehavior === 'wrap' && event.key !== 'Tab') { - nextFocusedIndex = focusableElements.length - 1 - } else { - nextFocusedIndex = 0 - } - } - if (nextFocusedIndex >= focusableElements.length) { - if (focusOutBehavior === 'wrap' && event.key !== 'Tab') { - nextFocusedIndex = 0 - } else { - nextFocusedIndex = focusableElements.length - 1 - } - } - if (lastFocusedIndex !== nextFocusedIndex) { - nextElementToFocus = focusableElements[nextFocusedIndex] - } + nextElementToFocus = handleKeydownInteraction(direction, event.key) } if (activeDescendantControl) { updateFocusedElement(nextElementToFocus || currentFocusedElement, true) } else if (nextElementToFocus) { + // If for some reason the next element to focus is not focusable (e.g. dynamically hidden), we don't want to attempt to focus it. + // Instead, we want to remove that specific element from focus management. + if (!isFocusable(nextElementToFocus)) { + endFocusManagement(nextElementToFocus) + nextElementToFocus = handleKeydownInteraction(direction, event.key) + } + lastKeyboardFocusDirection = direction // updateFocusedElement will be called implicitly when focus moves, as long as the event isn't prevented somehow - nextElementToFocus.focus({preventScroll}) + nextElementToFocus?.focus({preventScroll}) } // Tab should always allow escaping from this container, so only // preventDefault if tab key press already resulted in a focus movement From 755f319146bd176ce24019f8fa7bef91911eea13 Mon Sep 17 00:00:00 2001 From: Tyler Jones Date: Wed, 20 Dec 2023 13:30:00 -0500 Subject: [PATCH 2/5] Add test --- src/__tests__/focus-zone.test.tsx | 33 +++++++++++++++++++++++++++++++ 1 file changed, 33 insertions(+) diff --git a/src/__tests__/focus-zone.test.tsx b/src/__tests__/focus-zone.test.tsx index 9d94cbb..6593c68 100644 --- a/src/__tests__/focus-zone.test.tsx +++ b/src/__tests__/focus-zone.test.tsx @@ -649,3 +649,36 @@ it('Shoud move to tabbable elements if onlyTabbable', async () => { controller.abort() }) + +it('Should ignore hidden elements after focus zone is enabled', async () => { + const user = userEvent.setup() + const {container, rerender} = render( +
+ + + +
, + ) + + const focusZoneContainer = container.querySelector('#focusZone')! + const [firstButton, , thirdButton] = focusZoneContainer.querySelectorAll('button') + const controller = focusZone(focusZoneContainer) + + firstButton.focus() + expect(document.activeElement).toEqual(firstButton) + + rerender( +
+ + + +
, + ) + + await user.keyboard('{arrowdown}') + expect(document.activeElement).toEqual(thirdButton) + + controller.abort() +}) From 62acf7344fb7b669635ca311be37f9697133bc51 Mon Sep 17 00:00:00 2001 From: Tyler Jones Date: Wed, 20 Dec 2023 17:01:39 -0500 Subject: [PATCH 3/5] Adjust mutation observer to watch attributes (`hidden`, `disabled`) --- src/focus-zone.ts | 93 +++++++++++++++++++++++++---------------------- 1 file changed, 49 insertions(+), 44 deletions(-) diff --git a/src/focus-zone.ts b/src/focus-zone.ts index f91e7ea..c4e493f 100644 --- a/src/focus-zone.ts +++ b/src/focus-zone.ts @@ -1,6 +1,6 @@ import {polyfill as eventListenerSignalPolyfill} from './polyfills/event-listener-signal.js' import {isMacOS} from './utils/user-agent.js' -import {IterateFocusableElements, isFocusable, iterateFocusableElements} from './utils/iterate-focusable-elements.js' +import {IterateFocusableElements, iterateFocusableElements} from './utils/iterate-focusable-elements.js' import {uniqueId} from './utils/unique-id.js' eventListenerSignalPolyfill() @@ -527,6 +527,12 @@ export function focusZone(container: HTMLElement, settings?: FocusZoneSettings): endFocusManagement(...iterateFocusableElements(removedNode, iterateFocusableElementsOptions)) } } + // If an element is hidden or disabled, remove it from the list of focusable elements + if (mutation.type === 'attributes' && mutation.oldValue === null) { + if (mutation.target instanceof HTMLElement) { + endFocusManagement(mutation.target) + } + } } for (const mutation of mutations) { for (const addedNode of mutation.addedNodes) { @@ -534,12 +540,22 @@ export function focusZone(container: HTMLElement, settings?: FocusZoneSettings): beginFocusManagement(...iterateFocusableElements(addedNode, iterateFocusableElementsOptions)) } } + + // Similarly, if an element is un-hidden or "enabled", add it to the list of focusable elements + // If `mutation.oldValue` is not null, then we may assume that the element was previously hidden or disabled + if (mutation.type === 'attributes' && mutation.oldValue !== null) { + if (mutation.target instanceof HTMLElement) { + beginFocusManagement(mutation.target) + } + } } }) observer.observe(container, { subtree: true, childList: true, + attributeFilter: ['hidden', 'disabled'], + attributeOldValue: true, }) const controller = new AbortController() @@ -686,40 +702,6 @@ export function focusZone(container: HTMLElement, settings?: FocusZoneSettings): return focusedIndex !== -1 ? focusedIndex : fallbackIndex } - function handleKeydownInteraction(direction: Direction, key: string) { - const lastFocusedIndex = getCurrentFocusedIndex() - let nextFocusedIndex = lastFocusedIndex - if (direction === 'previous') { - nextFocusedIndex -= 1 - } else if (direction === 'start') { - nextFocusedIndex = 0 - } else if (direction === 'next') { - nextFocusedIndex += 1 - } else { - // end - nextFocusedIndex = focusableElements.length - 1 - } - - if (nextFocusedIndex < 0) { - // Tab should never cause focus to wrap. Use focusTrap for that behavior. - if (focusOutBehavior === 'wrap' && key !== 'Tab') { - nextFocusedIndex = focusableElements.length - 1 - } else { - nextFocusedIndex = 0 - } - } - if (nextFocusedIndex >= focusableElements.length) { - if (focusOutBehavior === 'wrap' && key !== 'Tab') { - nextFocusedIndex = 0 - } else { - nextFocusedIndex = focusableElements.length - 1 - } - } - if (lastFocusedIndex !== nextFocusedIndex) { - return focusableElements[nextFocusedIndex] - } - } - // "keydown" is the event that triggers DOM focus change, so that is what we use here keyboardEventRecipient.addEventListener( 'keydown', @@ -742,23 +724,46 @@ export function focusZone(container: HTMLElement, settings?: FocusZoneSettings): nextElementToFocus = settings.getNextFocusable(direction, document.activeElement ?? undefined, event) } if (!nextElementToFocus) { - nextElementToFocus = handleKeydownInteraction(direction, event.key) + const lastFocusedIndex = getCurrentFocusedIndex() + let nextFocusedIndex = lastFocusedIndex + if (direction === 'previous') { + nextFocusedIndex -= 1 + } else if (direction === 'start') { + nextFocusedIndex = 0 + } else if (direction === 'next') { + nextFocusedIndex += 1 + } else { + // end + nextFocusedIndex = focusableElements.length - 1 + } + + if (nextFocusedIndex < 0) { + // Tab should never cause focus to wrap. Use focusTrap for that behavior. + if (focusOutBehavior === 'wrap' && event.key !== 'Tab') { + nextFocusedIndex = focusableElements.length - 1 + } else { + nextFocusedIndex = 0 + } + } + if (nextFocusedIndex >= focusableElements.length) { + if (focusOutBehavior === 'wrap' && event.key !== 'Tab') { + nextFocusedIndex = 0 + } else { + nextFocusedIndex = focusableElements.length - 1 + } + } + if (lastFocusedIndex !== nextFocusedIndex) { + nextElementToFocus = focusableElements[nextFocusedIndex] + } } if (activeDescendantControl) { updateFocusedElement(nextElementToFocus || currentFocusedElement, true) } else if (nextElementToFocus) { - // If for some reason the next element to focus is not focusable (e.g. dynamically hidden), we don't want to attempt to focus it. - // Instead, we want to remove that specific element from focus management. - if (!isFocusable(nextElementToFocus)) { - endFocusManagement(nextElementToFocus) - nextElementToFocus = handleKeydownInteraction(direction, event.key) - } - lastKeyboardFocusDirection = direction // updateFocusedElement will be called implicitly when focus moves, as long as the event isn't prevented somehow - nextElementToFocus?.focus({preventScroll}) + nextElementToFocus.focus({preventScroll}) } // Tab should always allow escaping from this container, so only // preventDefault if tab key press already resulted in a focus movement From 5d7e0b1d5411d20ed9710bcec22ce3716132b47f Mon Sep 17 00:00:00 2001 From: Tyler Jones Date: Wed, 20 Dec 2023 17:12:44 -0500 Subject: [PATCH 4/5] Add changeset --- .changeset/eighty-steaks-camp.md | 5 +++++ 1 file changed, 5 insertions(+) create mode 100644 .changeset/eighty-steaks-camp.md diff --git a/.changeset/eighty-steaks-camp.md b/.changeset/eighty-steaks-camp.md new file mode 100644 index 0000000..d06107d --- /dev/null +++ b/.changeset/eighty-steaks-camp.md @@ -0,0 +1,5 @@ +--- +'@primer/behaviors': minor +--- + +Adjusts mutation observer to now track `hidden` and `disabled` attributes being applied or removed. From 160e95d19f93fcf5e0fe44d8387fa61f51db020f Mon Sep 17 00:00:00 2001 From: Tyler Jones Date: Wed, 20 Dec 2023 17:42:00 -0500 Subject: [PATCH 5/5] Add additional test cases --- src/__tests__/focus-zone.test.tsx | 69 +++++++++++++++++++++++++++++++ src/focus-zone.ts | 2 +- 2 files changed, 70 insertions(+), 1 deletion(-) diff --git a/src/__tests__/focus-zone.test.tsx b/src/__tests__/focus-zone.test.tsx index 6593c68..aa51c3a 100644 --- a/src/__tests__/focus-zone.test.tsx +++ b/src/__tests__/focus-zone.test.tsx @@ -682,3 +682,72 @@ it('Should ignore hidden elements after focus zone is enabled', async () => { controller.abort() }) + +it('Should respect unhidden elements after focus zone is enabled', async () => { + const user = userEvent.setup() + const {container, rerender} = render( +
+ + + +
, + ) + + const focusZoneContainer = container.querySelector('#focusZone')! + const [firstButton, secondButton, thirdButton] = focusZoneContainer.querySelectorAll('button') + const controller = focusZone(focusZoneContainer) + + firstButton.focus() + expect(document.activeElement).toEqual(firstButton) + + await user.keyboard('{arrowdown}') + expect(document.activeElement).toEqual(thirdButton) + + rerender( +
+ + + +
, + ) + + await user.keyboard('{arrowup}') + expect(document.activeElement).toEqual(secondButton) + + controller.abort() +}) + +it('Should ignore disabled elements after focus zone is enabled', async () => { + const user = userEvent.setup() + const {container, rerender} = render( +
+ + + +
, + ) + + const focusZoneContainer = container.querySelector('#focusZone')! + const [firstButton, , thirdButton] = focusZoneContainer.querySelectorAll('button') + const controller = focusZone(focusZoneContainer) + + firstButton.focus() + expect(document.activeElement).toEqual(firstButton) + + rerender( +
+ + + +
, + ) + + await user.keyboard('{arrowdown}') + expect(document.activeElement).toEqual(thirdButton) + + controller.abort() +}) diff --git a/src/focus-zone.ts b/src/focus-zone.ts index c4e493f..1ee75c0 100644 --- a/src/focus-zone.ts +++ b/src/focus-zone.ts @@ -541,7 +541,7 @@ export function focusZone(container: HTMLElement, settings?: FocusZoneSettings): } } - // Similarly, if an element is un-hidden or "enabled", add it to the list of focusable elements + // Similarly, if an element is unhidden or "enabled", add it to the list of focusable elements // If `mutation.oldValue` is not null, then we may assume that the element was previously hidden or disabled if (mutation.type === 'attributes' && mutation.oldValue !== null) { if (mutation.target instanceof HTMLElement) {