如何在 Vue3 中优雅地防止按钮重复点击

前言

在前端开发中,按钮重复点击问题一直是个让人头疼的问题。在请求未完成时,用户可能因为网络延迟等原因多次点击按钮,导致请求被多次发送。

这不仅浪费了服务器资源,还可能引发业务逻辑错误,尤其是在涉及数据提交或入库操作时。

本文将介绍如何在 Vue3 中优雅地解决这个问题。

业务背景

在实际的业务场景中,点击按钮后通常会触发一个网络请求。如果请求尚未完成,用户可能再次点击按钮,从而触发新的请求。

这种情况可能导致接口被多次调用,轻则浪费服务器资源,重则导致业务逻辑错误,比如重复提交表单或重复入库。

传统的解决方案包括使用防抖函数,但这种方法并不能完全解决问题,因为接口响应时间可能超过防抖时间,用户仍然可能在防抖时间后再次点击按钮。

另一种常见的方法是通过在按钮上添加 loading 状态来禁止重复点击,但这种方法需要在每个使用按钮的页面中单独维护 loading 变量,导致代码冗余。

Vue3 的解决方案

在 Vue3 中,我们可以利用组合式 API 来创建自定义 Hook,从而优雅地解决按钮重复点击问题。

创建自定义 Hook

首先,我们创建一个自定义 Hook useAsyncButton,用于管理按钮的 loading 状态和请求逻辑。

ini 复制代码
// useAsyncButton.js
import { ref, useCallback } from 'vue';

export function useAsyncButton(requestFn, options = {}) {
  const loading = ref(false);

  const run = useCallback(async (...args) => {
    if (loading.value) return;

    try {
      loading.value = true;
      const data = await requestFn(...args);
      options.onSuccess?.(data);
      return data;
    } catch (error) {
      options.onError?.(error);
      throw error;
    } finally {
      loading.value = false;
    }
  }, [loading, requestFn, options]);

  return {
    loading,
    run
  };
}

在组件中使用

在 Vue3 的组合式 API 中使用这个自定义 Hook。

xml 复制代码
// MyButton.vue
<template>
  <button
    @click="handleClick"
    :disabled="loading"
  >
    {{ loading ? '加载中...' : '点击请求' }}
  </button>
</template>

<script>
import { useAsyncButton } from './useAsyncButton';

export default {
  setup() {
    const fetchApi = async () => {
      const response = await fetch('your-api-endpoint');
      return await response.json();
    };

    const { loading, run } = useAsyncButton(fetchApi, {
      onSuccess: (data) => {
        console.log('请求成功:', data);
      },
      onError: (error) => {
        console.error('请求失败:', error);
      }
    });

    const handleClick = () => {
      run();
    };

    return {
      loading,
      handleClick
    };
  }
};
</script>

带冷却时间的按钮

为了进一步增强功能,我们可以在自定义 Hook 中添加冷却时间(cooldown)功能。这在某些场景下非常有用,比如发送验证码按钮需要等待一定时间才能再次点击。

ini 复制代码
// useAsyncButton.js
import { ref, useCallback } from 'vue';

export function useAsyncButton(requestFn, options = {}) {
  const loading = ref(false);
  const cooldownRemaining = ref(0);
  const timerRef = ref(null);

  const startCooldown = useCallback(() => {
    if (!options.cooldown) return;

    cooldownRemaining.value = options.cooldown / 1000;
    const startTime = Date.now();

    timerRef.value = setInterval(() => {
      const elapsed = Date.now() - startTime;
      const remaining = Math.ceil((options.cooldown - elapsed) / 1000);

      if (remaining <= 0) {
        clearInterval(timerRef.value);
        cooldownRemaining.value = 0;
      } else {
        cooldownRemaining.value = remaining;
      }
    }, 1000);
  }, [options.cooldown]);

  const run = useCallback(async (...args) => {
    if (loading.value || cooldownRemaining.value > 0) return;

    try {
      loading.value = true;
      const data = await requestFn(...args);
      options.onSuccess?.(data);
      startCooldown();
      return data;
    } catch (error) {
      options.onError?.(error);
      throw error;
    } finally {
      loading.value = false;
    }
  }, [loading, cooldownRemaining, requestFn, options, startCooldown]);

  return {
    loading,
    cooldownRemaining,
    run,
    disabled: loading || cooldownRemaining.value > 0
  };
}

使用示例:

xml 复制代码
// SendCodeButton.vue
<template>
  <button
    @click="handleClick"
    :disabled="disabled"
  >
    {{ loading ? '发送中...' : cooldownRemaining > 0 ? `${cooldownRemaining}秒后重试` : '发送验证码' }}
  </button>
</template>

<script>
import { useAsyncButton } from './useAsyncButton';

export default {
  setup() {
    const sendCode = async () => {
      const response = await fetch('/api/send-code');
      return await response.json();
    };

    const { loading, cooldownRemaining, disabled, run } = useAsyncButton(
      sendCode,
      {
        cooldown: 60000, // 60秒冷却时间
        onSuccess: () => {
          console.log('验证码发送成功');
        },
        onError: (error) => {
          console.error('验证码发送失败', error);
        }
      }
    );

    const handleClick = () => {
      run();
    };

    return {
      loading,
      cooldownRemaining,
      disabled,
      handleClick
    };
  }
};
</script>

总结

通过使用 Vue3 的组合式 API 创建自定义 Hook,我们能够优雅地解决按钮重复点击问题。这种方式不仅能够统一管理请求状态,还能通过简单的配置实现冷却时间功能,提升用户体验。

希望这篇文章能帮助你在项目中实现更高效的按钮点击管理。

相关推荐
anyup5 分钟前
AI 也救不了的前端坑,你遇到过吗?社区、AI、源码三重排查!
前端·数据可视化·cursor
tager12 分钟前
还在为跨框架的微信表情包烦恼?我写了个通用的,拿去吧!🚀
前端·vue.js·react.js
陈随易17 分钟前
一段时间没写文章了,花了10天放了个屁
前端·后端·程序员
Codebee23 分钟前
OneCode基础组件介绍——树形组件(Tree)
前端·编程语言
Cheishire_Cat23 分钟前
AI Coding宝藏组合:Cursor + Cloudbase-AI-Toolkit 开发游戏实战
前端
audience34 分钟前
uni-app运行环境版本和编译器版本不一致的问题
前端
零者36 分钟前
深度解析:React Native Android 上“调试JS”按钮失效的背后原因与修复
前端
前端付豪36 分钟前
Google Ads 广告系统排序与实时竞价架构揭秘
前端·后端·架构
邢行行36 分钟前
NPM 核心知识点:一份清晰易懂的复习指南
前端
颜漠笑年36 分钟前
看看DeepSeek是如何实现前端日历组件的?
前端·html·代码规范