From fb6039569c276aa860057ef92df5230cccc36e8d Mon Sep 17 00:00:00 2001 From: faultyserver Date: Sat, 16 May 2020 15:11:16 -0700 Subject: [PATCH] ignore previousElement in wrap focus calculation. If a consumer manually calls `.focus()`, `relatedTarget` won't be set on the focus event, which was causing problems where some people would force focus around in a modal on tab presses, only to see focus attempt to "wrap" back to the beginning of the modal from the middle of it. --- src/useFocusLock.tsx | 8 +++----- 1 file changed, 3 insertions(+), 5 deletions(-) diff --git a/src/useFocusLock.tsx b/src/useFocusLock.tsx index 02e1ac4..0f7680a 100644 --- a/src/useFocusLock.tsx +++ b/src/useFocusLock.tsx @@ -93,7 +93,7 @@ function createFocusWalker(root: HTMLElement) { }); } -function wrapFocus(root: HTMLElement, target: Element, previousElement?: Element | null) { +function wrapFocus(root: HTMLElement, target: Element) { const walker = createFocusWalker(root); const position = target.compareDocumentPosition(root); let wrappedTarget = null; @@ -101,7 +101,7 @@ function wrapFocus(root: HTMLElement, target: Element, previousElement?: Element // previousElement would be null if a) there wasn't an element focused before // or b) focus came into the document from outside. In either case, it can be // treated as a reset point to bring focus back into the layer. - if (position & Node.DOCUMENT_POSITION_PRECEDING || previousElement == null) { + if (position & Node.DOCUMENT_POSITION_PRECEDING) { wrappedTarget = walker.firstChild() as HTMLElement | null; } else if (position & Node.DOCUMENT_POSITION_FOLLOWING) { wrappedTarget = walker.lastChild() as HTMLElement | null; @@ -162,11 +162,9 @@ export function useFocusLock( if (root == null) return; const newFocusElement = (event.target as Element | null) || document.body; - const prevFocusElement = event.relatedTarget as Element | null; - if (!root.contains(newFocusElement)) { event.preventDefault(); - wrapFocus(root, newFocusElement, prevFocusElement); + wrapFocus(root, newFocusElement); } }