From f19a7a51be930da22a1067c48d0a5c0185a345e4 Mon Sep 17 00:00:00 2001 From: Eric Olkowski <70952936+thatblindgeye@users.noreply.github.com> Date: Wed, 23 Oct 2024 11:57:03 -0400 Subject: [PATCH] fix(Menus): added new fix for scroll jump bug (#11119) Co-authored-by: Donald Labaj --- .../src/components/Dropdown/Dropdown.tsx | 23 +++++++++++++++---- .../src/components/Menu/MenuContainer.tsx | 14 +++++++---- .../Pagination/PaginationOptionsMenu.tsx | 14 +++++++---- .../src/components/Select/Select.tsx | 21 ++++++++++++++--- .../src/components/Tabs/OverflowTab.tsx | 10 ++++++-- 5 files changed, 65 insertions(+), 17 deletions(-) diff --git a/packages/react-core/src/components/Dropdown/Dropdown.tsx b/packages/react-core/src/components/Dropdown/Dropdown.tsx index 0fd713e66cc..00c13f9782a 100644 --- a/packages/react-core/src/components/Dropdown/Dropdown.tsx +++ b/packages/react-core/src/components/Dropdown/Dropdown.tsx @@ -71,6 +71,10 @@ export interface DropdownProps extends MenuProps, OUIAProps { maxMenuHeight?: string; /** @beta Flag indicating the first menu item should be focused after opening the dropdown. */ shouldFocusFirstItemOnOpen?: boolean; + /** Flag indicating if scroll on focus of the first menu item should occur. */ + shouldPreventScrollOnItemFocus?: boolean; + /** Time in ms to wait before firing the toggles' focus event. Defaults to 0 */ + focusTimeoutDelay?: number; } const DropdownBase: React.FunctionComponent = ({ @@ -92,6 +96,8 @@ const DropdownBase: React.FunctionComponent = ({ menuHeight, maxMenuHeight, shouldFocusFirstItemOnOpen = false, + shouldPreventScrollOnItemFocus = true, + focusTimeoutDelay = 0, ...props }: DropdownProps) => { const localMenuRef = React.useRef(); @@ -112,8 +118,8 @@ const DropdownBase: React.FunctionComponent = ({ const firstElement = menuRef?.current?.querySelector( 'li button:not(:disabled),li input:not(:disabled),li a:not([aria-disabled="true"])' ); - firstElement && (firstElement as HTMLElement).focus(); - }, 10); + firstElement && (firstElement as HTMLElement).focus({ preventScroll: shouldPreventScrollOnItemFocus }); + }, focusTimeoutDelay); } prevIsOpen.current = isOpen; @@ -151,7 +157,16 @@ const DropdownBase: React.FunctionComponent = ({ window.removeEventListener('keydown', handleMenuKeys); window.removeEventListener('click', handleClick); }; - }, [isOpen, menuRef, toggleRef, onOpenChange, onOpenChangeKeys]); + }, [ + isOpen, + menuRef, + toggleRef, + onOpenChange, + onOpenChangeKeys, + shouldPreventScrollOnItemFocus, + shouldFocusFirstItemOnOpen, + focusTimeoutDelay + ]); const scrollable = maxMenuHeight !== undefined || menuHeight !== undefined || isScrollable; @@ -161,7 +176,7 @@ const DropdownBase: React.FunctionComponent = ({ ref={menuRef} onSelect={(event, value) => { onSelect && onSelect(event, value); - shouldFocusToggleOnSelect && toggleRef.current.focus(); + shouldFocusToggleOnSelect && toggleRef.current?.focus(); }} isPlain={isPlain} isScrollable={scrollable} diff --git a/packages/react-core/src/components/Menu/MenuContainer.tsx b/packages/react-core/src/components/Menu/MenuContainer.tsx index 72c19c57601..c1b96273b90 100644 --- a/packages/react-core/src/components/Menu/MenuContainer.tsx +++ b/packages/react-core/src/components/Menu/MenuContainer.tsx @@ -39,6 +39,10 @@ export interface MenuContainerProps { popperProps?: MenuPopperProps; /** @beta Flag indicating the first menu item should be focused after opening the dropdown. */ shouldFocusFirstItemOnOpen?: boolean; + /** Flag indicating if scroll on focus of the first menu item should occur. */ + shouldPreventScrollOnItemFocus?: boolean; + /** Time in ms to wait before firing the toggles' focus event. Defaults to 0 */ + focusTimeoutDelay?: number; } /** @@ -54,7 +58,9 @@ export const MenuContainer: React.FunctionComponent = ({ zIndex = 9999, popperProps, onOpenChangeKeys = ['Escape', 'Tab'], - shouldFocusFirstItemOnOpen = true + shouldFocusFirstItemOnOpen = true, + shouldPreventScrollOnItemFocus = true, + focusTimeoutDelay = 0 }: MenuContainerProps) => { const prevIsOpen = React.useRef(isOpen); React.useEffect(() => { @@ -64,8 +70,8 @@ export const MenuContainer: React.FunctionComponent = ({ const firstElement = menuRef?.current?.querySelector( 'li button:not(:disabled),li input:not(:disabled),li a:not([aria-disabled="true"])' ); - firstElement && (firstElement as HTMLElement).focus(); - }, 10); + firstElement && (firstElement as HTMLElement).focus({ preventScroll: shouldPreventScrollOnItemFocus }); + }, focusTimeoutDelay); } prevIsOpen.current = isOpen; @@ -102,7 +108,7 @@ export const MenuContainer: React.FunctionComponent = ({ window.removeEventListener('keydown', handleMenuKeys); window.removeEventListener('click', handleClick); }; - }, [isOpen, menuRef, onOpenChange, onOpenChangeKeys, toggleRef]); + }, [focusTimeoutDelay, isOpen, menuRef, onOpenChange, onOpenChangeKeys, shouldPreventScrollOnItemFocus, toggleRef]); return ( ; /** @beta The container to append the pagination options menu to. Overrides the containerRef prop. */ appendTo?: HTMLElement | (() => HTMLElement) | 'inline'; + /** Flag indicating if scroll on focus of the first menu item should occur. */ + shouldPreventScrollOnItemFocus?: boolean; + /** Time in ms to wait before firing the toggles' focus event. Defaults to 0 */ + focusTimeoutDelay?: number; } export const PaginationOptionsMenu: React.FunctionComponent = ({ @@ -81,7 +85,9 @@ export const PaginationOptionsMenu: React.FunctionComponent null as any, containerRef, - appendTo + appendTo, + shouldPreventScrollOnItemFocus = true, + focusTimeoutDelay = 0 }: PaginationOptionsMenuProps) => { const [isOpen, setIsOpen] = React.useState(false); const toggleRef = React.useRef(null); @@ -134,8 +140,8 @@ export const PaginationOptionsMenu: React.FunctionComponent { const firstElement = menuRef?.current?.querySelector('li button:not(:disabled)'); - firstElement && (firstElement as HTMLElement).focus(); - }, 0); + firstElement && (firstElement as HTMLElement).focus({ preventScroll: shouldPreventScrollOnItemFocus }); + }, focusTimeoutDelay); } // If the event is not on the toggle, close the menu @@ -155,7 +161,7 @@ export const PaginationOptionsMenu: React.FunctionComponent perPageOptions.map(({ value, title }) => ( diff --git a/packages/react-core/src/components/Select/Select.tsx b/packages/react-core/src/components/Select/Select.tsx index 2edf7638af7..ace74b9eeb1 100644 --- a/packages/react-core/src/components/Select/Select.tsx +++ b/packages/react-core/src/components/Select/Select.tsx @@ -78,6 +78,10 @@ export interface SelectProps extends MenuProps, OUIAProps { maxMenuHeight?: string; /** Indicates if the select menu should be scrollable */ isScrollable?: boolean; + /** Flag indicating if scroll on focus of the first menu item should occur. */ + shouldPreventScrollOnItemFocus?: boolean; + /** Time in ms to wait before firing the toggles' focus event. Defaults to 0 */ + focusTimeoutDelay?: number; } const SelectBase: React.FunctionComponent = ({ @@ -99,6 +103,8 @@ const SelectBase: React.FunctionComponent = ({ menuHeight, maxMenuHeight, isScrollable, + shouldPreventScrollOnItemFocus = true, + focusTimeoutDelay = 0, ...props }: SelectProps & OUIAProps) => { const localMenuRef = React.useRef(); @@ -116,8 +122,8 @@ const SelectBase: React.FunctionComponent = ({ if (prevIsOpen.current === false && isOpen === true && shouldFocusFirstItemOnOpen) { setTimeout(() => { const firstElement = menuRef?.current?.querySelector('li button:not(:disabled),li input:not(:disabled)'); - firstElement && (firstElement as HTMLElement).focus(); - }, 10); + firstElement && (firstElement as HTMLElement).focus({ preventScroll: shouldPreventScrollOnItemFocus }); + }, focusTimeoutDelay); } prevIsOpen.current = isOpen; @@ -156,7 +162,16 @@ const SelectBase: React.FunctionComponent = ({ window.removeEventListener('keydown', handleMenuKeys); window.removeEventListener('click', handleClick); }; - }, [isOpen, menuRef, toggleRef, onOpenChange, onOpenChangeKeys]); + }, [ + isOpen, + menuRef, + toggleRef, + onOpenChange, + onOpenChangeKeys, + shouldPreventScrollOnItemFocus, + shouldFocusFirstItemOnOpen, + focusTimeoutDelay + ]); const menu = ( { toggleAriaLabel?: string; /** z-index of the overflow tab */ zIndex?: number; + /** Flag indicating if scroll on focus of the first menu item should occur. */ + shouldPreventScrollOnItemFocus?: boolean; + /** Time in ms to wait before firing the toggles' focus event. Defaults to 0 */ + focusTimeoutDelay?: number; } export const OverflowTab: React.FunctionComponent = ({ @@ -30,6 +34,8 @@ export const OverflowTab: React.FunctionComponent = ({ defaultTitleText = 'More', toggleAriaLabel, zIndex = 9999, + shouldPreventScrollOnItemFocus = true, + focusTimeoutDelay = 0, ...props }: OverflowTabProps) => { const menuRef = React.useRef(); @@ -78,9 +84,9 @@ export const OverflowTab: React.FunctionComponent = ({ setTimeout(() => { if (menuRef?.current) { const firstElement = menuRef.current.querySelector('li > button,input:not(:disabled)'); - firstElement && (firstElement as HTMLElement).focus(); + firstElement && (firstElement as HTMLElement).focus({ preventScroll: shouldPreventScrollOnItemFocus }); } - }, 0); + }, focusTimeoutDelay); }; const overflowTab = (