news 2026/1/14 11:35:04

50天50个小项目 (React19 + Tailwindcss V4) ✨ | ButtonRippleEffect(按钮涟漪效果)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
50天50个小项目 (React19 + Tailwindcss V4) ✨ | ButtonRippleEffect(按钮涟漪效果)

📅 我们继续 50 个小项目挑战!—— ButtonRippleEffect 组件

仓库地址:https://gitee.com/hhm-hhm/50days50projects.git

构建一个带有点击波纹动画效果的按钮组件。这个按钮在点击时会从点击位置生成一个扩散的“涟漪”动画,增强用户交互体验。

🌀 组件目标

  • 创建一个具有点击波纹动画的按钮
  • 波纹动画从点击位置开始扩散
  • 支持多个连续点击的波纹动画
  • 使用 TailwindCSS快速构建现代 UI 界面

🔧 ButtonRippleEffect.tsx组件实现

import React, { useState, useRef } from 'react' const ButtonRippleEffect: React.FC = () => { const [ripples, setRipples] = useState<{ x: number; y: number; size: number }[]>([]) const buttonRef = useRef<HTMLButtonElement>(null) const addRipple = (event: React.MouseEvent) => { const button = buttonRef.current if (!button) return const rect = button.getBoundingClientRect() const x = event.clientX - rect.left const y = event.clientY - rect.top const size = Math.max(button.offsetWidth, button.offsetHeight) setRipples((prevState) => [...prevState, { x, y, size }]) // 600ms 后移除波纹效果,与动画持续时间匹配 setTimeout(() => { setRipples((prevState) => prevState.slice(1)) }, 600) } return ( <div className="flex h-screen items-center justify-center bg-gray-900"> <button ref={buttonRef} className="relative h-20 w-48 cursor-pointer overflow-hidden rounded-lg bg-blue-500 font-bold text-white transition-colors hover:bg-blue-600 active:bg-blue-700" onMouseDown={addRipple}> Click Me {ripples.map((ripple, index) => ( <span key={index} style={{ left: ripple.x + 'px', top: ripple.y + 'px', width: ripple.size + 'px', height: ripple.size + 'px', }} className="animate-ripple pointer-events-none absolute rounded-full bg-white/50 opacity-0"></span> ))} </button> <div className="fixed right-20 bottom-5 text-2xl text-red-500">CSDN@Hao_Harrision</div> </div> ) } export default ButtonRippleEffect

🦄样式动画

assets/style.css

... @keyframes ripple { 0% { transform: translate(-50%, -50%) scale(0); opacity: 1; } 100% { transform: translate(-50%, -50%) scale(1.5); /* 根据需要调整scale大小 */ opacity: 0; } } .animate-ripple { animation: ripple 0.6s ease-out; /* 确保这个值与setTimeout中的延迟相匹配 */ }

🔧 转换说明

功能 / 概念Vue 3 (Composition API)React (TS + Hooks)
响应式状态const count = ref(0)const [count, setCount] = useState(0)
事件处理@click="handler"onClick={handler}
模板循环渲染<div v-for="(item, i) in list" :key="i">{list.map((item, i) => <div key={i}>...)}
动态内联样式:style="{ left: x + 'px' }"style={{ left:${x}px}}
动态 class:class="[isActive ? 'active' : '']"className={${isActive ? 'active' : ''}}或使用clsx/classnames
生命周期(挂载)onMounted(() => { ... })useEffect(() => { ... }, [])
副作用清理onUnmounted(() => { ... })useEffect(() => () => { ... }, [])
引用 DOM 元素const el = ref(null)+ref="el"const elRef = useRef<HTMLButtonElement>(null)+ref={elRef}
监听原生事件@mousedown="handler"onMouseDown={handler}
动画类名手动绑定.animate-ripple同样使用className="animate-ripple",配合 CSS
CSS scoped 样式<style scoped>单独.css文件 + 手动引入,或使用 CSS Modules /styled-components

⚠️ 常见陷阱与注意事项

问题解决方案
NodeJS.Timeout报错浏览器中setInterval返回number,应使用useRef<number | null>
setState in useEffect警告首次初始化时调用setState是合理的,可加// eslint-disable-next-line react-hooks/set-state-in-effect忽略
动态定位偏移使用rotate + translateY + rotate(-angle)时,transformOrigin: 'center'是关键;避免直接用translate(x,y)除非加上中心偏移
Tailwind 不生效确保tailwind.config.js正确配置content包含你的组件路径
TypeScript 类型安全明确标注状态类型,如useState<{x: number, y: number}[]>([])

✅ 最佳实践建议

  1. 状态合并:多个相关状态(如日期+时间)可合并为一个格式化字符串,减少重渲染。
  2. 定时器管理:始终用useRef存储setIntervalID,并在useEffect清理函数中清除。
  3. 事件坐标计算:使用getBoundingClientRect()获取按钮位置,再计算点击点相对坐标。
  4. 动画同步setTimeout清除 ripple 的时间必须 ≥ CSS 动画持续时间。
  5. 避免内联回调:若性能敏感,可用useCallback包裹事件处理器(本例中非必需)。

🎨 TailwindCSS 样式重点讲解

🎯 TailwindCSS 样式说明
类名作用
h-screen,items-center,justify-center设置全屏高度并垂直居中布局
bg-gray-900设置深色背景
relative为按钮设置相对定位,方便内部绝对定位的波纹元素
h-20,w-48设置按钮高度和宽度
rounded-lg圆角按钮
bg-blue-500,hover:bg-blue-600,active:bg-blue-700按钮颜色及悬停、激活状态下的变色效果
font-bold,text-white文字加粗和白色字体
transition-colors添加颜色变化的过渡动画
absolute,rounded-full,bg-white/50波纹元素的基本样式
pointer-events-none避免波纹干扰按钮点击事件
opacity-0初始隐藏波纹,由动画控制显示

这些 Tailwind 工具类快速构建了一个视觉丰富、交互性强的按钮组件

🦌 路由组件 + 常量定义

router/index.tsxchildren数组中添加子路由

{ path: '/', element: <App />, children: [ ... { path: '/ButtonRipple', lazy: () => import('@/projects/ButtonRippleEffect.tsx').then((mod) => ({ Component: mod.default, })), }, ], },

constants/index.tsx 添加组件预览常量

import demo20Img from '@/assets/pic-demo/demo-20.png' 省略部分.... export const projectList: ProjectItem[] = [ 省略部分.... { id: 20, title: 'Button Ripple Effect', image: demo20Img, link: 'ButtonRipple', },

🚀 小结

扩展的功能推荐:

  • 支持不同颜色主题的波纹按钮
  • 支持禁用状态下的点击无反应
  • 支持自定义波纹颜色或透明度
  • 多个按钮共享波纹动画逻辑(封装为可复用组件)

📅 明日预告: 我们将完成DragNDrop组件,一个非常有意思的拖拽组件,可以对元素进行重新排列。🚀


原文链接:https://blog.csdn.net/qq_44808710/article/details/149080035

每天造一个轮子,码力暴涨不是梦!🚀

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2025/12/19 11:05:25

Oracle客户端安装图解:小白也能3分钟搞定

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 开发一个交互式Oracle客户端安装引导工具。要求&#xff1a;1. 分步骤图文指导 2. 实时错误检测 3. 一键修复常见问题 4. 视频演示嵌入 5. 安装进度可视化。使用Electron开发跨平台…

作者头像 李华
网站建设 2025/12/19 11:05:15

Keil Assistant终极指南:嵌入式开发效率提升300%的秘诀

Keil Assistant终极指南&#xff1a;嵌入式开发效率提升300%的秘诀 【免费下载链接】keil-assistant 项目地址: https://gitcode.com/gh_mirrors/ke/keil-assistant Keil Assistant是一款专为Visual Studio Code设计的插件&#xff0c;能够无缝集成Keil uVision项目&am…

作者头像 李华
网站建设 2025/12/19 11:05:11

5个理由告诉你为什么NixOS + Hyprland是最佳Linux桌面组合

5个理由告诉你为什么NixOS Hyprland是最佳Linux桌面组合 【免费下载链接】linux-nixos-hyprland-config-dotfiles Linux &#x1f427; configuration based on NixOS ❄️, Hyprland, and Catppuccin Macchiato theme &#x1f638; for a consistent, complete, and customi…

作者头像 李华
网站建设 2026/1/13 3:12:45

Junit5测试框架:Java开发者必备的单元测试工具

Junit5测试框架&#xff1a;Java开发者必备的单元测试工具 【免费下载链接】Junit5.jar包代码测试工具 本项目提供了一个便捷的Junit5.jar包下载资源&#xff0c;专为开发者进行代码测试而设计。Junit5作为JUnit的最新版本&#xff0c;拥有现代化的测试框架&#xff0c;能够帮助…

作者头像 李华
网站建设 2026/1/3 10:50:46

Open-AutoGLM日志分析从入门到精通(一线工程师20年经验总结)

第一章&#xff1a;Open-AutoGLM日志分析概述Open-AutoGLM 是一个面向自动化大语言模型任务的日志追踪与行为分析框架&#xff0c;专为开发者和系统运维人员设计&#xff0c;用于监控、解析和优化基于 GLM 架构的模型运行过程。其核心功能包括结构化日志采集、实时行为追踪、异…

作者头像 李华
网站建设 2026/1/13 7:02:54

为什么顶尖程序员都在用Open-AutoGLM做自动化订餐?真相令人震惊

第一章&#xff1a;Open-AutoGLM与美团自动订餐的融合背景随着人工智能技术在垂直领域的深度渗透&#xff0c;大语言模型&#xff08;LLM&#xff09;正逐步从通用对话系统向特定业务场景演进。Open-AutoGLM 作为一款开源的自动化生成语言模型框架&#xff0c;具备强大的任务理…

作者头像 李华