useIntersectionObserver
A hook that allows to observe the intersection of an element with the viewport.
Example
import { useEffect } from "react";
import useIntersectionObserver from "@/hooks/useIntersectionObserver";
export default function UseIntersectionObserverExample() {
const [ref, inView] = useIntersectionObserver();
useEffect(() => {
if (inView) {
alert("In view");
} else {
alert("Out of view");
}
}, [inView]);
return (
<div ref={ref} className="text-ctp-text">
{inView && <div>In view</div>}
</div>
);
}
Code
import { useEffect, useRef, useState } from "react";
type State = {
isIntersecting: boolean;
entry?: IntersectionObserverEntry;
};
type UseIntersectionObserverOptions = {
root?: Element | Document | null;
rootMargin?: string;
threshold?: number | number[];
freezeOnceVisible?: boolean;
onChange?: (
isIntersecting: boolean,
entry: IntersectionObserverEntry,
) => void;
initialIsIntersecting?: boolean;
};
type IntersectionReturn = [
(node?: Element | null) => void,
boolean,
IntersectionObserverEntry | undefined,
] & {
ref: (node?: Element | null) => void;
isIntersecting: boolean;
entry?: IntersectionObserverEntry;
};
/**
* A hook that allows to observe the intersection of an element with the viewport.
* @param {UseIntersectionObserverOptions} options - The options for the hook.
* @returns {UseIntersectionObserverReturn} An array of three elements:
* 1. A function to update the observed element.
* 2. A boolean indicating whether the element is currently intersecting the viewport.
* 3. An IntersectionObserverEntry object containing information about the intersection.
*/
export default function useIntersectionObserver({
threshold = 0,
root = null,
rootMargin = "0%",
freezeOnceVisible = false,
initialIsIntersecting = false,
onChange,
}: UseIntersectionObserverOptions = {}): IntersectionReturn {
const [ref, setRef] = useState<Element | null>(null);
const [state, setState] = useState<State>(() => ({
isIntersecting: initialIsIntersecting,
entry: undefined,
}));
const callbackRef = useRef<UseIntersectionObserverOptions["onChange"]>();
callbackRef.current = onChange;
const frozen = state.entry?.isIntersecting && freezeOnceVisible;
useEffect(() => {
// Ensure we have a ref to observe
if (!ref) return;
// Ensure the browser supports the Intersection Observer API
if (!("IntersectionObserver" in window)) return;
// Skip if frozen
if (frozen) return;
let unobserve: (() => void) | undefined;
const observer = new IntersectionObserver(
(entries: IntersectionObserverEntry[]): void => {
const thresholds = Array.isArray(observer.thresholds)
? observer.thresholds
: [observer.thresholds];
entries.forEach((entry) => {
const isIntersecting =
entry.isIntersecting &&
thresholds.some(
(threshold) => entry.intersectionRatio >= threshold,
);
setState({ isIntersecting, entry });
if (callbackRef.current) {
callbackRef.current(isIntersecting, entry);
}
if (isIntersecting && freezeOnceVisible && unobserve) {
unobserve();
unobserve = undefined;
}
});
},
{ threshold, root, rootMargin },
);
observer.observe(ref);
return () => {
observer.disconnect();
};
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [
ref,
// eslint-disable-next-line react-hooks/exhaustive-deps
JSON.stringify(threshold),
root,
rootMargin,
frozen,
freezeOnceVisible,
]);
// ensures that if the observed element changes, the intersection observer is reinitialized
const prevRef = useRef<Element | null>(null);
useEffect(() => {
if (
!ref &&
state.entry?.target &&
!freezeOnceVisible &&
!frozen &&
prevRef.current !== state.entry.target
) {
prevRef.current = state.entry.target;
setState({ isIntersecting: initialIsIntersecting, entry: undefined });
}
}, [ref, state.entry, freezeOnceVisible, frozen, initialIsIntersecting]);
const result = [
setRef,
!!state.isIntersecting,
state.entry,
] as IntersectionReturn;
// Support object destructuring, by adding the specific values.
result.ref = result[0];
result.isIntersecting = result[1];
result.entry = result[2];
return result;
}