/** * External dependencies */ import { useRef } from '@wordpress/element'; /** * Internal dependencies */ import './style.scss'; interface ScrollToTopProps { focusableSelector?: string; } const maybeScrollToTop = ( scrollPoint: HTMLElement ): void => { if ( ! scrollPoint ) { return; } const yPos = scrollPoint.getBoundingClientRect().bottom; const isScrollPointVisible = yPos >= 0 && yPos <= window.innerHeight; if ( ! isScrollPointVisible ) { scrollPoint.scrollIntoView(); } }; const moveFocusToElement = ( scrollPoint: HTMLElement, focusableSelector: string ): void => { const focusableElements = scrollPoint.parentElement?.querySelectorAll( focusableSelector ) || []; if ( focusableElements.length ) { const targetElement = focusableElements[ 0 ] as HTMLElement; maybeScrollToTop( targetElement ); targetElement?.focus(); } else { maybeScrollToTop( scrollPoint ); } }; const scrollToHTMLElement = ( scrollPoint: HTMLElement, options: ScrollToTopProps ): void => { const { focusableSelector } = options || {}; if ( ! window || ! Number.isFinite( window.innerHeight ) ) { return; } if ( focusableSelector ) { moveFocusToElement( scrollPoint, focusableSelector ); } else { maybeScrollToTop( scrollPoint ); } }; /** * HOC that provides a function to scroll to the top of the component. */ const withScrollToTop = ( OriginalComponent: React.FunctionComponent< Record< string, unknown > > ) => { return ( props: Record< string, unknown > ): JSX.Element => { const scrollPointRef = useRef< HTMLDivElement >( null ); const scrollToTop = ( args: ScrollToTopProps ) => { if ( scrollPointRef.current !== null ) { scrollToHTMLElement( scrollPointRef.current, args ); } }; return ( <>