Vue3实现图片懒加载及自定义懒加载指令

Vue3实现图片懒加载及自定义懒加载指令

  • 前言
  • 1.使用vue3-lazyload插件
  • 2.自定义v-lazy懒加载指令
    • 2.1 使用VueUse
    • 2.2 使用IntersectionObserver

前言

图片懒加载是一种常见性能优化的方式,它只去加载可视区域图片,而不是在网页加载完毕后就立即加载所有图片,能减少很多不必要的请求,极大的提升用户体验。

图片懒加载的实现原理:在图片没进入可视区域的时候,只需要让 img 标签的 src 属性指向一张默认图片,在它进入可视区后,再替换它的 src 指向真实图片地址即可。

本文就分享一下在vue3中实现图片懒加载的几种方式,包括使用插件以及自定义指令,实现的最终效果如下图所示:
在这里插入图片描述

1.使用vue3-lazyload插件

第一种方式就是使用插件,使用插件的方式非常简单,只需要简单的几步即可实现。

Vue2中可以使用vue-lazyload插件来实现图片懒加载,在Vue3中可以使用vue3-lazyload插件实现图片懒加载。

  • 1.安装vue3-lazyload插件
$ npm i vue3-lazyload
# or
$ yarn add vue3-lazyload
# or
$ pnpm i vue3-lazyload
  • 2.main.js入口文件注册插件
import { createApp } from "vue";
import App from "./App.vue";
//引入图片懒加载插件
import Lazyload from "vue3-lazyload";const app = createApp(App);//注册插件
app.use(Lazyload, {loading: "@/assets/images/default.png",//可以指定加载中的图像error: "@/assets/images/err.png",//可以指定加载失败的图像
});app.mount("#app");
  • 3.模板中使用v-lazy指令来延迟加载图像
<template><ul class="container"><li v-for="item in imgList" :key="item.id"><img v-lazy="item.url" class="item" /></li></ul>
</template><script lang="ts" setup>
import { reactive } from "vue";
const data = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10].map((i) => {return {id: `${i}`,url: `@/assets/images/${i}.jpg`,};
});
const imgList = reactive(data);
</script><style scoped lang="scss">
.container {width: 100vw;height: 100vh;overflow: auto;.item {width: 100%;height: 200px;}
}
</style>

2.自定义v-lazy懒加载指令

  • 下面一种方式是自定义一个懒加载的指令,如何实现呢?

图片懒加载的核心是监听图片是否进入可视区域,如果进入就替换src,即懒加载指令的核心。

网上看了很多教程,大多都使用Element.getBoundingClientRect()这个方法,该方法返回一个 DOMRect 对象,提供了元素的大小及其相对于视口的位置,然后监听滚动条事件,通过img.getBoundingClientRect()进行一系列的比较来判断图片是否在视口内,这种方式略显复杂。其实,只要我们能够简化判断图片是否进入可视区域这一流程,实现一个自定义的懒加载指令就很简单了。

  • 有没有什么简化的方式呢?

可以通过VueUse中的useIntersectionObserver原生的IntersectionObserver api来简化判断图片是否进入可视区域,下面就分别通过这两种简化的方式来实现一个自定义的懒加载指令。

2.1 使用VueUse

VueUse 是什么?

一款基于Vue组合式API的函数工具集。

以上是官方网站关于它的定义。

简单的说就是一个工具函数包,它可以帮助你快速实现一些常见的功能。比如下面的一些:

  • useLocalStorage:提供在本地存储中保存和获取数据的功能。
  • useMouse:提供跟踪鼠标位置和鼠标按下状态的功能。
  • useDebounce:提供防抖功能。
  • useThrottle:提供节流功能。
  • useIntersectionObserver:提供对元素是否可见进行观察的功能,可用于实现懒加载等效果。

本文要用到的就是其中的useIntersectionObserver这个函数,来监听图片的可见性。

  • 首先安装 VueUse
npm i @vueuse/core
  • main.js入口文件导入
import { createApp } from "vue";
import App from "./App.vue";//从@vueuse/core中导入useIntersectionObserver函数
import { useIntersectionObserver } from "@vueuse/core";const app = createApp(App);app.mount("#app");
  • directive注册v-lazy全局指令
//main.js
//注册v-lazy全局指令,使v-lazy在所有组件中都可用
app.directive("lazy", {//节点挂载完成后调用mounted(el, binding) {useIntersectionObserver(el, ([{ isIntersecting }]) => {//判断当前监听元素是否进入视口区域if (isIntersecting) {el.src = binding.value;}});},
});

一个指令定义对象可以提供多个钩子函数,比如 mounted、updated、unmounted 等,我们使用mounted,也就是在节点挂载完成后调用。指令的钩子有两个主要的参数:el和binding。el是指令绑定到的元素,binding中使用最多的是value,即传递给指令的值,例如在 v-lazy=“imgSrc” 中,值是 imgSrc对应的真实图片地址。

然后使用useIntersectionObserver函数,它的两个参数,一个是需要监听的元素,另一个是回调函数,参数值isIntersecting为一个布尔值,用来判断当前监听元素是否进入视口区域,如果进入视口区域,那么我们就可以将图片的真实url赋值给图片的src。

其实上述代码还有不完善的地方,首先是重复监听的问题,可以进行console调试一下:

useIntersectionObserver(el, ([{ isIntersecting }]) => {console.log(isIntersecting);//测试if (isIntersecting) {el.src = binding.value;}
});

此时的效果如下图所示:
在这里插入图片描述
从上图可以看到,往上滚动,监听过的图片会重复监听,这是我们不想要的,会造成性能浪费。

解决思路:在监听的图片第一次完成加载后就停止监听。可以利用useIntersectionObserver函数提供的stop方法,修改后的代码如下:

app.directive("lazy", {mounted(el, binding) {const { stop } = useIntersectionObserver(el, ([{ isIntersecting }]) => {console.log(isIntersecting);if (isIntersecting) {el.src = binding.value;//在监听的图片第一次完成加载后就停止监听stop();}});},
});

完善后的效果如下,解决了重复监听问题。
在这里插入图片描述
我们还可以设置一个默认图片,当图片还没加载完成时,就显示默认图片。

app.directive("lazy", {mounted(el, binding) {el.src = "@/assets/images/default.png"; // 使用默认图片const { stop } = useIntersectionObserver(el, ([{ isIntersecting }]) => {if (isIntersecting) {el.src = binding.value;//在监听的图片第一次完成加载后就停止监听stop();}});},
});

此时还存在着的一个问题是,当前注册了一个全局的自定义指令,所有的代码逻辑全写在入口文件中,这样会造成代码的臃肿。

解决思路:拆分代码,通过插件的方法把懒加载指令封装为插件,main.js入口文件只需负责注册插件即可。

src下新建directive/index.js文件,专门存放自定义的插件,把代码逻辑进行转移。

// src/directive/index.js
import { useIntersectionObserver } from "@vueuse/core";
// 封装插件
export const lazyPlugin = {install(app) {app.directive("lazy", {mounted(el, binding) {el.src = "@/assets/images/default.png"; const { stop } = useIntersectionObserver(el, ([{ isIntersecting }]) => {if (isIntersecting) {el.src = binding.value;stop();}});},});},
};

然后在main.js中注册插件

import { createApp } from "vue";
import App from "./App.vue";
import { lazyPlugin } from "./directive";const app = createApp(App);
//注册插件
app.use(lazyPlugin);
app.mount("#app");

定义插件可以参考Vue官网。通常一个 Vue3 的插件会暴露 install 函数,当 app 实例 use 该插件时,就会执行该函数。然后在 install 函数内部,通过 app.directive 去注册一个全局指令,这样就可以在组件中使用它们了。

现在的效果就和一开始介绍的效果一致了。
在这里插入图片描述

2.2 使用IntersectionObserver

其实查看vue3-lazy源码和useIntersectionObserver源码,会发现,它们使用的就是原生IntersectionObserver api。那么接下来我们也可以使用这个api来实现一个自定义的懒加载指令。

  • MDN:IntersectionObserver 提供了一种异步观察目标元素与其祖先元素或顶级文档视口交叉状态的方法。当它被创建时,其被配置为监听根中一段给定比例的可见区域。当其监听到目标元素的可见部分(的比例)超过了一个或多个阈值(threshold)时,会执行指定的回调函数。

简单来说就是IntersectionObserver可以来判断图片是否进入可视区

它对应的回调函数的参数 entries,是 IntersectionObserverEntry 对象数组。当观测的元素可见比例超过指定阈值时,就会执行该回调函数(默认阈值为 0,表示目标元素刚进入根元素可见范围时触发回调函数),对 entries 进行遍历,拿到每一个 entry,然后判断 entry.isIntersecting 是否为 true,如果是则说明 entry 对象对应的 DOM 元素进入了可视区。

具体代码如下:

// src/directive/index.js
import defaultImg from "@/assets/images/default.png";
//定义一个数组用来存储尚未加载的图片
let imgsList = [];//加载图片
function loadingImg(imgDOM) {//获得图片的srclet imgSrc = imgsList.filter((item) => item.el === imgDOM)[0].src;//新建Image对象实例来代替当前图片的加载,图片加载完毕就会触发onload事件,替换img元素的src属性const img = new Image();img.src = imgSrc; img.onload = function () {// 当图片加载完成之后 替换img元素的src属性imgDOM.src = imgSrc;};//将已加载好的图片从数组中删除imgsList = imgsList.filter((item) => item.el !== imgDOM);
}const io = new IntersectionObserver((entries) => {entries.forEach((item) => {// isIntersecting属性判断目标元素当前是否可见if (item.isIntersecting) {//加载图片,加载完后停止监听loadingImg(item.target);io.unobserve(item.target); }});
});export const lazyPlugin = {install(app) {app.directive("lazy", {mounted(el, binding) {el.src = defaultImg; // 使用默认图片io.observe(el); //监听图片imgsList.push({ el: el, src: binding.value }); //数组中加入当前图片},beforeUnmount(el) {//某个img元素解绑时,停止监听,从数组中删除io.unobserve(el);imgsList = imgsList.filter((item) => item.el !== el);},});},
};

主要思路就是:定义一个数组用来存储尚未加载的图片,observe方法对每个图片进行监听,如果当前图片在可视区域,就加载图片,并且从数组中删除图片,然后unobserve停止监听。

最后依然需要在main.js中注册插件,即可使用v-lazy自定义指令。

参考资料:
https://www.npmjs.com/package/vue3-lazyload
https://cn.vuejs.org/guide/reusability/custom-directives.html
https://cn.vuejs.org/guide/reusability/plugins.html
https://www.vueusejs.com/core/useIntersectionObserver/
https://developer.mozilla.org/zh-CN/docs/Web/API/IntersectionObserver

好了,以上就是本文的全部内容,如有问题,欢迎指出!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/92357.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

CentOS 7 下 Keepalived + Nginx 实现双机高可用

CentOS 7 下 Keepalived Nginx 实现双机高可用 文章目录 CentOS 7 下 Keepalived Nginx 实现双机高可用服务器准备服务信息服务架构 服务安装nginxKeepalived 服务配置nginxKeepalived 启动服务nginxkeepalived 服务验证查看 VIP 状态CURL 命令访问浏览器访问 高可用验证停止…

(7)(7.1) 使用航点和事件规划任务

文章目录 前言 7.1.1 设置Home位置 7.1.2 视频&#xff1a;制作并保存多路点任务 7.1.3 视频&#xff1a;加载已保存的多航点任务 7.1.4 使用说明 7.1.5 提示 7.1.6 自动网格 7.1.7 任务指令 7.1.8 任务结束 7.1.9 任务重置 7.1.10 MIS_OPTIONS 7.1.11 任务再出发 …

时序预测 | MATLAB实现基于CNN卷积神经网络的时间序列预测-递归预测未来(多指标评价)

时序预测 | MATLAB实现基于CNN卷积神经网络的时间序列预测-递归预测未来(多指标评价) 目录 时序预测 | MATLAB实现基于CNN卷积神经网络的时间序列预测-递归预测未来(多指标评价)预测结果基本介绍程序设计参考资料 预测结果 基本介绍 1.Matlab实现CNN卷积神经网络时间序列预测未…

docker 学习--03 环境安装(本人使用的win10 Linux也是在win10下模拟)

docker 学习–03 环境安装&#xff08;本人使用的win10 Linux也是在win10下模拟&#xff09; docker 学习-- 01 基础知识 docker 学习-- 02 常用命令 文章目录 docker 学习--03 环境安装&#xff08;本人使用的win10 Linux也是在win10下模拟&#xff09;[TOC](文章目录) 1. wi…

深入源码分析kubernetes informer机制(二)Reflector

[阅读指南] 这是该系列第二篇 基于kubernetes 1.27 stage版本 为了方便阅读&#xff0c;后续所有代码均省略了错误处理及与关注逻辑无关的部分。 文章目录 Reflector是什么整体结构工作流程list拉取数据缓存resync操作watch监听操作 总结 Reflector是什么 reflector在informer…

如何实现安全上网

l 场景描述 政府、军工、科研等涉密单位或企业往往要比其他组织更早接触高精尖的技术与产品&#xff0c;相对应的数据保密性要求更高。常规的内外网物理隔离手段&#xff0c;已经满足不了这些涉密单位的保密需求&#xff0c;发展到现在&#xff0c;需求已经演变成既要保证网络…

(十六)大数据实战——安装使用mysql版的hive服务

前言 hive默认使用的是内嵌据库derby&#xff0c;Derby 是一个嵌入式数据库&#xff0c;可以轻松地以库的形式集成到应用程序中。它不需要独立的服务器进程&#xff0c;所有的数据存储在应用程序所在的文件系统中。为了支持hive服务更方便的使用&#xff0c;我们使用mysql数据…

【数据结构】“栈”的模拟实现

&#x1f490; &#x1f338; &#x1f337; &#x1f340; &#x1f339; &#x1f33b; &#x1f33a; &#x1f341; &#x1f343; &#x1f342; &#x1f33f; &#x1f344;&#x1f35d; &#x1f35b; &#x1f364; &#x1f4c3;个人主页 &#xff1a;阿然成长日记 …

MySQL 索引 详解(保姆级教程)

一、索引概述 索引是帮助 MySQL 高效获取数据的数据结构&#xff08;有序&#xff09;。在数据之外&#xff0c;数据库系统还维护着满足特定查找算法的数据结构&#xff0c;这些数据结构以某种方式引用&#xff08;指向&#xff09;数据&#xff0c;这样就可以在这些数据结构上…

本地oracle登录账号锁定处理,the account is locked

1.打开cmd命令窗口 2.打开sqlplus: sqlplus /nolog(加/nolog是不登录服务器的意思&#xff0c;不加就需要输账号密码) 3.切换到管理员&#xff1a;conn / as sysdba; 第2步第3步可以合并&#xff0c;直接使用sysdba登录&#xff1a;sqlplus / as sysdba; 4.解锁账号&#x…

Python web实战之Django 的 WebSocket 支持详解

关键词&#xff1a;Python, Django, WebSocket, Web 如何使用 Django 实现 WebSocket 功能&#xff1f;本文将详细介绍 WebSocket 的概念、Django 的 WebSocket 支持以及如何利用它来创建动态、响应式的 Web 应用。 1. WebSocket 简介 1.1 什么是 WebSocket&#xff1f; 在 W…

阿里云FRP内网穿透挂载多台服务器

1. FRP介绍 FRP (Fast Reverse Proxy) 是比较流行的一款。FRP 是一个免费开源的用于内网穿透的反向代理应用&#xff0c;它支持 TCP、UDP 协议&#xff0c; 也为 http 和 https 协议提供了额外的支持。你可以粗略理解它是一个中转站&#xff0c; 帮你实现 公网 ←→ FRP(服务器…

OpenCV importerror:dll load failed

从预编译的二进制文件安装OpenCV&#xff0c;从github下载opencv-4.8.0-windows.exe 编译好的文件。按照官方文档拖入cv2.pyd文件。 https://docs.opencv.org/4.8.0/d5/de5/tutorial_py_setup_in_windows.html 使用pycharm运行时&#xff0c;出现报错&#xff0c;importerror…

ITK-SNAP医学影像处理软件无法打开问题

安装ITK-SNAP后成功打开了一次之后再次打开就一直显示无法打开的提示, 在检查全路径为英文和其他版本的ITK-SNAP仍然无法打开&#xff1b; 解决办法&#xff1a; 根据报错的提示的路径&#xff0c;找到UserPreferences.xml 文件&#xff0c;将xml文件删掉&#xff0c;然后就可以…

好用的networkx绘图包

1. NetworkX简介 NetworkX 是一个用于创建、操作和研究复杂网络的 Python 库。它可以创建、分析和可视化各种类型的网络(包括有向图和无向图)&#xff0c;例如社交网络、Web图、生物网络等。 NetworkX 提供了许多图的算法和分析工具&#xff0c;比如节点的度、网络的直径、最短…

【linux】2 软件管理器yum和编辑器vim

目录 1. linux软件包管理器yum 1.1 什么是软件包 1.2 关于rzsz 1.3 注意事项 1.4 查看软件包 1.5 如何安装、卸载软件 1.6 centos 7设置成国内yum源 2. linux开发工具-Linux编辑器-vim使用 2.1 vim的基本概念 2.2 vim的基本操作 2.3 vim正常模式命令集 2.4 vim末行…

GT Code - 图译算法编辑器(集成QT、C++、C、Linux、Git、java、web、go、高并发、服务器、分布式、网络编程、云计算、大数据项目)

目录 项目概述 发文意义 项目介绍 功能分析 设计概要 功能展示 项目文档 项目概述 “GT Code 图译算法编辑器”是一款跨平台、轻量级的代码编辑器&#xff0c;主要面向软件开发人员&#xff0c;它实现了编辑、编译、绘制代码流程图、生成调试演示动画等功能&#xff0c;以…

透过源码理解Flutter中widget、state和element的关系

1、framework源码组成 Flutter中widget、state、element的源码位于framework.dart中&#xff0c;整个文件6693行(版本Flutter 3.12.0-14.0.pre.28)。整个代码可划分为若干部分&#xff0c;主要包括key、widget、state、element四部分。 1.1 key 关于key的代码65行到272行&am…

如何让CSDN学习成就个人能力六边形全是100分:解析个人能力雷达图的窍门

&#x1f337;&#x1f341; 博主猫头虎 带您 Go to New World.✨&#x1f341; &#x1f984; 博客首页——猫头虎的博客&#x1f390; &#x1f433;《面试题大全专栏》 文章图文并茂&#x1f995;生动形象&#x1f996;简单易学&#xff01;欢迎大家来踩踩~&#x1f33a; &a…

142. 环形链表 II

题目描述 给定一个链表的头节点 head &#xff0c;返回链表开始入环的第一个节点。 如果链表无环&#xff0c;则返回 null。 如果链表中有某个节点&#xff0c;可以通过连续跟踪 next 指针再次到达&#xff0c;则链表中存在环。 为了表示给定链表中的环&#xff0c;评测系统内…