43 lines
1003 B
TypeScript
43 lines
1003 B
TypeScript
|
import React from "react";
|
||
|
|
||
|
const isInteractionObserverSupported = typeof window !== "undefined" && "IntersectionObserver" in window;
|
||
|
|
||
|
export const useInViewObserver = (onInViewCallback: () => void) => {
|
||
|
const [node, setRef] = React.useState<any>(null);
|
||
|
|
||
|
const onInViewCallbackRef = React.useRef(onInViewCallback);
|
||
|
onInViewCallbackRef.current = onInViewCallback;
|
||
|
|
||
|
React.useEffect(() => {
|
||
|
if (!isInteractionObserverSupported) {
|
||
|
// Skip interaction check if not supported in browser
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
let observer: IntersectionObserver;
|
||
|
if (node && node.parentElement) {
|
||
|
observer = new IntersectionObserver(
|
||
|
([entry]) => {
|
||
|
if (entry.isIntersecting) {
|
||
|
onInViewCallbackRef.current();
|
||
|
}
|
||
|
},
|
||
|
{
|
||
|
root: document.body,
|
||
|
}
|
||
|
);
|
||
|
observer.observe(node);
|
||
|
}
|
||
|
|
||
|
return () => {
|
||
|
if (observer) {
|
||
|
observer.disconnect();
|
||
|
}
|
||
|
};
|
||
|
}, [node]);
|
||
|
|
||
|
return {
|
||
|
ref: setRef,
|
||
|
};
|
||
|
};
|