mirror of
https://github.com/oiov/wr.do.git
synced 2026-05-14 17:48:35 +08:00
44 lines
1.1 KiB
TypeScript
44 lines
1.1 KiB
TypeScript
import { RefObject, useEffect, useState } from "react";
|
|
|
|
interface Args extends IntersectionObserverInit {
|
|
freezeOnceVisible?: boolean;
|
|
}
|
|
|
|
function useIntersectionObserver(
|
|
elementRef: RefObject<Element>,
|
|
{
|
|
threshold = 0,
|
|
root = null,
|
|
rootMargin = "0%",
|
|
freezeOnceVisible = false,
|
|
}: Args,
|
|
): IntersectionObserverEntry | undefined {
|
|
const [entry, setEntry] = useState<IntersectionObserverEntry>();
|
|
|
|
const frozen = entry?.isIntersecting && freezeOnceVisible;
|
|
|
|
const updateEntry = ([entry]: IntersectionObserverEntry[]): void => {
|
|
setEntry(entry);
|
|
};
|
|
|
|
useEffect(() => {
|
|
const node = elementRef?.current; // DOM Ref
|
|
const hasIOSupport = !!window.IntersectionObserver;
|
|
|
|
if (!hasIOSupport || frozen || !node) return;
|
|
|
|
const observerParams = { threshold, root, rootMargin };
|
|
const observer = new IntersectionObserver(updateEntry, observerParams);
|
|
|
|
observer.observe(node);
|
|
|
|
return () => observer.disconnect();
|
|
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
}, [threshold, root, rootMargin, frozen]);
|
|
|
|
return entry;
|
|
}
|
|
|
|
export default useIntersectionObserver;
|