news 2026/6/13 13:22:16

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进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 14:00:08

企业IT运维:批量解决多台电脑的启动设备错误

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 创建一个企业级批量修复工具&#xff0c;能够通过局域网远程诊断和修复多台计算机的Reboot and Select Proper Boot Device错误。功能包括&#xff1a;自动识别网络中的受影响设备&…

作者头像 李华
网站建设 2026/6/12 14:02:39

揭秘多设备协同控制难题:如何用Open-AutoGLM实现10台手机同时操控

第一章&#xff1a;多设备协同控制的技术挑战在现代分布式系统与物联网&#xff08;IoT&#xff09;架构中&#xff0c;多设备协同控制已成为实现智能自动化的核心能力。然而&#xff0c;设备异构性、网络延迟、状态同步与安全通信等问题构成了主要技术障碍。设备异构性与协议兼…

作者头像 李华
网站建设 2026/6/12 20:20:20

如何让Open-AutoGLM在老旧笔记本上稳定运行?9个你必须知道的优化细节

第一章&#xff1a;Open-AutoGLM在低配设备上的运行挑战在资源受限的设备上部署大型语言模型&#xff08;LLM&#xff09;如Open-AutoGLM&#xff0c;面临诸多性能与兼容性难题。尽管该模型具备强大的自动化推理能力&#xff0c;但其高内存占用和计算密集型特性使其难以在低配环…

作者头像 李华
网站建设 2026/6/12 17:40:15

5分钟掌握Libreddit:Reddit替代前端的终极个性化配置指南

5分钟掌握Libreddit&#xff1a;Reddit替代前端的终极个性化配置指南 【免费下载链接】libreddit Private front-end for Reddit 项目地址: https://gitcode.com/gh_mirrors/li/libreddit 您是否厌倦了Reddit官方界面的繁琐操作&#xff1f;想要一个更简洁、更私密的浏览…

作者头像 李华
网站建设 2026/6/13 10:15:02

视觉AI革命:零基础构建工业缺陷检测系统的完整指南

还在为人工质检效率低下而烦恼吗&#xff1f;每天面对成千上万的零件&#xff0c;眼睛疲劳不说&#xff0c;漏检率还居高不下&#xff1f;本文将为你揭秘如何利用AudioGPT项目中的视觉AI技术&#xff0c;轻松构建一套高精度的工业缺陷检测系统。从零开始&#xff0c;只需7个步骤…

作者头像 李华
网站建设 2026/6/10 14:06:23

VSCode运行Python实战:从数据科学到Web开发

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 开发一个多功能的Python项目模板&#xff0c;包含&#xff1a;1. 数据科学&#xff08;Jupyter Notebook集成、Pandas/Matplotlib示例&#xff09;&#xff1b;2. Web开发&#xff…

作者头像 李华