前端动画库:让你的网站动起来

前端动画库:让你的网站动起来

毒舌时刻

前端动画?这不是用CSS就够了吗?

"CSS动画简单,我只用CSS"——结果复杂动画难以实现,
"JavaScript动画性能差,我不用"——结果交互体验差,
"Framer Motion?GSAP?没听说过,肯定不如CSS"——结果错过了更强大的动画能力。

醒醒吧,前端动画不是简单的CSS过渡,而是需要根据场景选择合适的工具!

为什么你需要这个?

  • 用户体验:流畅的动画提升用户体验
  • 交互反馈:动画可以提供清晰的交互反馈
  • 视觉吸引力:动画让网站更具视觉吸引力
  • 品牌识别:独特的动画风格可以强化品牌识别

反面教材

/* 反面教材:过度使用CSS动画 */ .animation { /* 复杂的CSS动画,难以维护 */ animation: rotate 2s linear infinite, scale 1s ease-in-out infinite alternate, color-change 3s ease-in-out infinite; } @keyframes rotate { from { transform: rotate(0deg); } to { transform: rotate(360deg); } } @keyframes scale { from { transform: scale(1); } to { transform: scale(1.1); } } @keyframes color-change { 0% { color: red; } 50% { color: blue; } 100% { color: red; } } 
// 反面教材:使用setTimeout实现动画 function animateElement() { const element = document.getElementById('element'); let position = 0; const interval = setInterval(() => { if (position >= 100) { clearInterval(interval); } else { position += 1; element.style.left = position + 'px'; } }, 16); } 

正确的做法

// 正确的做法:使用Framer Motion import React from 'react'; import { motion } from 'framer-motion'; function FramerMotionExample() { return ( <div> <h2>Framer Motion 示例</h2> {/* 基础动画 */} <motion.div initial={{ opacity: 0, y: 20 }} animate={{ opacity: 1, y: 0 }} transition={{ duration: 0.5 }} className="box" > 淡入动画 </motion.div> {/* 循环动画 */} <motion.div animate={{ rotate: 360 }} transition={{ duration: 2, repeat: Infinity, ease: "linear" }} className="circle" > 旋转动画 </motion.div> {/* 交互动画 */} <motion.button whileHover={{ scale: 1.05 }} whileTap={{ scale: 0.95 }} className="button" > 悬停按钮 </motion.button> {/* 序列动画 */} <motion.div className="container"> {[1, 2, 3, 4, 5].map((item) => ( <motion.div key={item} initial={{ opacity: 0, y: 20 }} animate={{ opacity: 1, y: 0 }} transition={{ delay: item * 0.1, duration: 0.5 }} className="item" > 项目 {item} </motion.div> ))} </motion.div> </div> ); } // 正确的做法:使用GSAP import React, { useEffect, useRef } from 'react'; import gsap from 'gsap'; function GSAPExample() { const containerRef = useRef(null); const boxRef = useRef(null); useEffect(() => { // 基础动画 gsap.to(boxRef.current, { x: 100, y: 50, rotate: 45, duration: 1, ease: "power2.out" }); // 时间线动画 const tl = gsap.timeline({ repeat: -1, yoyo: true }); tl.to(".item", { x: 100, duration: 0.5, stagger: 0.1 }) .to(".item", { y: 50, duration: 0.5 }) .to(".item", { opacity: 0.5, duration: 0.5 }); }, []); return ( <div ref={containerRef}> <h2>GSAP 示例</h2> <div ref={boxRef} className="box"> GSAP 动画 </div> <div className="container"> {[1, 2, 3, 4, 5].map((item) => ( <div key={item} className="item"> 项目 {item} </div> ))} </div> </div> ); } // 正确的做法:使用React Spring import React from 'react'; import { useSpring, animated } from 'react-spring'; function ReactSpringExample() { // 基础动画 const fadeIn = useSpring({ from: { opacity: 0, transform: 'translateY(20px)' }, to: { opacity: 1, transform: 'translateY(0)' }, config: { tension: 100, friction: 10 } }); // 交互动画 const [isHovered, setIsHovered] = React.useState(false); const buttonAnimation = useSpring({ scale: isHovered ? 1.1 : 1, config: { tension: 300, friction: 10 } }); return ( <div> <h2>React Spring 示例</h2> <animated.div style={fadeIn} className="box" > 淡入动画 </animated.div> <animated.button style={buttonAnimation} onMouseEnter={() => setIsHovered(true)} onMouseLeave={() => setIsHovered(false)} className="button" > 悬停按钮 </animated.button> {/* 循环动画 */} <animated.div style={useSpring({ rotate: 360, from: { rotate: 0 }, config: { duration: 2000 }, loop: true })} className="circle" > 旋转动画 </animated.div> </div> ); } // 正确的做法:选择合适的动画库 function AnimationLibraryComparison() { return ( <div> <h1>前端动画库比较</h1> <div className="comparison"> <div className="library"> <h3>CSS Animations</h3> <p>优点:简单、性能好、无需依赖</p> <p>缺点:复杂动画难以实现、交互性差</p> <p>适用场景:简单的过渡效果、加载动画</p> </div> <div className="library"> <h3>Framer Motion</h3> <p>优点:API友好、交互性强、React集成好</p> <p>缺点:包体积较大</p> <p>适用场景:React应用、复杂交互动画</p> </div> <div className="library"> <h3>GSAP</h3> <p>优点:功能强大、性能优异、支持复杂动画</p> <p>缺点:学习曲线较陡</p> <p>适用场景:复杂动画、时间线动画、SVG动画</p> </div> <div className="library"> <h3>React Spring</h3> <p>优点:基于物理的动画、性能好、API简洁</p> <p>缺点:功能相对较少</p> <p>适用场景:流畅的物理动画、交互反馈</p> </div> </div> </div> ); } 

毒舌点评

看看,这才叫前端动画!不是简单地使用CSS或setTimeout,而是根据不同的场景选择合适的动画库。

记住,每种动画库都有其优缺点,你需要根据项目需求和复杂度选择合适的工具。Framer Motion适合React应用,GSAP适合复杂动画,React Spring适合物理动画,CSS适合简单过渡。

所以,别再固守一种动画方式了,灵活选择合适的工具,让你的网站动起来!

总结

  • CSS Animations:适合简单的过渡效果和加载动画
  • Framer Motion:适合React应用和复杂交互动画
  • GSAP:适合复杂动画、时间线动画和SVG动画
  • React Spring:适合基于物理的流畅动画
  • 性能优化:使用transform和opacity属性,避免重排
  • 动画策略:合理使用动画,避免过度动画
  • 可访问性:考虑动画对用户的影响,提供减少动画的选项
  • 响应式:确保动画在不同设备上都能正常工作

前端动画,让你的网站更具生命力!

Read more

企业管理系统前端组件化设计实战:OA、CRM、ERP 表单为什么不能直接用 Element UI / Ant Design?

企业管理系统前端组件化设计实战:OA、CRM、ERP 表单为什么不能直接用 Element UI / Ant Design?

企业管理系统前端组件化设计实战:OA、CRM、ERP 表单为什么不能直接用 Element UI / Ant Design? 🌐 文档地址:http://ruoyioffice.com | 📦 源码1:https://gitee.com/yqzy1688/ruoyi-office-vben.git |📦 源码2:https://gitee.com/yqzy1688/ruoyi-office.git |📦 源码3:https://github.com/yuqing2026/ruoyi-office.git | 💬 :17156169080(备注「RuoYi Office」) 做过企业管理系统的前端开发者都有一个共同痛点:每做一个新模块,就要重复写一堆表单、表格、状态标签、操作按钮的代码。 更糟糕的是,无论你用 Element UI(Element Plus)

【Linux篇章】穿越网络迷雾:揭开 HTTP 应用层协议的终极奥秘!从请求响应到实战编程,从静态网页到动态交互,一文带你全面吃透并征服 HTTP 协议,打造属于你的 Web 通信利刃!

【Linux篇章】穿越网络迷雾:揭开 HTTP 应用层协议的终极奥秘!从请求响应到实战编程,从静态网页到动态交互,一文带你全面吃透并征服 HTTP 协议,打造属于你的 Web 通信利刃!

本篇摘要 本篇将介绍何为HTTP协议,以及它的请求与答复信息的格式(请求行,请求包头,正文等),对一些比较重要的部分来展开讲解,其他不常用的即一概而过,从静态网页到动态网页的过渡,最后底层基于TCP实现简单的HTTP服务器的代码编写构建一个简单的网页(包含对应的跳转,重定向,动态交互等功能),采取边讲解http结构边用代码形成效果展示的形式进行讲解,望有助! 欢迎拜访:点击进入博主主页 本篇主题:探秘HTTP应用层那些事儿! 制作日期:2025.07.21 隶属专栏:点击进入所属Linux专栏 本文将要介绍的内容的大致流程图如下: 一· 认识HTTP * 在互联网世界中, HTTP(HyperText Transfer Protocol, 超文本传输协议) 是一个至关重要的协议。 它定义了客户端(如浏览器) 与服务器之间如何通信, 以交换或传输超文本(如 HTML 文档) 。 * HTTP 协议是客户端与服务器之间通信的基础。 * 客户端通过 HTTP 协议向服务器发送请求, 服务器收到请求后处理并返回响应。 HTTP 协议是一个无连接、

前端虚拟列表实现:别再渲染10000个DOM节点了

前端虚拟列表实现:别再渲染10000个DOM节点了

前端虚拟列表实现:别再渲染10000个DOM节点了 毒舌时刻 这代码写得跟网红滤镜似的——仅供参考。 各位前端同行,咱们今天聊聊前端虚拟列表。别告诉我你还在一次性渲染10000个列表项,那感觉就像把10000本书全部摆在桌面上——既占地方又难找。 为什么你需要虚拟列表 最近看到一个项目,一个下拉列表有5000个选项,全部渲染导致页面卡死,我差点当场去世。我就想问:你是在做列表还是在做性能杀手? 反面教材 // 反面教材:一次性渲染所有数据 function BigList({ items }) { return ( <ul style={{ height: '400px', overflow: 'auto' }}> {items.map(item => ( <li key={item.id} style={{ height: '50px'