告别传统的防抖机制,提交按钮的新时代来临

news2025/5/17 20:38:37

目录

背景

目标

核心代码

样式定义:让图标居中、响应父级颜色 

SVG 图标:轻量、无依赖的 loading 图标 

 指令注册:全局注册 v-bLoading

DOM 操作:添加与清除 loading 图标

1. 添加 loading 图标

2. 清除 loading 图标

 动画控制:实现 loading 图标旋转

完整代码 

main.js里全局引入

使用案例

结语


 

背景

在现代 Web 开发中,用户体验(UX)是至关重要的。当用户点击一个提交按钮或执行某个异步操作时,如果没有明确的反馈机制,很容易造成重复点击、数据冲突等问题。

为了解决这个问题,我们常常会使用 loading 加载状态 来提示用户“正在处理”,并同时禁用按钮防止多次触发。Vue 提供了强大的自定义指令功能,我们可以借助它来封装一个通用的 v-bLoading 指令,实现优雅的加载交互体验。

本文将从背景出发,逐步分析如何通过 Vue 3 的自定义指令机制,结合 DOM 操作和动画控制,实现一个可复用的按钮 loading 功能。

目标

  1. 当按钮被点击时:
    • 显示 loading 图标;
    • 禁用按钮;
    • 执行传入的异步函数;
  2. 异步操作完成后:
    • 移除 loading 图标;
    • 启用按钮;
    • 如果原来有图标,恢复原图标。 

核心代码

封装js文件,这里我们导入了 Vue 3 中的 AppDirectiveBinding 类型,用于类型检查和保证代码的健壮性。以下是代码模块讲解表格

模块功能
样式部分定义 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.firstElementChild
    el.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) return

  let angle = 0
  const speed = 2 // 每帧旋转角度

  function animate() {
    angle = (angle + speed) % 360
    element.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.firstElementChild
    el.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) return

  let angle = 0
  const speed = 2 // 每帧旋转角度

  function animate() {
    angle = (angle + speed) % 360
    element.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。该指令具有以下优点:

  • 🧩 模块化结构清晰;
  • 🎨 样式可定制;
  • ⚙️ 支持异步操作;
  • 🔄 可恢复原始图标;
  • 🐞 易于调试和扩展。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

InternVL3: 利用AI处理文本、图像、视频、OCR和数据分析

InternVL3推动了视觉-语言理解、推理和感知的边界。 在其前身InternVL 2.5的基础上,这个新版本引入了工具使用、GUI代理操作、3D视觉和工业图像分析方面的突破性能力。 让我们来分析一下是什么让InternVL3成为游戏规则的改变者 — 以及今天你如何开始尝试使用它。 InternVL…

重构金融数智化产业版图:中电金信“链主”之道

近日&#xff0c;《商学院》杂志独家专访了中电金信常务副总经理&#xff08;主持经营工作&#xff09;冯明刚&#xff0c;围绕“金融科技”“数字底座”“架构转型”“AI驱动”等议题&#xff0c;展开了一场关于未来架构、技术变革与系统创新的深入对话。 当下&#xff0c;数字…

2025年PMP 学习十六 第11章 项目风险管理 (总章)

2025年PMP 学习十六 第11章 项目风险管理 &#xff08;总章&#xff09; 第11章 项目风险管理 序号过程过程组1规划风险管理规划2识别风险规划3实施定性风险分析规划4实施定量风险分析规划5规划风险应对执行6实施风险应对执行7监控风险监控 目标: 提高项目中积极事件的概率和…

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

模板方法模式 定义了一个操作中的算法骨架&#xff0c;将算法的一些步骤推迟到子类&#xff0c;使得子类可以不改变该算法结构的情况下重定义该算法的某些步骤 【主要角色】&#xff1a; 抽象类&#xff1a;给出一个算法的轮廓和骨架&#xff08;包括一个模板方法 和 若干基…

AI神经网络降噪算法在语音通话产品中的应用优势与前景分析

采用AI降噪的语言通话环境抑制模组性能效果测试 一、引言 随着人工智能技术的快速发展&#xff0c;AI神经网络降噪算法在语音通话产品中的应用正逐步取代传统降噪技术&#xff0c;成为提升语音质量的关键解决方案。相比传统DSP&#xff08;数字信号处理&#xff09;降噪&#…

springboot连接高斯数据库(GaussDB)踩坑指南

1. 用户密码加密类型与gsjdbc4版本不兼容问题 我的数据库&#xff0c;设置的加密类型(password_encryption_type)是2&#xff0c; 直接使用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开发酒店管理系统

目录&#xff1a; 1、后端代码初始化2、使用Cursor打开spingboot项目3、前端代码初始化4、切换其他大模型5、Curosr无限续杯 1、后端代码初始化 找一个目录&#xff0c;使用idea在这个目录下新建springboot的项目。 2、使用Cursor打开spingboot项目 在根目录下新建.cursor文件…

图像对比度调整(局域拉普拉斯滤波)

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

如何在本地打包 StarRocks 发行版

字数 615&#xff0c;阅读大约需 4 分钟 最近我们在使用 StarRocks 的时候碰到了一些小问题&#xff1a; • 重启物化视图的时候会导致视图全量刷新&#xff0c;大量消耗资源。- 修复 PR&#xff1a;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链接有问题&#xff0c;其实是Windows的安全中心限制了&…

区块链blog1__合作与信任

&#x1f342;我们的世界 &#x1f33f;不是孤立的&#xff0c;而是网络化的 如果是单独孤立的系统&#xff0c;无需共识&#xff0c;而我们的社会是网络结构&#xff0c;即结点间不是孤立的 &#x1f33f;网络化的原因 而目前并未发现这样的理想孤立系统&#xff0c;即现实中…

从数据包到可靠性:UDP/TCP协议的工作原理分析

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

【CanMV K230】AI_CUBE1.4

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

vscode 默认环境路径

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

支付宝授权登录

支付宝授权登录 一、场景 支付宝小程序登录&#xff0c;获取用户userId 二、注册支付宝开发者账号 1、支付宝开放平台 2、点击右上角–控制台&#xff0c;创建小程序 3、按照步骤完善信息&#xff0c;生成密钥时会用到的工具 4、生成的密钥&#xff0c;要保管好&#xff…

Fabric 服务端插件开发简述与聊天事件监听转发

原文链接&#xff1a;Fabric 服务端插件开发简述与聊天事件监听转发 < Ping通途说 0. 引言 以前写过Spigot的插件&#xff0c;非常简单&#xff0c;仅需调用官方封装好的Event类即可。但Fabric这边在开发时由于官方文档和现有互联网资料来看&#xff0c;可能会具有一定的误…

电商物流管理优化:从网络重构到成本管控的全链路解析

大家好&#xff0c;我是沛哥儿。作为电商行业&#xff0c;我始终认为物流是电商体验的“最后一公里”&#xff0c;更是成本控制的核心战场。随着行业竞争加剧&#xff0c;如何通过物流网络优化实现降本增效&#xff0c;已成为电商企业的必修课。本文将从物流网络的各个环节切入…

Unity:延迟执行函数:Invoke()

目录 Unity 中的 Invoke() 方法详解 什么是 Invoke()&#xff1f; 基本使用方法 使用要点 延伸功能 ❗️Invoke 的局限与注意事项 在Unity中&#xff0c;延迟执行函数是游戏逻辑中常见的需求&#xff0c;比如&#xff1a; 延迟切换场景 延迟播放音效或动画 给玩家时间…