关于react.js:React技巧之检查元素是否可见

原文链接:https://bobbyhadz.com/blog/react-check-if-element-in-viewport

作者:Borislav Hadzhiev

注释从这开始~

总览

在React中,查看元素是否在视口范畴内:

  1. 在元素上设置ref属性。
  2. 应用IntersectionObserver API来跟踪元素是否与视口相交。
import {useEffect, useRef, useState, useMemo} from 'react';

export default function App() {
  const ref1 = useRef(null);
  const ref2 = useRef(null);

  const isInViewport1 = useIsInViewport(ref1);
  console.log('isInViewport1: ', isInViewport1);

  const isInViewport2 = useIsInViewport(ref2);
  console.log('isInViewport2: ', isInViewport2);

  return (
    <div>
      <div ref={ref1}>Top div {isInViewport1 && '| in viewport ✅'}</div>

      <div style={{height: '155rem'}} />

      <div ref={ref2}>Bottom div {isInViewport2 && '| in viewport ✅'}</div>
    </div>
  );
}

function useIsInViewport(ref) {
  const [isIntersecting, setIsIntersecting] = useState(false);

  const observer = useMemo(
    () =>
      new IntersectionObserver(([entry]) =>
        setIsIntersecting(entry.isIntersecting),
      ),
    [],
  );

  useEffect(() => {
    observer.observe(ref.current);

    return () => {
      observer.disconnect();
    };
  }, [ref, observer]);

  return isIntersecting;
}

该示例向咱们展现了,如何查看元素是否在视口范畴内。IntersectionObserver API使咱们可能查看一个给定的元素是否与文档相交。

useIsInViewport钩子接管一个指向咱们想要追踪的元素的ref对象。

IntersectionObserver

IntersectionObserver构造函数接管一个函数,该函数被调用时带有一个entry数组。entry是一个数组,其蕴含了所有的obeserver的指标元素。这些元素的可见度曾经高于或低于intersection observer的比率之一。

每个entry都形容了一个给定元素与根元素(文档)相交的水平。咱们解构了这个entry,因为咱们的IntersectionObserver只能跟踪一个元素(就是咱们设置ref的那个元素)。

咱们调用observe()办法,将咱们要跟踪的元素传给它 – observer.observe(ref.current)

每当元素进入视口或者存在于视口中时,咱们传递给IntersectionObserver()构造函数的函数就会被调用,而后更新state变量。

// 👇️ gets called every time element enters or leaves viewport
new IntersectionObserver(([entry]) =>
  setIsIntersecting(entry.isIntersecting),
)

如果咱们设置ref对象的元素在视口中,useIsInViewport钩子将会返回true。如果元素不在视口中,该钩子将会返回false

须要留神的是,在初始渲染时,useIsInViewport 钩子将会返回false 。因为咱们为useState传递的初始值为falseconst [isIntersecting, setIsIntersecting] = useState(false);

如果你想跟踪钩子的返回值的变动,请应用useEffect钩子,并将该值增加到钩子的依赖关系中。

const isInViewport1 = useIsInViewport(ref1);
console.log('isInViewport1: ', isInViewport1);

useEffect(() => {
  // 👇️ listen for changes
  console.log(isInViewport1);
}, [isInViewport1]);

评论

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

这个站点使用 Akismet 来减少垃圾评论。了解你的评论数据如何被处理