File "index.tsx"

Full Path: /home/warrior1/public_html/languages/wp-content/plugins/woocommerce/packages/woocommerce-blocks/assets/js/base/hocs/with-scroll-to-top/index.tsx
File size: 1.94 KB
MIME-type: text/x-java
Charset: utf-8

/**
 * 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;