woocommerce/plugins/woocommerce-blocks/assets/js/base/hocs/with-scroll-to-top/index.tsx

88 lines
1.9 KiB
TypeScript

/**
* 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 (
<>
<div
className="with-scroll-to-top__scroll-point"
ref={ scrollPointRef }
aria-hidden
/>
<OriginalComponent { ...props } scrollToTop={ scrollToTop } />
</>
);
};
};
export default withScrollToTop;