实现思路:
利用浏览器提供的 IntersectionObserver,监听图片元素是否进入可视区域,进入后才真正去设置图片元素的 src 属性进行图片加载。
1. IntersectionObserver的基本使用
详细请看阮一峰IntersectionObserver API 使用教程
- 作用:监听图片元素是否进入可视区域,进入后才真正去设置图片元素的
src 属性进行图片加载。 - 格式:
var dom = dom元素
var observer = new IntersectionObserver( entries => {
console.log(entries);
}, 其他配置)
observer.observe(dom)
observer.disconnect()
observer.unobserve(dom)
callback一般会触发两次。一次是目标元素刚刚进入视口(开始可见),另一次是完全离开视口(开始不可见)。
- 参数1 :callback 参数
- isIntersecting:当前DOM是否进入到视口 是一个布尔值
- time:可见性发生变化的时间,是一个高精度时间戳,单位为毫秒
- target:被观察的目标元素,是一个 DOM 节点对象
- rootBounds:根元素的矩形区域的信息,getBoundingClientRect()方法的返回值,如果没有根元素(即直接相对于视口滚动),则返回null
- boundingClientRect:目标元素的矩形区域的信息
- intersectionRect:目标元素与视口(或根元素)的交叉区域的信息
- intersectionRatio:目标元素的可见比例,即intersectionRect占boundingClientRect的比例,完全可见时为1,完全不可见时小于等于0
- 其他配置:
- threshold 属性:threshold属性决定了什么时候触发回调函数。它是一个数组,每个成员都是一个门槛值,默认为[0],即交叉比例(intersectionRatio)达到0时触发回调函数。
- root属性:指定目标元素所在的容器节点(即根元素)。注意,容器元素必须是目标元素的祖先节点。
- rootMargin属性:后者定义根元素的margin,用来扩展或缩小rootBounds这个矩形的大小,从而影响intersectionRect交叉区域的大小。它使用CSS的定义方法,比如10px 20px 30px 40px,表示 top、right、bottom 和 left 四个方向的值。
2.示例代码
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
</head>
<body>
<div>
<p style="padding: 30px;">1</p>
<p style="padding: 30px;">2</p>
<p style="padding: 30px;">3</p>
<p style="padding: 30px;">3</p>
<p style="padding: 30px;">3</p>
<p style="padding: 30px;">3</p>
<p style="padding: 30px;">3</p>
<p style="padding: 30px;">3</p>
<p style="padding: 30px;">3</p>
<p style="padding: 30px;">3</p>
<p style="padding: 30px;">4</p>
<p style="padding: 30px;">5</p>
<p style="padding: 30px;">6</p>
<p style="padding: 30px;">7</p>
<p style="padding: 30px;">8</p>
<p style="padding: 30px;">9</p>
<!-- <img height="200" src="https://gimg2.baidu.com/image_search/src=http%3A%2F%2Ffzn.cc%2Fwp-content%2Fuploads%2F2020%2F04%2F640-8.jpg&refer=http%3A%2F%2Ffzn.cc&app=2002&size=f9999,10000&q=a80&n=0&g=0n&fmt=jpeg?sec=1641365183&t=b5d7bdae0fe3f2c4831b52e3985abdf1" /> -->
<img height="200"
data-src="https://gimg2.baidu.com/image_search/src=http%3A%2F%2Ffzn.cc%2Fwp-content%2Fuploads%2F2020%2F04%2F640-8.jpg&refer=http%3A%2F%2Ffzn.cc&app=2002&size=f9999,10000&q=a80&n=0&g=0n&fmt=jpeg?sec=1641365183&t=b5d7bdae0fe3f2c4831b52e3985abdf1" />
<p style="padding: 30px;">1</p>
<p style="padding: 30px;">2</p>
<p style="padding: 30px;">3</p>
<p style="padding: 30px;">4</p>
<p style="padding: 30px;">5</p>
<p style="padding: 30px;">6</p>
<p style="padding: 30px;">7</p>
<p style="padding: 30px;">8</p>
<p style="padding: 30px;">9</p>
</div>
<script>
var img = document.querySelector("img")
var io = new IntersectionObserver(entries => {
if (entries[0].isIntersecting) {
console.log(img.getAttribute('data-src'));
img.src = img.getAttribute('data-src')
img.onerror = function () {
console.log('图片加载失败');
}
console.log(img.src);
io.disconnect()
}
}, { rootMargin: '100px' });
io.observe(img)
</script>
</body>
</html>
3. 封装一个React懒加载组件
以下是搭配TS些的如果是js就之前删除类型即可 (type,<HTMLImageElement>…这一类) scr/components/Image.tsx
import classnames from 'classnames'
import { useEffect, useRef, useState } from 'react'
import Icon from '../Icon'
import styles from './index.module.scss'
type Props = {
src: string
className?: string
alt?:string
}
const Image = ({ src, className, alt }: Props) => {
const [upload, setUpload] = useState(true)
const [lose, setLose] = useState(true)
const imgRef = useRef<HTMLImageElement>(null)
useEffect(() => {
const ob = new IntersectionObserver((entries) => {
if (entries[0].isIntersecting) {
imgRef.current!.src = imgRef.current?.getAttribute('data-src')!
ob.unobserve(imgRef.current!)
}
}, { rootMargin: '100px' })
ob.observe(imgRef.current!)
return () => {
ob.disconnect()
}
}, [])
return (
<div className={classnames(styles.root, className)}>
{}
{
upload && <div className="image-icon">
<Icon type="iconphoto" />
</div>
}
{}
{
!lose && <div className="image-icon">
<Icon type="iconphoto-fail" />
</div>
}
{}
{<img
onLoad={() => { setUpload(false) }}
onError={() => setLose(false)}
alt={alt}
data-src={src}
ref={imgRef} />
}
</div>
)
}
export default Image
|