如何使用CSS和GSAP实现有多个关键帧的连续动画(附源码)

本篇文章给大家带来的内容是关于如何使用cssgsap实现有多个关键帧的连续动画(附源码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。

效果预览

4094838559-5b985d3ec76bc_articlex.gif

源代码下载

https://github.com/comehope/front-end-daily-challenges

代码解读

定义 dom,容器中包含 10 个 p 子元素,每个 p 中包含 1 个 span 元素:

居中显示:

body {    margin: 0;    height: 100vh;    display: flex;    align-items: center;    justify-content: center;    background-color: lightyellow;}

定义容器的尺寸和样式:

立即学习“前端免费学习笔记(深入)”;

.container {    width: 400px;    height: 400px;    background: linear-gradient(45deg, tomato, gold);    border-radius: 3%;    box-shadow: 0 0 10px rgba(0, 0, 0, 0.3);}

画出容器里的 1 个元素,它有一个外壳 p,里面是一个白色的小方块 span

.container {    position: relative;}.container p {    position: absolute;    width: inherit;    height: inherit;    display: flex;    align-items: center;    justify-content: center;}.container p span {    position: absolute;    width: 40px;    height: 40px;    background-color: white;}

为容器中的元素定义下标变量,并让元素的外壳依次旋转,围合成一个圆形,其中 outline 是辅助线:

.container p {    outline: 1px dashed black;    transform: rotate(calc((var(--n) - 1) * 36deg));}.container p:nth-child(1) { --n: 1; }.container p:nth-child(2) { --n: 2; }.container p:nth-child(3) { --n: 3; }.container p:nth-child(4) { --n: 4; }.container p:nth-child(5) { --n: 5; }.container p:nth-child(6) { --n: 6; }.container p:nth-child(7) { --n: 7; }.container p:nth-child(8) { --n: 8; }.container p:nth-child(9) { --n: 9; }.container p:nth-child(10) { --n: 10; }

至此,子元素绘制完成,接下来开始写动画脚本。
引入 GSAP 库:


定义一个变量,代表子元素选择器:

let elements = '.container p span';

声明一个时间线对象:

let animation = new TimelineMax();

先设定入场方式为由小(第1帧)变大(第2帧),其中并没有第 2 帧的代码,它是隐含在语义中的:

animation.from(elements, 1, {scale: 0});

让子元素变成竖长条,向四周散开(第3帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25});

让竖长条旋转着变成小方块(第4帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180});

让小方块变成横长条,围成一个圆形(第5帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180})    .to(elements, 1, {scaleX: 1});

注意,因 scrimba 在录制过多帧时会崩溃,所以第 6 帧至第 11 帧没有在视频中体现。
让圆形向内收敛,同时线条变细(第6帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180})    .to(elements, 1, {scaleX: 1})    .to(elements, 1, {y: '-60px', scaleY: 0.1});

让线条向左摆动(第7帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180})    .to(elements, 1, {scaleX: 1})    .to(elements, 1, {y: '-60px', scaleY: 0.1})    .to(elements, 1, {x: '-30px'});

再让线条向右摆动(第8帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180})    .to(elements, 1, {scaleX: 1})    .to(elements, 1, {y: '-60px', scaleY: 0.1})    .to(elements, 1, {x: '-30px'})    .to(elements, 1, {x: '30px'});

再把横线变为竖线,造型与第 3 帧相似,只是线更细,更向内收敛(第9帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180})    .to(elements, 1, {scaleX: 1})    .to(elements, 1, {y: '-60px', scaleY: 0.1})    .to(elements, 1, {x: '-30px'})    .to(elements, 1, {x: '30px'})    .to(elements, 1, {x: '0', scaleX: 0.1, scaleY: 1});

再把竖线变为横线,造型与第 5 帧相似,但线短一些(第10帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180})    .to(elements, 1, {scaleX: 1})    .to(elements, 1, {y: '-60px', scaleY: 0.1})    .to(elements, 1, {x: '-30px'})    .to(elements, 1, {x: '30px'})    .to(elements, 1, {x: '0', scaleX: 0.1, scaleY: 1})    .to(elements, 1, {scaleX: 0.5, scaleY: 0.1})

横线稍向外扩散,变为圆点(第11帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180})    .to(elements, 1, {scaleX: 1})    .to(elements, 1, {y: '-60px', scaleY: 0.1})    .to(elements, 1, {x: '-30px'})    .to(elements, 1, {x: '30px'})    .to(elements, 1, {x: '0', scaleX: 0.1, scaleY: 1})    .to(elements, 1, {scaleX: 0.5, scaleY: 0.1})    .to(elements, 1, {y: '-80px', scaleY: 0.5, borderRadius: '50%'});

让圆点变形为竖线,并向内收缩,这个变化的距离长,所以动画时间也要长一些(第12帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180})    .to(elements, 1, {scaleX: 1})    .to(elements, 1, {y: '-60px', scaleY: 0.1})    .to(elements, 1, {x: '-30px'})    .to(elements, 1, {x: '30px'})    .to(elements, 1, {x: '0', scaleX: 0.1, scaleY: 1})    .to(elements, 1, {scaleX: 0.5, scaleY: 0.1})    .to(elements, 1, {y: '-80px', scaleY: 0.5, borderRadius: '50%'})    .to(elements, 1, {y: '-10px', scaleX: 0.1, scaleY: 0.5, borderRadius: '0%', rotation: 0});

让竖线从中心向外快速扩散,扩散前稍停片刻,好像线条都被发射出一样(第13帧):

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180})    .to(elements, 1, {scaleX: 1})    .to(elements, 1, {y: '-60px', scaleY: 0.1})    .to(elements, 1, {x: '-30px'})    .to(elements, 1, {x: '30px'})    .to(elements, 1, {x: '0', scaleX: 0.1, scaleY: 1})    .to(elements, 1, {scaleX: 0.5, scaleY: 0.1})    .to(elements, 1, {y: '-80px', scaleY: 0.5, borderRadius: '50%'})    .to(elements, 1, {y: '-10px', scaleX: 0.1, scaleY: 0.5, borderRadius: '0%', rotation: 0})    .to(elements, 1, {y: '-300px', delay: 0.5});

用时间尺度缩放函数让动画播放速度加快一倍:

animation.from(elements, 1, {scale: 0})    .to(elements, 1, {y: '-100px', scaleX: 0.25})    .to(elements, 1, {scaleY: 0.25, rotation: 180})    .to(elements, 1, {scaleX: 1})    .to(elements, 1, {y: '-60px', scaleY: 0.1})    .to(elements, 1, {x: '-30px'})    .to(elements, 1, {x: '30px'})    .to(elements, 1, {x: '0', scaleX: 0.1, scaleY: 1})    .to(elements, 1, {scaleX: 0.5, scaleY: 0.1})    .to(elements, 1, {y: '-80px', scaleY: 0.5, borderRadius: '50%'})    .to(elements, 1, {y: '-10px', scaleX: 0.1, scaleY: 0.5, borderRadius: '0%', rotation: 0})    .to(elements, 1, {y: '-300px', delay: 0.5})    .timeScale(2);

修改声明时间线的代码,使动画重复播放:

let animation = new TimelineMax({repeat: -1, repeatDelay: 1});

至此,动画完成。
隐藏容器外的内容,并删掉辅助线;

.container {    overflow: hidden;}.container p {    /* outline: 1px dashed black; */}

最后,装饰一下页面的角落:

body {    overflow: hidden;}body::before,body::after {    content: '';    position: absolute;    width: 60vmin;    height: 60vmin;    border-radius: 50%;    background: radial-gradient(        transparent 25%,        gold 25%, gold 50%,        tomato 50%    );}body::before {    left: -30vmin;    bottom: -30vmin;}body::after {    right: -30vmin;    top: -30vmin;}

大功告成!

相关推荐:

如何使用纯CSS实现一把剪刀的效果(附源码)

如何使用纯CSS实现条纹错觉的动画效果(附源码)

以上就是如何使用CSS和GSAP实现有多个关键帧的连续动画(附源码)的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1611646.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月24日 02:27:37
下一篇 2025年12月24日 02:27:50

相关推荐

  • 深入理解CSS之overflow:hidden,scroll,visible

    在布局时,如果遇到内容超过盒子大小时,可以用css属性overflow将超出对象的内容实现隐藏,同时也可以将超出部分显示或者隐藏滚动条的作用,接下来我们来了解学习css overflow样式。 一、Overflow语法值 overflow : visible | auto | hidden | sc…

    好文分享 2025年12月24日
    000
  • 如何使用纯CSS实现一只会动的手(附源码)

    本篇文章给大家带来的内容是关于如何使用纯css实现一只会动的手(附源码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 效果预览 源代码下载 https://github.com/comehope/front-end-daily-challenges 代码解读 定义 dom,容器中…

    2025年12月24日
    000
  • 如何用CSS纯代码画一个旋转的太极图(附代码)

    这是一个中国人都非常熟悉的图案——太极图,它分为阴阳两级,会旋转(用到css3 animation动画属性),此图通过css3纯代码实现,它由 若干个小圆组合而成,适合初学css3的人练练手,娱以 致学,增加学习的乐趣。 HTML部分: CSS部分: .square { width: 400px; …

    2025年12月24日
    000
  • 如何用css画正六边形?用css画正六边形的两种方法(代码实例)

    本章给大家介绍如何用css画正六边形?用css画正六边形的两种方法(代码实例)。有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 在之前要先了解一下正六边形内角和边的关系,正六边形的每个内角是60deg,如图(√3其实是根号3): 方法一:原理把正六边形分成三部分,左中右分别是:bef…

    2025年12月24日 好文分享
    000
  • CSS实现四个方向箭头的代码

    本篇文章给大家带来的内容是关于CSS实现四个方向箭头的代码,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 自学教程(如约智惠.com) i { border:solid black; border-width:0 3px 3px 0; display:inline-block; p…

    好文分享 2025年12月24日
    000
  • css如何实现footer定位(完整代码)

    本篇文章给大家带来的内容是关于css如何实现footer定位(完整代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 css实现footer定位 Document html,body{ width: 100%; height: 100%; margin: 0; padding: …

    好文分享 2025年12月24日
    000
  • 如何使用css3实现魔方的动画效果(完整代码)

    本篇文章给大家带来的内容是关于如何使用css3实现魔方的动画效果(完整代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 Document *{ margin: 0; padding: 0; } .box{ width: 300px; height: 300px; margin:…

    好文分享 2025年12月24日
    000
  • 什么是浮动以及浮动的清除方法

    本章给大家介绍什么是浮动以及浮动的清除方法。有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 一、浮动 说浮动就必须提及一下文档流,html当中的元素按照从左到右,从上到下的顺序进行排列称之为文档流,也就是正常排列。 而浮动是什么呢?浮动会让元素脱离文档流,假如A元素浮动了,原本排在该…

    2025年12月24日 好文分享
    000
  • css实现横向滚动条的两种方式(代码实例)

    本章给大家介绍用css实现横向滚动条的两种方式。有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 html代码: 全部 Adobe 微软 会计 绘画 Adobe 微软 会计 绘画 一、 原始css + jquery 实现横向滚动条(原生js可以实现,为了方便才用的jQuery) css…

    好文分享 2025年12月24日
    000
  • CSS定位position的多种方式以及不同方式之间的区别

    在进行前端布局的时候,我们经常会用到定位,定位是css中重要的一部分,定位是用定位属性position来进行定位的,position的值也有很多,具体它有如下几个属性值。常见的属性有如下所示: 值描述 absolute  生成绝对定位的元素,相对于static定位以外的第一个父元素进行定位。元素的位…

    好文分享 2025年12月24日
    000
  • 如何使用纯CSS实现一把剪刀的效果(附源码)

    本篇文章给大家带来的内容是关于如何使用纯css实现一把剪刀的效果(附源码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 效果预览 源代码下载 https://github.com/comehope/front-end-daily-challenges 代码解读 定义 dom,容器…

    2025年12月24日
    000
  • 如何使用CSS在线字体和D3实现Google的信息图

    本篇文章给大家带来的内容是如何使用css在线字体和d3实现google的信息图 ,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 效果预览 源代码下载 https://github.com/comehope/front-end-daily-challenges 代码解读 定义 dom…

    2025年12月24日
    000
  • CSS中什么是BEM 命名规范?

    本章为大家带来css中什么是bem 命名规范?,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 一 什么是 BEM 命名规范 Bem 是块(block)、元素(element)、修饰符(modifier)的简写,由 Yandex 团队提出的一种前端 CSS 命名方法论。   &#82…

    好文分享 2025年12月24日
    000
  • css为select添加样式(无脚本)实现

    改变select默认的样式,一般情况下通过ul,li来模拟来实现。 有很多jquery插件就是通过这样的方式来改变select默认样式的。 根据程序哥哥那边的反映,此种方式在form提交后无法获取数据,后来经过实验,用了不同的js/jquery插件,都是同样的结果:无法获取数据。 后来看一篇外国人写…

    好文分享 2025年12月24日
    000
  • css实现双飞翼布局的四种方法(附代码)

    本篇文章给大家带来的内容是关于css实现双飞翼布局的四种方法(附代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 圣杯布局、双飞翼布局效果图 从效果图来看圣杯布局、双飞翼布局效果是一样一样的。圣杯布局、双飞翼布局就是左右两侧宽度固定,中间内容宽度自适应,即100% 圣杯布局 *…

    2025年12月24日
    000
  • CSS入门教程之margin属性

    设置外边距的最简单的办法就是运用 margin属性。margin用于设置对象标签之间间隔距离,比方2个上下排列的div盒子,我们就能够运用margin款式完成上下2个盒子间距。margin 属性承受任何长度单位,能够是像素、英寸、毫米或 em。margin 能够设置为 auto。更常见的做法是为外边…

    2025年12月24日
    000
  • css怎么添加阴影效果?(代码实例)

    css怎么添加阴影效果?可以使用text-shadow属性与box-shadow属性来添加。下面本篇文章就来给大家介绍一下text-shadow属性与box-shadow属性是如何添加阴影效果的,有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 推荐手册:CSS3最新版参考手册 一:添…

    2025年12月24日
    000
  • css怎么让图片自适应?css图片自适应大小的方法介绍

    对于一个网页来说,有一张好看又清晰的背景图片是非常吸引人的,但是并不是每张图片都是有相同的大小,所以就需要晒西安图片的自适应,那么,css怎么让图片自适应呢?本篇文章将来介绍关于css图片自适应大小的方法。 我们下面就来直接看一个例子: #web_bg{ position:fixed; top: 0…

    好文分享 2025年12月24日
    000
  • 分享CSS按钮的多种制作方式,附CSS 按钮样式代码

    在项目中,几乎所有的网站都会用到按钮,多则十几个,少则几个。一个实用又好看的按钮,能给页面增色不少,从而更能吸引用户的眼球。本文主要讲述css按钮的多种写法和css按钮的美化,最后会展示一款常见的css按钮样式代码,供参考。好,继续往下看吧! 一、CSS按钮的多种写法 CSS按钮可以用button直…

    2025年12月24日
    000
  • css自定义属性和聚光灯效果的实现(代码)

    本篇文章给大家带来的内容是关于css自定义属性和聚光灯效果的实现(代码),有一定的参考价值,有需要的朋友可以参考一下,希望对你有所帮助。 简介 CSS Variables,一个并不是那么新的东西,但对css来说绝对是一场革命。 之前使用变量的时候,需要借助sass、less等预处理工具来实现,现在我…

    2025年12月24日 好文分享
    000

发表回复

登录后才能评论
关注微信