如何使用CSS创建高级动画,这个函数必须掌握
myzbx 2025-02-03 14:11 21 浏览
我们每天都在网上摸鱼,作为前端开发人员,网站上微妙的细节变化通过比别人会更关注。我一直注意到的一件事是网站上的动画的流畅性。动画对于用户体验来说是非常好的,有时我们可以一些有趣的动画来留住用户。
创建高级动画听起来是一个很难的话题,但好消息是,在CSS中,可以将多个简单的动画相互叠加,以创建一个更复杂的动画
在这节课中,我们会学习如下几点:
- 什么是贝塞尔曲线,以及如何用一行CSS来创建一个 "复杂"的动画
- 如何将动画相互叠加以创建一个高级动画
- 如何通过应用上面学到的两点来创建一个过山车动画
什么是贝塞尔曲线
CSS中的 cubic-bezier 函数是一个缓动函数,可以让我们完全控制动画在时间上的表现。下面是官方的定义:
贝塞尔缓动函数是一种由四个实数定义的缓和函数,指定了贝塞尔曲线的两个控制点P1和P2,其端点P0和P3分别固定在(0, 0)和(1, 1)。P1和P2的x坐标被限制在[0, 1]范围内。
什么是缓动函数?
线性曲线
想象两个点P0和P1,其中P0是动画的起点,P1是结束点。现在想象另一个点在两点之间线性移动,如下所示
这就是所谓的线性曲线,也是最简单的动画。
二次贝塞尔曲线
如下图所示,有三个点。P0、P1和P2。我们想让动画从P0移动到P2。在这种情况下,P1是一个控制点,控制动画的曲线。
二次方贝塞尔概念:
- 在P0和P1之间以及P1和P2之间(用灰线表示)连接虚线
- 点Q0沿着P0和P1之间的直线移动。同时,点Q1沿着P1和P2之间的直线移动
- 在Q0和Q1之间连接一条虚线(用绿线表示)
- 在Q0和Q1开始移动的同时,点B开始沿着绿线移动,B点所走的路径就是动画路径
请注意,Q0、Q1和B不以相同的速度移动。它们都必须在同一时间开始,并在同一时间完成它们的路径。因此,每一个点都是根据它所移动的线长以适当的速度移动的。
三次贝塞尔曲线
三次贝塞尔曲线由4个点组成。P0, P1, P2和P3。动画开始于P0,结束于P3。P1和P2是我们的控制点。
三次贝赛尔的工作原理如下:
- 在(P0, P1)、(P1, P2)和(P2, P3)之间连接虚线,由灰线表示
- 点Q0、Q1和Q2分别沿直线(P0,P1)、(P1,P2)和(P2,P3)移动
- 在(Q0, Q1)和(Q1, Q2)之间连接虚线,它们由绿线表示。
- 点R0和R1分别沿直线(Q0, Q1)和(Q1, Q2)移动
- 连接R0和R1之间的线(用蓝线表示)
- 最后,B点沿着R0和R1之间的连接线移动,B点所走的路径就是动画路径
如果你想更好地了解三次体贝塞尔的工作原理,建议你看看这个desmos链接。玩玩控制点,看看动画如何随时间变化。(注意,链接中的动画是由黑线表示的)。
叠加动画
有很多步骤的大动画可以被分解成多个小动画。在 css 中,通过添加animation-delay属性来实现这一点。计算延迟很简单,把你要计算动画延迟的那个动画之前的所有动画的时间加起来。
例如:
animation: movePointLeft 4s linear forwards, movePointDown 3s linear forwards;
这里,我们有两个动画,movePointLeft和movePointDown。movePointLeft的动画延迟是零,因为它是我们想先运行的动画。movePointDown的动画延迟是4秒,因为movePointLeft将在这段时间后完成。
因此,animation-delay属性:
animation-delay: 0s, 4s;
注意,如果有两个或更多的动画同时开始,它们的动画延迟将是一样的。此外,当你计算即将开始的动画的延迟时,把它们视为一个动。例如 :
animation: x 4s linear forwards, y 4s linear forwards, jump 2s linear forwards;
假设x和y同时开始。在这种情况下,x和y的动画延迟都将为零,而 jump 动画的延迟将为4秒(而不是8秒!)。
animation-delay: 0s, 0s, 4s;
创建过山车
掌握了上面的知识,是时候应用一下了。
了解动画
过山车路径由三部分组成:
- 滑动部分
- 循环部分
- 还会有一些动画,在上面的两个动画之间创造水平空间
我们将首先创建一个简单的球,作为我们过山车的 "车"。
hmtl 部分:
<div id="the-cart" class="cart"></div>
css 部分:
.cart {
background-color: rgb(100, 210, 128);
height: 50px;
width: 50px;
border: 1px solid black;
border-radius: 50px;
position: absolute;
left: 10vw;
top: 30vh;
}
滑动部分
创建小球滑动的部分可以用cubic-bezier函数来完成! 这个动画是由2个动画组成的,一个是沿x轴的动画,另一个是沿y轴的动画。X轴动画是一个沿X轴的普通线性动画。它的关键帧如下:
@keyframes x {
to {
left: 40vw;
}
将其添加到球路径的 animation 属性中,如下所示
animation: x 4s linear forwards
y轴动画是我们将使用cubic-bezier函数的部分。首先定义动画的关键帧。我们希望起始点和结束点之间的差异很小,以至于球达到的高度几乎相同。
@keyframes y {
to {
top: 29.99vh;
}
}}
现在让我们来思考一下cubic-bezier函数。我们希望我们的路径先向右缓慢移动,然后当它滑动时,它应该走得更快。
- 向右缓慢移动意味着$P1$将沿x轴移动。所以,我们知道它是在(V,0)。
- 我们需要选择一个合适的V,使我们的动画缓慢地向右移动,但又不能太多,以免占用整个空间。在这种情况下,我发现0.55最适合。
- 为了达到滑动效果,我们需要将P2向Y轴下移(负值),所以P2=(X,-Y)。
- Y应该是一个大值。在这种情况下,我选择 Y=5000。
- 为了得到X,我们知道我们的动画速度在滑动时应该更快,在再次上升时应该更慢。所以,X越接近于零,动画在滑动时就越陡峭。在这种情况下,让X = 0.8。
现在,我们得到了一个cubic-bezier函数:
cubic-bezier(0.55, 0, 0.2, -800).
为动画属性添加关键帧:
animation: x 4s linear forwards,
y 4s cubic-bezier(0.55, 0, 0.2, -5000) forwards;
这是我们动画的第一部分,所以动画延迟为零。我们应该添加一个animation-delay属性,因为从下面的动画开始,动画的开始时间将与第一个动画不同。
animation-delay: 0s, 0s;
地址:https://codepen.io/smashingmag/pen/VwxXBQb
添加水平空间
在做循环之前,球应该沿着X轴移动一小会儿,所以两个动画之间有空间。
定义关键帧
@keyframes x2 {
to {
left: 50vw;
}
}
把它添加到 animation 属性中:
animation: x 4s linear forwards,
y 4s cubic-bezier(0.55, 0, 0.2, -5000) forwards, x2 0.5s linear forwards;
这个动画应该在滑动动画之后开始,而滑动动画需要4秒,因此,动画延迟将是4秒。
animation-delay: 0s, 0s, 4s;
地址:https://codepen.io/smashingmag/pen/dyemExY
循环部分
要在CSS中创建一个圆(循环),我们需要把圆移到循环的中心,然后从那里开始做动画。圆的半径是100px,所以我们把圆的位置改为top: 20vh(30是期望的半径(这里是10vh))。然而,这需要在滑动动画完成后发生,所以我们将创建另一个持续时间为0秒的动画,并添加一个合适的动画延迟。
关键帧:
@keyframes pointOfCircle {
to {
top: 20vh;
}
}
添加到 animation 动画中:
animation: x 4s linear forwards,
y 4s cubic-bezier(0.55, 0, 0.2, -5000) forwards, x2 0.5s linear forwards,
pointOfCircle 0s linear forwards;
添加动画延迟, 4.5s:
animation-delay: 0s, 0s, 4s, 4.5s;
循环本身
创建一个循环动画:
- 创建一个关键帧,将球移回原来的位置,然后旋转球。
@keyframes loop {
from {
transform: rotate(0deg) translateY(10vh) rotate(0deg);
}
to {
transform: rotate(-360deg) translateY(10vh) rotate(360deg);
}
}
添加到 animation 中:
animation: x 4s linear forwards,
y 4s cubic-bezier(0.55, 0, 0.2, -5000) forwards, x2 0.5s linear forwards,
pointOfCircle 0s linear forwards, loop 3s linear forwards;
添加动画延迟,这里是4.5s:
animation-delay: 0s, 0s, 4s, 4.5s, 4.5s;
地址:https://codepen.io/smashingmag/pen/mdLxZdR
添加水平空间
快完成了,最后 只需要在动画之后沿着x轴移动球,这样球就不会像上图中那样在循环之后完全停止。
关键帧:
@keyframes x3 {
to {
left: 70vw;
}
}
添加到 animation 中:
animation: x 4s linear forwards,
y 4s cubic-bezier(0.55, 0, 0.2, -800) forwards, x2 0.5s linear forwards,
pointOfCircle 0s linear forwards, loop 3s linear forwards,
x3 2s linear forwards;
加上适当的延迟,这里是7.5s:
animation-delay: 0s, 0s, 4s, 4.5s, 4.5s, 7.5s;
地址:https://codepen.io/smashingmag/pen/wvjmLKp
总结
在本节中,我们介绍了如何结合多个关键帧来创建一个复杂的动画路径。我们还介绍了贝塞尔以及如何使用它们来创建你自己的缓动函数。建议大家自己多多动手,才能更好的掌握 css 动画。
来源:https://www.smashingmagazine.com/2022/10/advanced-animations-css/
相关推荐
- 判断变量是否为数组(判断变量是否为数组的函数)
-
大家好,我是前端西瓜哥,今天带大家学习在JS中如何判断一个对象是否为数组。Array.isArray最好的写法是使用Array.isArray(val)。因为该方法能正确判断iframe传过...
- 2023-03-13:给定一个整数数组 A,坡是元组 (i, j),其中 i < j 且 A[i]
-
2023-03-13:给定一个整数数组A,坡是元组(i,j),其中i<j且A[i]<=A[j],这样的坡的宽度为j-i。找出A中的坡的最大宽度,如果不存在,返...
- 世界上形形色色的“奇葩”小望远镜②
-
上一篇我们介绍了在探索伽玛射线暴、超新星上大放光彩的小望远镜,但天文学所研究的领域却是非常的多,比如我们所好奇的暗物质以及太阳系外行星等,那么在这些方面做出突出贡献的“奇葩”小望远镜都是谁呢?1、蜻蜓...
- C#解析多层嵌套的JSON数组(多层嵌套json转换成map)
-
首先引用开源类库:Newtonsoft.Json.dll,目前最低支持.NET3.5版本。官方帮助文档:http://www.newtonsoft.com/json/help/html/Samples...
- Vlookup函数的这7个应用技巧都不掌握,那就真的Out了
-
查询引用,用到最多的函数为Vlookup,但你真的会用吗?其实,Vlookup函数除了常规的查询引用外,还有多种使用技巧一、Vlookup函数:功能及语法结构。功能:在指定的数据范围内返回符合查询要...
- C语言-闲聊一维、二维数组(c语言中二维数组的定义和使用)
-
①若a[i]为一维数组则有,a[0],为数组的一个元素。a[i]=*(&a[i]),为数组的一个元素。a+i=&a[i],为元素a[i]的地址。*(*(a+i))=*(*&a[...
- Excel常用技能分享与探讨(5-宏与VBA简介之VBA的数组与集合)
-
总结数组:适合处理固定大小、类型一致、需要快速访问的数据。集合:适合动态增删、键值查找或混合类型数据。根据具体需求选择合适的数据结构,可显著提升代码效率和可读性。一、从仓库管理理解数据结构(场景化入门...
- 数据结构串和数组(一)(数据结构串的概念)
-
一、串的基本概念串是由零个或多个字符组成的有限序列。记作str="a0a1…an-1"(n≥0)。串中所包含的字符个数n称为串长度,当n=0时,称为空串。一个串中任意连续的字符组成的子...
- C语言进阶教程:指针数组与数组指针
-
在C语言中,指针和数组是两个核心且紧密相关的概念。当它们结合时,就产生了指针数组(ArrayofPointers)和数组指针(PointertoanArray)。这两者在语法、含义和用途上都...
- 一篇文章搞懂数组的所有知识点(一篇文章搞懂数组的所有知识点怎么写)
-
1.一维数组数组是一种数据结构,用来存储多个相同类型的数据,并通过索引来访问每个元素。概念描述示例代码什么是数组?数组是一种数据结构,用来存储一组相同类型的值。你可以把它想象成一个排好序的储物柜,每...
- 这些Java基础知识,诸佬们都还记得嘛(学习,复习,面试均可)
-
方法重载和方法重写的区别方法重写重写体现在继承关系上。在Java中,子类继承父类,子类就会具备父类所以的特征,以及父类的方法和变量比如动物类有“叫”的方法,小狗小猫分别继承了动物类,重写方法时就可以...
- js将list转化为tree格式的几种写法
-
最近在考虑一个树状结构存储。最终需要将list转化为tree格式源数据示例源数据共401条[{"menuId":"5f50c5fb8f0d74536bbfb7a4"...
- Java学习之数组——java基础篇(java数组知识)
-
如果希望保存一组有相同类型的数据,可以使用数组。数组的定义和内存分配Java中定义数组的语法有两种:typearrayName[];type[]arrayName;type为Java中的任...
- C语言-数组平均值与排序(c语言数组平均数)
-
①目标输入一个正数数组,求平均值,并根据平均值重新排序,大于平均值的数前置,小于等于平均值的值后置。~②命令行#include<stdio.h>调用输入输出函数库#include<...
- 数据结构串和数组(二)(数据结构串的概念)
-
数组的基本概念数组是一个二元组(idx,value)的集合,对每个idx,都有一个value值与之对应。idx称为下标,可以由一个整数、两个整数或多个整数构成,下标含有d(d≥1)个整数称为维数是d。...
- 一周热门
- 最近发表
- 标签列表
-
- HTML 简介 (30)
- HTML 响应式设计 (31)
- HTML URL 编码 (32)
- HTML Web 服务器 (31)
- HTML 表单属性 (32)
- HTML 音频 (31)
- HTML5 支持 (33)
- HTML API (36)
- HTML 总结 (32)
- HTML 全局属性 (32)
- HTML 事件 (31)
- HTML 画布 (32)
- HTTP 方法 (30)
- 键盘快捷键 (30)
- CSS 语法 (35)
- CSS 选择器 (30)
- CSS 轮廓宽度 (31)
- CSS 谷歌字体 (33)
- CSS 链接 (31)
- CSS 定位 (31)
- CSS 图片库 (32)
- CSS 图像精灵 (31)
- SVG 文本 (32)
- 时钟启动 (33)
- HTML 游戏 (34)