本文由 简悦 SimpRead 转码, 原文地址 mp.weixin.qq.com
最近开发了一个功能,页面首页会加载大量的图片,初次进入页面时,会导致页面性能下降,
于是乎,我改进了这个功能,可以让所有图片自动懒加载。
🚀 原理
这个功能主要的底层逻辑是是使用IntersectionObserver API,IntersectionObserver用于在浏览器中观察元素的可见性和位置变化。它可以帮助开发者实现一些动态行为,如图片的懒加载、无限滚动等。
简单的示例如下:
// 创建IntersectionObserver实例const observer = new IntersectionObserver((entries, observer) => { // 遍历观察的元素 entries.forEach(entry => { // 如果元素可见 if (entry.isIntersecting) { // 加载图片 const img = entry.target; const src = img.getAttribute('data-src'); img.setAttribute('src', src); // 停止观察该元素 observer.unobserve(img); } });});// 获取所有需要懒加载的图片元素const lazyImages = document.querySelectorAll('.lazy-image');// 观察每个图片元素lazyImages.forEach(image => { observer.observe(image);});🚀 实践
接下来我们实现一个通用的 hook,基本的功能如下:
给图片提供默认的占位图片
src,同时提供data-src属性传入图片对应的
ref属性。当图片进入可视区域时,使用
data-src属性替换src属性
import { onMounted, Ref } from "vue";const options = { // root: document.querySelector(".container"), // 根元素,默认为视口 rootMargin: "0px", // 根元素的边距 threshold: 0.5, // 可见性比例阈值 once: true,};function callback( entries: IntersectionObserverEntry[], observer: IntersectionObserver) { entries.forEach((entry) => { // 处理每个目标元素的可见性变化 if (entry.intersectionRatio <= 0) return; const img: Element = entry.target; const src = img.getAttribute("data-src"); img.setAttribute("src", src ?? ""); // 将真实的图片地址赋给 src 属性 observer.unobserve(img); });}export const useInView = (ref: Ref) => { const observer = new IntersectionObserver(callback, options); onMounted(() => { Object.keys(ref.value).forEach((e) => observer.observe(ref.value[e])); });};<script setup lang="ts">import { ref } from "vue";import { useInView } from "./hooks/useInView";const imgRef = ref(null);useInView(imgRef);</script><template> <h4>公众号:萌萌哒草头将军</h4> <div v-for="(_, idx) in new Array(200).fill(11)" > <img ref="imgRef" src="https://via.placeholder.com/200" :data-src="`https://picsum.photos/200/${180 + idx}`" alt="b" /> </div></template>实际效果如下
虽然基本的功能要求已经完成了,但是现在还不够优雅!!!
🚀 优化
接下来,我们增加个过渡动画。每次当加载完图片,就从占位图过渡到正常图片模式。
img.onload = () => { img.setAttribute('class', 'fade-in')}@keyframes fadeIn { from { opacity: 0; } to { opacity: 1; }}/* 应用淡入动画到元素 */.fade-in { animation: fadeIn 0.6s ease-in;}完整代码如下:
import { onMounted, Ref } from "vue";const options = { // root: document.querySelector(".container"), // 根元素,默认为视口 rootMargin: "0px", // 根元素的边距 threshold: 0.5, // 可见性比例阈值 once: true,};function callback( entries: IntersectionObserverEntry[], observer: IntersectionObserver) { entries.forEach((entry) => { if (entry.intersectionRatio <= 0) return; const img = entry.target as HTMLImageElement; const src = img.getAttribute("data-src"); img.setAttribute("src", src ?? ""); // 将真实的图片地址赋给 src 属性 img.onload = () => { img.setAttribute("class", "fade-in"); }; observer.unobserve(img); });}export const useInView = (ref: Ref) => { const observer = new IntersectionObserver( callback, options ); onMounted(() => { Object.keys(ref.value) .forEach((e) => observer.observe(ref.value[e])); });};<script setup lang="ts">import { ref } from "vue";import { useInView } from "./hooks/useInView";const imgRef = ref(null);useInView(imgRef);</script><template> <h4>公众号:萌萌哒草头将军</h4> <div v-for="(_, idx) in new Array(200).fill(11)" style="width: 200px height: 200px;" > <img ref="imgRef" style="height: 100%" src="https://via.placeholder.com/200" :data-src="`https://picsum.photos/200/${180 + idx}`" alt="b" /> </div></template><style scoped>/* 定义淡入动画 */@keyframes fadeIn { from { opacity: 0; } to { opacity: 1; }}/* 应用淡入动画到元素 */.fade-in { animation: fadeIn 0.6s ease-in;}</style>