什么是无限滚动以及它的必要性?
滚动是水平或垂直移动网页上部分内容的用户操作(在大多数情况下)。
就像您在阅读本文时所做的那样。
无限意味着当您向下滚动网页时,新内容会自动加载。
好吧,但是为什么每个人都应该实现它?
可发现性
让我们想象一下您最喜欢的电子商务商店正在举办黑色星期五促销活动。
您在探索页面上找到了几个产品,但当您滚动到网页底部而不是更多产品时,您发现了一个按钮,可将您带到下一个产品列表。
您将能够看到新产品(但前提是您注意到该操作按钮)。
无限滚动只是帮助用户找到更多他们可能错过的内容。
执行
为了实现无限滚动,我们需要检查用户是否到达页面底部或容器。
但是检测滚动的位置是非常昂贵的,并且由于不同的浏览器和设备,其位置值不可靠。
所以一种方法是观看页面的最后内容(元素)及其与视口或容器的交点。
我们如何找到交点?
路口观察者
它是一个 Web API,允许观察内容或列表末尾的元素。
当这个元素(“哨兵”)变得可见(与视口相交时,它会触发回调函数。
通过这个函数我们可以获取更多数据并将其加载到网页中。
整个观察是异步发生的,这最小化对主线程的影响。
为了在 Reactjs 中实现 Intersection Observer,我们将以社交提要为例,我们将在帖子列表上进行无限滚动。
看一下这个组件,您就可以了解下面每个部分的详细情况。
import { useEffect, useRef, useState } from "react"; interface IIntersectionObserverProps {} const allItems = [ "https://picsum.photos/200", "https://picsum.photos/200", "https://picsum.photos/200", "https://picsum.photos/200", ]; const IntersectionObserverImplement: React.FunctionComponent = (props) => { const cardRefs = useRef([]); // Initialize as an empty array const containerRef = useRef<htmldivelement null>(null); const [listItems, setListItems] = useState(allItems); useEffect(() => { const options = { root: containerRef.current, rootMargin: "0px", threshold: 0.5, }; const observer = new IntersectionObserver((entries, observer) => { entries.forEach((entry) => { if (entry.isIntersecting) { setListItems((prevItems) => [ ...prevItems, "https://picsum.photos/200", ]); observer.unobserve(entry.target); // Stop observing the current element } }); }, options); // Observe the last card only const lastCard = cardRefs.current[listItems.length - 1]; if (lastCard) { observer.observe(lastCard); } return () => observer.disconnect(); // Clean up observer on unmount }, [listItems]); return ( <div classname="container" ref="{containerRef}"> {listItems.map((eachItem, index) => ( <div classname="card" ref="{(el)"> (cardRefs.current[index] = el)} // Assign refs correctly key={index} > <h5 id="Post-index">Post {index}</h5> <img src="/static/imghwm/default1.png" data-src="https://img.php.cn/upload/article/000/000/000/173443591411756.jpg?x-oss-process=image/resize,p_40" class="lazy" style="max-width:90%"200"}' height='{"150"}' alt="Reactjs 教程:使用 Intersection Observer 进行无限滚动。" > </div> ))} </div> ); }; export default IntersectionObserverImplement; </htmldivelement>
目标是检测提要列表中的最后一个帖子(称为哨兵)何时与视口相交。一旦发生这种情况,就会加载并显示更多帖子。
一个。初始化状态和引用
const cardRefs = useRef([]); // For storing references to each card const containerRef = useRef<htmldivelement null>(null); // Reference to the scrollable container const [listItems, setListItems] = useState(allItems); // State to hold the list of items </htmldivelement>
cardRefs 一个数组,用于跟踪表示列表中卡片的 DOM 元素。
containerRef 指的是可滚动容器。
listItems 保存页面上当前可见项目的数组。
b.渲染列表并分配引用
return ( <div classname="container" ref="{containerRef}"> {listItems.map((eachItem, index) => ( <div classname="card" ref="{(el)"> (cardRefs.current[index] = el)} // Assign a ref to each card key={index} > <h5 id="Post-index">Post {index}</h5> <img src="/static/imghwm/default1.png" data-src="https://img.php.cn/upload/article/000/000/000/173443591411756.jpg?x-oss-process=image/resize,p_40" class="lazy" style="max-width:90%"200"}' height='{"150"}' alt="Reactjs 教程:使用 Intersection Observer 进行无限滚动。" > </div> ))} </div> );
containerRef 标记将发生滚动的容器。
cardRefs 为列表中的每张卡片分配一个参考。这确保我们可以告诉观察者要监视哪个元素(例如,最后一张卡片)。
映射 listItems 以呈现列表中的每个项目。
每个 div 都被设计成一张卡片,并且有一个唯一的 React 键。
c.观察最后一个帖子(项目)。
import { useEffect, useRef, useState } from "react"; interface IIntersectionObserverProps {} const allItems = [ "https://picsum.photos/200", "https://picsum.photos/200", "https://picsum.photos/200", "https://picsum.photos/200", ]; const IntersectionObserverImplement: React.FunctionComponent = (props) => { const cardRefs = useRef([]); // Initialize as an empty array const containerRef = useRef<htmldivelement null>(null); const [listItems, setListItems] = useState(allItems); useEffect(() => { const options = { root: containerRef.current, rootMargin: "0px", threshold: 0.5, }; const observer = new IntersectionObserver((entries, observer) => { entries.forEach((entry) => { if (entry.isIntersecting) { setListItems((prevItems) => [ ...prevItems, "https://picsum.photos/200", ]); observer.unobserve(entry.target); // Stop observing the current element } }); }, options); // Observe the last card only const lastCard = cardRefs.current[listItems.length - 1]; if (lastCard) { observer.observe(lastCard); } return () => observer.disconnect(); // Clean up observer on unmount }, [listItems]); return ( <div classname="container" ref="{containerRef}"> {listItems.map((eachItem, index) => ( <div classname="card" ref="{(el)"> (cardRefs.current[index] = el)} // Assign refs correctly key={index} > <h5 id="Post-index">Post {index}</h5> <img src="/static/imghwm/default1.png" data-src="https://img.php.cn/upload/article/000/000/000/173443591411756.jpg?x-oss-process=image/resize,p_40" class="lazy" style="max-width:90%"200"}' height='{"150"}' alt="Reactjs 教程:使用 Intersection Observer 进行无限滚动。" > </div> ))} </div> ); }; export default IntersectionObserverImplement; </htmldivelement>
选项对象
const cardRefs = useRef([]); // For storing references to each card const containerRef = useRef<htmldivelement null>(null); // Reference to the scrollable container const [listItems, setListItems] = useState(allItems); // State to hold the list of items </htmldivelement>
root 指定滚动容器。
containerRef.current 指的是包裹所有卡片的 div。
如果 root 为 null,则默认观察视口。
rootMargin:定义根周围的额外边距。
“0px”表示没有多余的空间。您可以使用“100px”之类的值来提前触发观察者(例如,当元素即将出现时)。
阈值:确定观察者触发时目标元素必须可见的程度。
0.5 表示当最后一张卡片的 50% 可见时触发回调。
创建观察者
return ( <div classname="container" ref="{containerRef}"> {listItems.map((eachItem, index) => ( <div classname="card" ref="{(el)"> (cardRefs.current[index] = el)} // Assign a ref to each card key={index} > <h5 id="Post-index">Post {index}</h5> <img src="/static/imghwm/default1.png" data-src="https://img.php.cn/upload/article/000/000/000/173443591411756.jpg?x-oss-process=image/resize,p_40" class="lazy" style="max-width:90%"200"}' height='{"150"}' alt="Reactjs 教程:使用 Intersection Observer 进行无限滚动。" > </div> ))} </div> );
IntersectionObserver 接受回调函数和之前定义的选项对象。
每当观察到的元素满足选项中指定的条件时,回调就会运行。
entries 参数是观察到的元素的数组。每个条目都包含有关元素是否相交(可见)的信息。
如果entry.isIntersecting为true,则意味着最后一张卡片现在可见:
- 使用 setListItems 将新项目添加到列表中。
- 取消观察当前元素(entry.target)以防止冗余触发器。
观察最后一张牌
useEffect(() => { const options = { root: containerRef.current, rootMargin: "0px", threshold: 0.5, }; const observer = new IntersectionObserver((entries, observer) => { entries.forEach((entry) => { if (entry.isIntersecting) { setListItems((prevItems) => [ ...prevItems, "https://picsum.photos/200", ]); observer.unobserve(entry.target); // Stop observing the current element } }); }, options); // Observe each card const lastCard = cardRefs.current[listItems.length - 1]; if (lastCard) { observer.observe(lastCard); } return () => observer.disconnect(); // Clean up observer on unmount }, [listItems]);
cardRefs.current:跟踪对所有卡片的引用。
listItems.length - 1:标识列表中的最后一项。
如果lastCard存在,使用observer.observe(lastCard)开始观察它。
观察者会监听这张卡片,并在它可见时触发回调。
清理
const options = { root: containerRef.current, // Observe within the container rootMargin: "0px", // No margin around the root container threshold: 0.5, // Trigger when 50% of the element is visible };
observer.disconnect() 删除此 useEffect 创建的所有观察者。
这确保了当组件卸载或重新渲染时,旧的观察者被清理。
每个阶段会发生什么?
1。用户滚动
当用户滚动时,最后一张卡片进入视图
2。路口观察者触发器
当最后一张牌的50%可见时,观察者的回调
运行。
3。添加项目
回调将新项目添加到列表中 (setListItems)。
4。重复
观察者与旧的最后一张卡断开连接并附加到
新的最后一张卡。
这就是我们如何使用 Intersection Observer.
实现无限滚动希望这对您有帮助:)
谢谢。
以上是Reactjs 教程:使用 Intersection Observer 进行无限滚动。的详细内容。更多信息请关注PHP中文网其他相关文章!

JavaScript字符串替换方法详解及常见问题解答 本文将探讨两种在JavaScript中替换字符串字符的方法:在JavaScript代码内部替换和在网页HTML内部替换。 在JavaScript代码内部替换字符串 最直接的方法是使用replace()方法: str = str.replace("find","replace"); 该方法仅替换第一个匹配项。要替换所有匹配项,需使用正则表达式并添加全局标志g: str = str.replace(/fi

本教程向您展示了如何将自定义的Google搜索API集成到您的博客或网站中,提供了比标准WordPress主题搜索功能更精致的搜索体验。 令人惊讶的是简单!您将能够将搜索限制为Y

因此,在这里,您准备好了解所有称为Ajax的东西。但是,到底是什么? AJAX一词是指用于创建动态,交互式Web内容的一系列宽松的技术。 Ajax一词,最初由Jesse J创造

本文系列在2017年中期进行了最新信息和新示例。 在此JSON示例中,我们将研究如何使用JSON格式将简单值存储在文件中。 使用键值对符号,我们可以存储任何类型的

利用轻松的网页布局:8个基本插件 jQuery大大简化了网页布局。 本文重点介绍了简化该过程的八个功能强大的JQuery插件,对于手动网站创建特别有用

核心要点 JavaScript 中的 this 通常指代“拥有”该方法的对象,但具体取决于函数的调用方式。 没有当前对象时,this 指代全局对象。在 Web 浏览器中,它由 window 表示。 调用函数时,this 保持全局对象;但调用对象构造函数或其任何方法时,this 指代对象的实例。 可以使用 call()、apply() 和 bind() 等方法更改 this 的上下文。这些方法使用给定的 this 值和参数调用函数。 JavaScript 是一门优秀的编程语言。几年前,这句话可

jQuery是一个很棒的JavaScript框架。但是,与任何图书馆一样,有时有必要在引擎盖下发现发生了什么。也许是因为您正在追踪一个错误,或者只是对jQuery如何实现特定UI感到好奇

该帖子编写了有用的作弊表,参考指南,快速食谱以及用于Android,BlackBerry和iPhone应用程序开发的代码片段。 没有开发人员应该没有他们! 触摸手势参考指南(PDF) Desig的宝贵资源


热AI工具

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Undress AI Tool
免费脱衣服图片

Clothoff.io
AI脱衣机

AI Hentai Generator
免费生成ai无尽的。

热门文章

热工具

适用于 Eclipse 的 SAP NetWeaver 服务器适配器
将Eclipse与SAP NetWeaver应用服务器集成。

MinGW - 适用于 Windows 的极简 GNU
这个项目正在迁移到osdn.net/projects/mingw的过程中,你可以继续在那里关注我们。MinGW:GNU编译器集合(GCC)的本地Windows移植版本,可自由分发的导入库和用于构建本地Windows应用程序的头文件;包括对MSVC运行时的扩展,以支持C99功能。MinGW的所有软件都可以在64位Windows平台上运行。

VSCode Windows 64位 下载
微软推出的免费、功能强大的一款IDE编辑器

螳螂BT
Mantis是一个易于部署的基于Web的缺陷跟踪工具,用于帮助产品缺陷跟踪。它需要PHP、MySQL和一个Web服务器。请查看我们的演示和托管服务。

mPDF
mPDF是一个PHP库,可以从UTF-8编码的HTML生成PDF文件。原作者Ian Back编写mPDF以从他的网站上“即时”输出PDF文件,并处理不同的语言。与原始脚本如HTML2FPDF相比,它的速度较慢,并且在使用Unicode字体时生成的文件较大,但支持CSS样式等,并进行了大量增强。支持几乎所有语言,包括RTL(阿拉伯语和希伯来语)和CJK(中日韩)。支持嵌套的块级元素(如P、DIV),