UNPKG

1.22 kBJavaScriptView Raw
1import { useState } from 'react';
2import useEffect from './useIsomorphicEffect';
3const targetMap = new WeakMap();
4let resizeObserver;
5function getResizeObserver() {
6 // eslint-disable-next-line no-return-assign
7 return resizeObserver = resizeObserver || new window.ResizeObserver(entries => {
8 entries.forEach(entry => {
9 const handler = targetMap.get(entry.target);
10 if (handler) handler(entry.contentRect);
11 });
12 });
13}
14
15/**
16 * Efficiently observe size changes on an element. Depends on the `ResizeObserver` api,
17 * and polyfills are needed in older browsers.
18 *
19 * ```ts
20 * const [ref, attachRef] = useCallbackRef(null);
21 *
22 * const rect = useResizeObserver(ref);
23 *
24 * return (
25 * <div ref={attachRef}>
26 * {JSON.stringify(rect)}
27 * </div>
28 * )
29 * ```
30 *
31 * @param element The DOM element to observe
32 */
33export default function useResizeObserver(element) {
34 const [rect, setRect] = useState(null);
35 useEffect(() => {
36 if (!element) return;
37 getResizeObserver().observe(element);
38 setRect(element.getBoundingClientRect());
39 targetMap.set(element, rect => {
40 setRect(rect);
41 });
42 return () => {
43 targetMap.delete(element);
44 };
45 }, [element]);
46 return rect;
47}
\No newline at end of file