Skip to content

Commit fcc232b

Browse files
authored
fix(useintersectionobserver): remove ref from props and createRef in useIntersectionObserver (#516)
* fix(useintersectionobserver): remove ref from props and createRef in useIntersectionObserver * fix(useintersectionobserver): change return, use variable
1 parent 96edb4d commit fcc232b

File tree

5 files changed

+38
-31
lines changed

5 files changed

+38
-31
lines changed

src/image/index.tsx

+3-4
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import React, { CSSProperties, useRef } from 'react';
1+
import React, { CSSProperties } from 'react';
22
import { Spin } from 'antd';
33

44
import useIntersectionObserver from '../useIntersectionObserver';
@@ -52,8 +52,7 @@ const ImageComponent = (props: IProps) => {
5252

5353
const LazyImage = (props: IProps) => {
5454
const { src, ...rest } = props;
55-
const imgRef = useRef<HTMLImageElement>(null);
56-
useIntersectionObserver(([entry]) => {
55+
const imgRef = useIntersectionObserver<HTMLImageElement>(([entry]) => {
5756
const { target, isIntersecting } = entry;
5857
if (isIntersecting) {
5958
const _target = target as HTMLImageElement;
@@ -62,7 +61,7 @@ const LazyImage = (props: IProps) => {
6261
_target.style.opacity = '1';
6362
};
6463
}
65-
}, imgRef);
64+
});
6665
return <img ref={imgRef} {...rest} data-src={src} />;
6766
};
6867

src/useIntersectionObserver/__tests__/useIntersectionObserver.test.ts

+18-9
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
import { RefObject } from 'react';
21
import { act, renderHook } from '@testing-library/react-hooks';
32

43
import useIntersectionObserver from '../index';
@@ -31,25 +30,35 @@ describe('useIntersectionObserver', () => {
3130
});
3231

3332
it('should observe target element and disconnect on unmount', () => {
34-
const ref = { current: document.createElement('div') };
3533
const callback = jest.fn();
3634
const options = { threshold: 0, root: null, rootMargin: '0%' };
37-
const { unmount } = renderHook(() => useIntersectionObserver(callback, ref, options));
35+
36+
const divElement = document.createElement('div');
37+
document.body.appendChild(divElement);
38+
39+
const { unmount } = renderHook(() => {
40+
const ref = useIntersectionObserver(callback, options);
41+
ref.current = divElement;
42+
return ref;
43+
});
3844
expect(window.IntersectionObserver).toHaveBeenCalledWith(expect.any(Function), options);
39-
expect(observeMock).toHaveBeenCalledWith(ref.current);
45+
expect(observeMock).toHaveBeenCalledWith(divElement);
4046
act(() => {
4147
unmount();
4248
});
4349
expect(disconnectMock).toHaveBeenCalled();
4450
});
4551

46-
it('should not observe target element if not provided', () => {
52+
it('should not observe target element if ref is null', () => {
4753
const callback = jest.fn();
4854
const options = { threshold: 0, root: null, rootMargin: '0%' };
49-
const { unmount } = renderHook(() =>
50-
useIntersectionObserver(callback, null as unknown as RefObject<Element>, options)
51-
);
52-
expect(window.IntersectionObserver).toHaveBeenCalledWith(expect.any(Function), options);
55+
56+
const { unmount } = renderHook(() => {
57+
const ref = useIntersectionObserver(callback, options);
58+
ref.current = null;
59+
return ref;
60+
});
61+
5362
expect(observeMock).not.toHaveBeenCalled();
5463
act(() => {
5564
unmount();

src/useIntersectionObserver/demos/basic.tsx

+3-5
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,18 @@
1-
import React, { useRef } from 'react';
1+
import React from 'react';
22

33
import useIntersectionObserver from '..';
44

55
const Basic = () => {
6-
const divRef = useRef<HTMLDivElement>(null);
7-
86
const handleObserverCb = ([entry]: IntersectionObserverEntry[]) => {
97
if (entry.isIntersecting) alert('hi, 我展示了');
108
};
119

12-
useIntersectionObserver(handleObserverCb, divRef);
10+
const ref = useIntersectionObserver<HTMLDivElement>(handleObserverCb);
1311

1412
return (
1513
<div style={{ height: 300, overflow: 'scroll' }}>
1614
<div style={{ height: 330 }}>占位,往下滑动</div>
17-
<div ref={divRef}>
15+
<div ref={ref}>
1816
<div>展示了</div>
1917
</div>
2018
</div>

src/useIntersectionObserver/demos/imgLazy.tsx

+2-4
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,8 @@
1-
import React, { useRef } from 'react';
1+
import React from 'react';
22

33
import useIntersectionObserver from '..';
44

55
const Basic = () => {
6-
const imgRef = useRef<HTMLImageElement>(null);
7-
86
const handleObserverCb = ([entry]: IntersectionObserverEntry[]) => {
97
const { target, isIntersecting } = entry;
108
if (isIntersecting) {
@@ -16,7 +14,7 @@ const Basic = () => {
1614
}
1715
};
1816

19-
useIntersectionObserver(handleObserverCb, imgRef, {});
17+
const imgRef = useIntersectionObserver<HTMLImageElement>(handleObserverCb);
2018

2119
return (
2220
<div style={{ height: 300, overflow: 'scroll' }}>

src/useIntersectionObserver/index.ts

+12-9
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,11 @@
1-
import { RefObject, useEffect, useState } from 'react';
1+
import { MutableRefObject, useEffect, useRef, useState } from 'react';
22

3-
const useIntersectionObserver = (
3+
const useIntersectionObserver = <T extends Element>(
44
callback: IntersectionObserverCallback,
5-
target: RefObject<Element>,
65
options: IntersectionObserverInit & { freezeOnceVisible?: boolean } = {}
76
) => {
7+
const ref = useRef<T | null>(null);
8+
89
const { threshold = 0, root = null, rootMargin = '0%', freezeOnceVisible = false } = options;
910
const [entry, setEntry] = useState<IntersectionObserverEntry>();
1011
const frozen = entry?.isIntersecting && freezeOnceVisible;
@@ -18,17 +19,17 @@ const useIntersectionObserver = (
1819
};
1920

2021
useEffect(() => {
21-
const node = target?.current; // DOM Ref
22+
const node = ref.current;
2223
const hasIOSupport = !!window.IntersectionObserver;
2324

2425
if (frozen || !node) return;
2526
if (!hasIOSupport) {
2627
// 如果不支持 IntersectionObserver 执行一个默认行为
2728
const callbackEntry = {
28-
boundingClientRect: node?.getBoundingClientRect() ?? null,
29-
intersectionRatio: node ? 1 : 0,
30-
intersectionRect: node?.getBoundingClientRect() ?? null,
31-
isIntersecting: !!node,
29+
boundingClientRect: node.getBoundingClientRect() ?? null,
30+
intersectionRatio: 1,
31+
intersectionRect: node.getBoundingClientRect() ?? null,
32+
isIntersecting: true,
3233
rootBounds: null,
3334
target: node,
3435
time: Date.now(),
@@ -40,7 +41,9 @@ const useIntersectionObserver = (
4041

4142
observer.observe(node);
4243
return () => observer.disconnect();
43-
}, [target?.current, JSON.stringify(threshold), root, rootMargin, frozen]);
44+
}, [JSON.stringify(threshold), root, rootMargin, frozen]);
45+
46+
return ref as MutableRefObject<T | null>;
4447
};
4548

4649
export default useIntersectionObserver;

0 commit comments

Comments
 (0)