import { cn } from "helpers/common.helper"; import React, { useState, useRef, useEffect, ReactNode, MutableRefObject } from "react"; type Props = { defaultHeight?: string; verticalOffset?: number; horizonatlOffset?: number; root?: MutableRefObject; children: ReactNode; as?: keyof JSX.IntrinsicElements; classNames?: string; alwaysRender?: boolean; placeholderChildren?: ReactNode; pauseHeightUpdateWhileRendering?: boolean; changingReference?: any; }; const RenderIfVisible: React.FC = (props) => { const { defaultHeight = "300px", root, verticalOffset = 50, horizonatlOffset = 0, as = "div", children, classNames = "", alwaysRender = false, //render the children even if it is not visble in root placeholderChildren = null, //placeholder children pauseHeightUpdateWhileRendering = false, //while this is true the height of the blocks are maintained changingReference, //This is to force render when this reference is changed } = props; const [shouldVisible, setShouldVisible] = useState(alwaysRender); const placeholderHeight = useRef(defaultHeight); const intersectionRef = useRef(null); const isVisible = alwaysRender || shouldVisible; // Set visibility with intersection observer useEffect(() => { if (intersectionRef.current) { const observer = new IntersectionObserver( (entries) => { if (typeof window !== undefined && window.requestIdleCallback) { window.requestIdleCallback(() => setShouldVisible(entries[0].isIntersecting), { timeout: 300, }); } else { setShouldVisible(entries[0].isIntersecting); } }, { root: root?.current, rootMargin: `${verticalOffset}% ${horizonatlOffset}% ${verticalOffset}% ${horizonatlOffset}%`, } ); observer.observe(intersectionRef.current); return () => { if (intersectionRef.current) { observer.unobserve(intersectionRef.current); } }; } }, [root?.current, intersectionRef, children, changingReference]); //Set height after render useEffect(() => { if (intersectionRef.current && isVisible) { placeholderHeight.current = `${intersectionRef.current.offsetHeight}px`; } }, [isVisible, intersectionRef, alwaysRender, pauseHeightUpdateWhileRendering]); const child = isVisible ? <>{children} : placeholderChildren; const style = isVisible && !pauseHeightUpdateWhileRendering ? {} : { height: placeholderHeight.current, width: "100%" }; const className = isVisible ? classNames : cn(classNames, "bg-custom-background-80"); return React.createElement(as, { ref: intersectionRef, style, className }, child); }; export default RenderIfVisible;