如何在 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,我们能够优雅地解决按钮重复点击问题。这种方式不仅能够统一管理请求状态,还能通过简单的配置实现冷却时间功能,提升用户体验。

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

相关推荐
oioihoii8 分钟前
CRT调试堆检测:从原理到实战的资源泄漏排查指南
开发语言·前端·c++·c
不如吃茶去14 分钟前
开源推荐:LocalSqueeze - 隐私优先的本地图片压缩工具
前端·react.js·electron
anyup14 分钟前
uView Pro 正式开源!70+ Vue3 组件重构全记录,助力 uni-app 组件生态,你会选择吗?
前端·架构·uni-app
一点一木15 分钟前
PromptPilot 与豆包新模型:从图片到视频,解锁 AI 新玩法
前端·人工智能
一只小风华~22 分钟前
BOM Cookie操作详解
开发语言·前端·javascript
xianxin_29 分钟前
HTML5 表单元素
前端
LaoZhangAI30 分钟前
ChatGPT 5发布日期揭秘:2025年8月上线,多模态推理能力全面升级
前端·后端
dingdong8586431 分钟前
前端工程化2
前端
cvpv35 分钟前
我将封装史上最优雅的 Axios
前端