告别传统的防抖机制,提交按钮的新时代来临
目录
背景
目标
核心代码
样式定义:让图标居中、响应父级颜色
SVG 图标:轻量、无依赖的 loading 图标
指令注册:全局注册 v-bLoading
DOM 操作:添加与清除 loading 图标
1. 添加 loading 图标
2. 清除 loading 图标
动画控制:实现 loading 图标旋转
完整代码
main.js里全局引入
使用案例
结语
背景
在现代 Web 开发中,用户体验(UX)是至关重要的。当用户点击一个提交按钮或执行某个异步操作时,如果没有明确的反馈机制,很容易造成重复点击、数据冲突等问题。
为了解决这个问题,我们常常会使用 loading 加载状态 来提示用户“正在处理”,并同时禁用按钮防止多次触发。Vue 提供了强大的自定义指令功能,我们可以借助它来封装一个通用的 v-bLoading
指令,实现优雅的加载交互体验。
本文将从背景出发,逐步分析如何通过 Vue 3 的自定义指令机制,结合 DOM 操作和动画控制,实现一个可复用的按钮 loading 功能。
目标
- 当按钮被点击时:
- 显示 loading 图标;
- 禁用按钮;
- 执行传入的异步函数;
- 异步操作完成后:
- 移除 loading 图标;
- 启用按钮;
- 如果原来有图标,恢复原图标。
核心代码
封装js文件,这里我们导入了 Vue 3 中的 App
和 DirectiveBinding
类型,用于类型检查和保证代码的健壮性。以下是代码模块讲解表格
模块 | 功能 |
---|---|
样式部分 | 定义 loading 图标的样式 |
SVG 图标 | 使用内联 SVG 实现 loading 动画图标 |
核心逻辑 | 注册 v-bLoading 指令,绑定点击事件 |
DOM 操作 | 添加/移除 loading 图标,保存/恢复原有图标 |
动画控制 | 使用 requestAnimationFrame 实现旋转动画 |
样式定义:让图标居中、响应父级颜色
const className = `.el-icon {--color: inherit;-webkit-box-align: center;-ms-flex-align: center;align-items: center;display: -webkit-inline-box;display: -ms-inline-flexbox;display: inline-flex;height: 1em;width: 1em;-webkit-box-pack: center;-ms-flex-pack: center;justify-content: center;line-height: 1em;position: relative;fill: currentColor;color: var(--color);font-size: inherit;
}`
SVG 图标:轻量、无依赖的 loading 图标
const i = `<i class="${className}" id="loading"><svg t="1745215287730" class="icon" viewBox="0 0 1024 1024" version="1.1"xmlns="http://www.w3.org/2000/svg" p-id="2663" width="200" height="200"><!-- path 数据省略 --></svg></i>
`
指令注册:全局注册 v-bLoading
export function bLoading(app: App<Element>) {app.directive('bLoading', {mounted(el: HTMLElement, binding: DirectiveBinding) {if (typeof binding.value !== 'function') {throw new Error('Directive value must be a function')}el.addEventListener('click', () => {addNode(el)setTimeout(() => {binding.value(() => {cleanNode(el)})}, 0)})}})
}
mounted
生命周期钩子用于绑定点击事件。binding.value
必须是一个函数,该函数接收一个回调参数done
。- 在点击按钮后,先添加 loading 图标,然后执行传入的异步函数。
- 函数执行完毕后调用
done
清除 loading。
DOM 操作:添加与清除 loading 图标
1. 添加 loading 图标
function addNode(el: HTMLElement): void {if (el.firstElementChild && el.firstElementChild.tagName === 'I') {tag = el.firstElementChildel.removeChild(el.firstElementChild)}el.insertAdjacentHTML('afterbegin', i)el.setAttribute('disabled', 'true')rotate('loading')
}
- 判断是否已有图标,有的话先保存起来;
- 插入新的 loading 图标;
- 设置按钮为禁用状态;
- 触发动画函数
rotate
。
2. 清除 loading 图标
function cleanNode(el: HTMLElement): void {el.removeAttribute('disabled')if (el.firstElementChild?.id === 'loading') {el.removeChild(el.firstElementChild)}if (tag) {el.prepend(tag)tag = null}
}
- 移除禁用;
- 删除当前的 loading 图标;
- 如果之前有图标,则恢复回去。
动画控制:实现 loading 图标旋转
function rotate(id: string): void {const element = document.getElementById(id)if (!element) returnlet angle = 0const speed = 2 // 每帧旋转角度function animate() {angle = (angle + speed) % 360element.style.transform = `rotate(${angle}deg)`requestAnimationFrame(animate)}animate()
}
使用 requestAnimationFrame
实现平滑的旋转动画,避免卡顿或性能问题。
完整代码
import type { App, DirectiveBinding } from 'vue'// 全局 loading 图标 SVG 字符串
const className = `.el-icon {--color: inherit;-webkit-box-align: center;-ms-flex-align: center;align-items: center;display: -webkit-inline-box;display: -ms-inline-flexbox;display: inline-flex;height: 1em;width: 1em;-webkit-box-pack: center;-ms-flex-pack: center;justify-content: center;line-height: 1em;position: relative;fill: currentColor;color: var(--color);font-size: inherit;
}`const i = `<i class="${className}" id="loading"><svg t="1745215287730" class="icon" viewBox="0 0 1024 1024" version="1.1"xmlns="http://www.w3.org/2000/svg" p-id="2663" width="200" height="200"><path d="M834.7648 736.3584a5.632 5.632 0 1 0 11.264 0 5.632 5.632 0 0 0-11.264 0z m-124.16 128.1024a11.1616 11.1616 0 1 0 22.3744 0 11.1616 11.1616 0 0 0-22.3744 0z m-167.3216 65.8944a16.7936 16.7936 0 1 0 33.6384 0 16.7936 16.7936 0 0 0-33.6384 0zM363.1616 921.6a22.3744 22.3744 0 1 0 44.7488 0 22.3744 22.3744 0 0 0-44.7488 0z m-159.744-82.0224a28.0064 28.0064 0 1 0 55.9616 0 28.0064 28.0064 0 0 0-56.0128 0zM92.672 700.16a33.6384 33.6384 0 1 0 67.2256 0 33.6384 33.6384 0 0 0-67.2256 0zM51.2 528.9984a39.168 39.168 0 1 0 78.336 0 39.168 39.168 0 0 0-78.336 0z m34.1504-170.0864a44.8 44.8 0 1 0 89.6 0 44.8 44.8 0 0 0-89.6 0zM187.904 221.7984a50.432 50.432 0 1 0 100.864 0 50.432 50.432 0 0 0-100.864 0zM338.432 143.36a55.9616 55.9616 0 1 0 111.9232 0 55.9616 55.9616 0 0 0-111.9744 0z m169.0112-4.9152a61.5936 61.5936 0 1 0 123.2384 0 61.5936 61.5936 0 0 0-123.2384 0z m154.7776 69.632a67.1744 67.1744 0 1 0 134.3488 0 67.1744 67.1744 0 0 0-134.3488 0z m110.0288 130.816a72.8064 72.8064 0 1 0 145.5616 0 72.8064 72.8064 0 0 0-145.5616 0z m43.7248 169.472a78.3872 78.3872 0 1 0 156.8256 0 78.3872 78.3872 0 0 0-156.8256 0z"fill="" p-id="2664"></path></svg></i>
`let tag: Element | null = null/*** 注册一个全局自定义指令 v-bLoading* @param app Vue 应用实例*/
export function bLoading(app: App<Element>) {app.directive('bLoading', {mounted(el: HTMLElement, binding: DirectiveBinding) {if (typeof binding.value !== 'function') {throw new Error('Directive value must be a function')}el.addEventListener('click', () => {addNode(el)setTimeout(() => {binding.value(() => {cleanNode(el)})}, 0)})}})
}/*** 添加 loading 图标到按钮中* @param el 按钮元素*/
function addNode(el: HTMLElement): void {if (el.firstElementChild && el.firstElementChild.tagName === 'I') {// 如果已经有图标,先保存旧图标tag = el.firstElementChildel.removeChild(el.firstElementChild)}el.insertAdjacentHTML('afterbegin', i)el.setAttribute('disabled', 'true')rotate('loading')
}/*** 移除 loading 图标,并恢复原有图标(如果存在)* @param el 按钮元素*/
function cleanNode(el: HTMLElement): void {el.removeAttribute('disabled')if (el.firstElementChild?.id === 'loading') {el.removeChild(el.firstElementChild)}if (tag) {el.prepend(tag)tag = null}
}/*** 实现 loading 图标的旋转动画* @param id loading 图标元素的 ID*/
function rotate(id: string): void {const element = document.getElementById(id)if (!element) returnlet angle = 0const speed = 2 // 每帧旋转角度function animate() {angle = (angle + speed) % 360element.style.transform = `rotate(${angle}deg)`requestAnimationFrame(animate)}animate()
}
main.js里全局引入
import { createApp } from 'vue'import App from './App.vue'import { bLoading } from './utils/loading'
const app = createApp(App)
bLoading(app)
使用案例
<button type="primary" v-bLoading="(next) => handleSubmit(next)">疯狂点击</button>function handleSubmit(next){ setTimeout(()=>{ next()
},3000)}
结语
通过这篇文章,我们学习了如何使用 Vue 3 的自定义指令机制,结合 DOM 操作和动画控制,实现了一个实用的按钮 loading 指令 v-bLoading
。该指令具有以下优点:
- 🧩 模块化结构清晰;
- 🎨 样式可定制;
- ⚙️ 支持异步操作;
- 🔄 可恢复原始图标;
- 🐞 易于调试和扩展。
相关文章:
告别传统的防抖机制,提交按钮的新时代来临
目录 背景 目标 核心代码 样式定义:让图标居中、响应父级颜色 SVG 图标:轻量、无依赖的 loading 图标 指令注册:全局注册 v-bLoading DOM 操作:添加与清除 loading 图标 1. 添加 loading 图标 2. 清除 loading 图标 动画…...

InternVL3: 利用AI处理文本、图像、视频、OCR和数据分析
InternVL3推动了视觉-语言理解、推理和感知的边界。 在其前身InternVL 2.5的基础上,这个新版本引入了工具使用、GUI代理操作、3D视觉和工业图像分析方面的突破性能力。 让我们来分析一下是什么让InternVL3成为游戏规则的改变者 — 以及今天你如何开始尝试使用它。 InternVL…...
关于 Web安全:1. Web 安全基础知识
一、HTTP/HTTPS 协议详解 1. HTTP协议基础 什么是 HTTP? HTTP(HyperText Transfer Protocol)是互联网中浏览器和服务器之间传输数据的协议,基于请求-响应模式。它是一个无状态协议,意思是每次请求都是独立的&#x…...
西门子 S1500 PLC 通过 Profinet 对 6 台施耐德 ATV304 变频器的控制,用于 6 台升降台的位置控制。
西门子 S1500 PLC 通过 Profinet 对 6 台施耐德 ATV304 变频器的控制,用于 6 台升降台的位置控制。程序主要特点: 模块化设计:采用功能块数组结构,实现对多台设备的统一控制循环控制:使用 FOR 循环遍历每台升降台&…...

重构金融数智化产业版图:中电金信“链主”之道
近日,《商学院》杂志独家专访了中电金信常务副总经理(主持经营工作)冯明刚,围绕“金融科技”“数字底座”“架构转型”“AI驱动”等议题,展开了一场关于未来架构、技术变革与系统创新的深入对话。 当下,数字…...
高光谱遥感图像处理之数据分类的fcm算法
基于模糊C均值聚类(FCM)的高光谱遥感图像分类MATLAB实现示例 %% FCM高光谱图像分类示例 clc; clear; close all;%% 数据加载与预处理 % 加载示例数据(此处使用公开数据集Indian Pines的简化版) load(indian_pines.mat); % 包含变…...

2025年PMP 学习十六 第11章 项目风险管理 (总章)
2025年PMP 学习十六 第11章 项目风险管理 (总章) 第11章 项目风险管理 序号过程过程组1规划风险管理规划2识别风险规划3实施定性风险分析规划4实施定量风险分析规划5规划风险应对执行6实施风险应对执行7监控风险监控 目标: 提高项目中积极事件的概率和…...
IEEE 列表会议第五届机器人、自动化与智能控制国际会议
会议地点:中国 成都 会议官网:ICRAIC 主办单位:成都理工大学 协办单位:成都大学 早鸟截稿:2025年7月15日 截稿时间:2025年8月20日 出版信息:IEEE出版&EI数据库 会议时间:…...
基于 React Hook 封装 Store 的三种方案
基于 React Hook 封装 Store 的三种方案 方案一:基于 useSyncExternalStore 的轻量级 Store(推荐) import { useSyncExternalStore } from react;type Store<T> {state: T;listeners: Set<() > void>; };function createSt…...
Gmsh 读取自定义轮廓并划分网格:深入解析与实践指南
一、Gmsh 简介 (一)Gmsh 是什么 Gmsh 是一款功能强大的开源有限元网格生成器,广泛应用于工程仿真、数值模拟以及计算机图形学等领域。它为用户提供了从几何建模到网格划分的一整套解决方案,能够有效处理复杂几何形状,生成高质量的二维和三维网格,满足多种数值方法的需求…...

bili.png
import pygame as pg import sys import time import randompg.init() screen pg.display.set_mode((800,500)) pg.display.set_caption(runcool) screen.fill((135, 206, 235)) bili pg.image.load(bili.png)#得分 coin 0 game_font pg.font.Font(None, 50)#人物大小…...

【设计模式】- 行为型模式1
模板方法模式 定义了一个操作中的算法骨架,将算法的一些步骤推迟到子类,使得子类可以不改变该算法结构的情况下重定义该算法的某些步骤 【主要角色】: 抽象类:给出一个算法的轮廓和骨架(包括一个模板方法 和 若干基…...
GMT之Bash语言使用
GMT的操作有自己的逻辑和“命令”,但GMT是可以用Bash语言控制的,所以常常以.sh为后缀写GMT程序。 GMT程序运行步骤如下: 采用cd ,定位到指定文件夹;以sh ***.sh运行GMT,得到结果。 另外,遇到…...

AI神经网络降噪算法在语音通话产品中的应用优势与前景分析
采用AI降噪的语言通话环境抑制模组性能效果测试 一、引言 随着人工智能技术的快速发展,AI神经网络降噪算法在语音通话产品中的应用正逐步取代传统降噪技术,成为提升语音质量的关键解决方案。相比传统DSP(数字信号处理)降噪&#…...
ISBI 2012 EM 神经元结构分割数据集复现UNet
一些笔记在代码的注释中 因为使用的数据集比较简单,所以没有使用模型可视化和调试的内容,只是简单的数据集预处理和模型的搭建以及训练。 # 1. PyTorch 基础模块 import torch # 张量操作 import torch.nn as nn # 构建神经网…...
Java视频流RTMP/RTSP协议解析与实战代码
在Java中实现视频直播的输入流处理,通常需要结合网络编程、多媒体处理库以及流媒体协议(如RTMP、HLS、RTSP等)。以下是实现视频直播输入流的关键步骤和技术要点: 1. 视频直播输入流的核心组件 网络输入流:通过Socket或…...

springboot连接高斯数据库(GaussDB)踩坑指南
1. 用户密码加密类型与gsjdbc4版本不兼容问题 我的数据库,设置的加密类型(password_encryption_type)是2, 直接使用gsjdbc4.jar连接数据库报错。 org.postgresql.util.PSQLException: Invalid or unsupported by client SCRAM mechanisms 后使用gsjdb…...

c++20引入的三路比较操作符<=>
目录 一、简介 二、三向比较的返回类型 2.1 std::strong_ordering 2.2 std::weak_ordering 2.3 std::partial_ordering 三、对基础类型的支持 四、自动生成的比较运算符函数 4.1 std::rel_ops的作用 4.2 使用<> 五、兼容他旧代码 一、简介 c20引入了三路比较操…...

Cursor开发酒店管理系统
目录: 1、后端代码初始化2、使用Cursor打开spingboot项目3、前端代码初始化4、切换其他大模型5、Curosr无限续杯 1、后端代码初始化 找一个目录,使用idea在这个目录下新建springboot的项目。 2、使用Cursor打开spingboot项目 在根目录下新建.cursor文件…...
nosqlbooster pojie NoSQLBooster for MongoDB
测过可用,注意 asar的安装使用报错改用 npx asar extract app.asar app 路径 C:\Users{computerName}\AppData\Local\Programs\nosqlbooster4mongo\resources npm install asar -g asar extract app.asar app 打开shared\lmCore.js 修改MAX_TRIAL_DAYS3000 修改…...
基于 Flink 的实时推荐系统:从协同过滤到多模态语义理解
基于 Flink 的实时推荐系统:从协同过滤到多模态语义理解 嘿,各位技术小伙伴们!在这个信息爆炸的时代,你是不是常常惊叹于各大平台仿佛能 “读懂你的心”,精准推送你感兴趣的内容呢?今天,小编就…...
【HBase整合Hive】HBase-1.4.8整合Hive-2.3.3过程
HBase-1.4.8整合Hive-2.3.3过程 一、摘要二、整合过程三、注意事项 一、摘要 HBase集成Hive,由Hive来编写SQL语句操作HBase有以下好处: 简化操作:Hive提供了类SQL的查询语言HiveQL,对于熟悉SQL的用户来说,无需学习HBas…...

图像对比度调整(局域拉普拉斯滤波)
一、背景介绍 之前刷对比度相关调整算法,找到效果不错,使用局域拉普拉斯做图像对比度调整,尝试复现和整理了下相关代码。 二、实现流程 1、基本原理 对输入图像进行高斯金字塔拆分,对每层的每个像素都针对性处理,生产…...

如何在本地打包 StarRocks 发行版
字数 615,阅读大约需 4 分钟 最近我们在使用 StarRocks 的时候碰到了一些小问题: • 重启物化视图的时候会导致视图全量刷新,大量消耗资源。- 修复 PR:https://github.com/StarRocks/starrocks/pull/57371• excluded_refresh_tab…...

git使用的DLL错误
安装好git windows客户端打开git bash提示 Error: Could not fork child process: Resource temporarily unavailable (-1). DLL rebasing may be required; see ‘rebaseall / rebase –help’. 提示 MINGW64的DLL链接有问题,其实是Windows的安全中心限制了&…...
Elasticsearch倒排索引核心原理面试题
倒排索引核心原理面试题 🚀 目录 基础概念性能优化应用场景数据结构设计问题排查扩展思考基础概念 🔍 面试题1:基础概念 题目:Elasticsearch/Lucene的倒排索引(Inverted Index)是如何工作的?请描述从关键词搜索到返回文档的完整流程。 👉 查看参考答案 倒排索引…...

区块链blog1__合作与信任
🍂我们的世界 🌿不是孤立的,而是网络化的 如果是单独孤立的系统,无需共识,而我们的社会是网络结构,即结点间不是孤立的 🌿网络化的原因 而目前并未发现这样的理想孤立系统,即现实中…...

从数据包到可靠性:UDP/TCP协议的工作原理分析
之前我们已经使用udp/tcp的相关接口写了一些简单的客户端与服务端代码。也了解了协议是什么,包括自定义协议和知名协议比如http/https和ssh等。现在我们再回到传输层,对udp和tcp这两传输层巨头协议做更深一步的分析。 一.UDP UDP相关内容很简单…...

【CanMV K230】AI_CUBE1.4
《k230-AI 最近小伙伴有做模型的需求。所以我重新捡起来了。正好把之前没测过的测一下。 这次我们用的是全新版本。AICUBE1.4.dotnet环境9.0 注意AICUBE训练模型对硬件有所要求。最好使用独立显卡。 有小伙伴说集显也可以。emmmm可以试试哈 集显显存2G很勉强了。 我们依然用…...

vscode 默认环境路径
目录 1.下面放在项目根目录上: 2.settings.json内容: 自定义conda环境断点调试 启动默认参数: 1.下面放在项目根目录上: .vscode/settings.json 2.settings.json内容: {"python.analysis.extraPaths"…...