带你玩转css3动画-animation

写在前面

总结是一种学习方式,取长补短是一种学习态度

我们今天聊一些你不知道的 CSS3 Animation。很多人都觉得 css3 很容易,不用怎么学,就能熟练掌握,但是一个不懂 css3 动画的前端工程师不能称之为掌握 css3。在日常开发中遇到一些要做动画的模块就毫无头绪,要不就是用 js 来写动画,所以,把平时在工作用过的一些 css3 动画做一个总结,让大家在以后能更顺手的使用动画来提高自己的页面逼格。

animation 属性是 8 个属性的简写:

animation属性
animation属性理解

通过几个例子来了解一下 animation 动画,看如何用 animation 来打造各种各样的动画,让你的页面更加酷炫。

你是否被gif loading加载太慢或有锯齿而感到困扰?

项目中,当页面内容或图片比较多或在加载一些比较大的数据接口的时候,加载时间会比较长,此时可能出现页面白屏的情况,用户体验较差。为了让用户会有一个等待的效果,会在加载成功前显示一个 loading 来过渡,当页面完全加载完后,让 loading 消失即可。
loading 可以用 gif loading 显示,但是,优雅的前端工程师觉得这样又多了一个请求和有时候也会加载太慢,于是,他奇思妙想,觉得可以用纯 css loading 来达到效果。

css loading
<div class="box">
    <div class="loader">
        <div class="loading-1">
            <i></i>
        </div>
    </div>
    <div class="loader">
        <div class="loading-2">
            <i></i><i></i><i></i><i></i><i></i>
        </div>
    </div>
    <div class="loader">
        <div class="loading-3">
            <i></i><i></i><i></i><i></i><i></i><i></i><i></i><i></i>
        </div>
    </div>
</div>

关键的css样式

.loading-1{width:35px;height:35px;position:relative}
@-webkit-keyframes loading-1{
    0%{transform:rotate(0)}
    50%{transform:rotate(180deg)}
    100%{transform:rotate(360deg)}
}
.loading-1 i{display:block;width:100%;height:100%;border-radius:50%;background:linear-gradient(transparent 0,transparent 70%,#333 30%,#333 100%);-webkit-animation:loading-1 .6s linear 0s infinite}

@-webkit-keyframes loading-2{
    0%{transform:scale(1)}
    50%{transform:scale(.4)}
    100%{transform:scale(1)}
}
.loading-2 i{display:inline-block;width:4px;height:35px;margin:0 2px;background:#333;border-radius:2px;-webkit-animation:loading-2 1s ease-in .1s infinite}
.loading-2 i:nth-child(1){-webkit-animation:loading-2 1s ease-in .1s infinite}
.loading-2 i:nth-child(2){-webkit-animation:loading-2 1s ease-in .2s infinite}
.loading-2 i:nth-child(3){-webkit-animation:loading-2 1s ease-in .3s infinite}
.loading-2 i:nth-child(4){-webkit-animation:loading-2 1s ease-in .4s infinite}
.loading-2 i:nth-child(5){-webkit-animation:loading-2 1s ease-in .5s infinite}

@-webkit-keyframes loading-3{
    50%{transform:scale(.4);opacity:.3}
    100%{transform:scale(1);opacity:1}
}
.loading-3{position:relative}
.loading-3 i{display:block;width:15px;height:15px;border-radius:50%;position:absolute;background:#333}
.loading-3 i:nth-child(1){top:25px;left:0;-webkit-animation:loading-3 1s ease 0s infinite}
.loading-3 i:nth-child(2){top:17px;left:17px;-webkit-animation:loading-3 1s ease .12s infinite}
.loading-3 i:nth-child(3){top:0;left:25px;-webkit-animation:loading-3 1s ease .24s infinite}
.loading-3 i:nth-child(4){top:-17px;left:17px;-webkit-animation:loading-3 1s ease .36s infinite}
.loading-3 i:nth-child(5){top:-25px;left:0;-webkit-animation:loading-3 1s ease .48s infinite}
.loading-3 i:nth-child(6){top:-17px;left:-17px;-webkit-animation:loading-3 1s ease .6s infinite}
.loading-3 i:nth-child(7){top:0;left:-25px;-webkit-animation:loading-3 1s ease .72s infinite}
.loading-3 i:nth-child(8){top:17px;left:-17px;-webkit-animation:loading-3 1s ease .84s infinite}

上面例子,主要通过 css animation 设置不同时间段的 animation-delay 来延迟展示效果,再通过设置 animation-iteration-countinfinite 来进行无限的转动,展示出 loading 的效果,给人有一种在等待的过程;代码非常简洁实用,我们可以根据项目需要,开发出适用于项目的 loading

骨架屏:一个比loading更加优雅的加载

据研究,用户大概会在 200ms 内获取到界面的具体关注点,为了优化首屏渲染时间这个指标,减少白屏时间,在数据获取或页面加载完成之前,给用户首先展现骨架屏,骨架屏的样式、布局和真实数据渲染的页面保持一致,这样用户在骨架屏中获取到关注点,并能够预知页面什么地方将要展示文字什么地方展示图片,这样也就能够将关注焦点移到感兴趣的位置。下面是完全通过 HTMLCSS 手写的就是在项目中通过简单的模拟骨架屏,来达到数据未请求到数据前的过渡。

骨架屏
<div class="skeleton-wrap">
    <div class="skeleton">
        <div class="left-right item"></div><div class="line-zero item"></div><div class="line-one item"></div><div class="line-two item"></div><div class="line-three item"></div>
    </div>
</div>

关键css代码

.skeleton-wrap {
    box-sizing: border-box;
    height: 8rem;
    background-color: #fff;
    padding: 0;
    .skeleton {
        height: 6rem;
        box-sizing: border-box;
        position: relative;
        animation-duration: 1s;
        animation-fill-mode: forwards;
        animation-iteration-count: infinite;
        animation-name: placeHolderShimmer;
        animation-timing-function: linear;
        background: linear-gradient(to right, #eeeeee, #dddddd 10%, #eeeeee 30%);
        background-size: 400% 400%;
        .item {position: absolute;}
        .left-right {height: 6rem;width: 2rem;top: 0;left: 6rem;background-color: #fff;}
        .line-zero {height: 0.4rem;width: calc(100% - 8rem);background-color: #fff;left: 8rem;top: 0;}
        .line-one {height: 1rem;width: calc(100% - 8rem);background-color: #fff;left: 8rem;top: 2rem;}
        .line-two {height: 1rem;width: 6rem;background-color: #fff;top: 3rem;right: 0rem;}
        .line-three {width: calc(100% - 8rem);height: 2rem;background-color: #fff;bottom: 0;right: 0;}
    }
}
@keyframes placeHolderShimmer {
    0% {background-position: 100% 50%;}
    100% {background-position: 0 50%;}
}

上面例子,通过拉伸背景图片 background-position,动态 animation 设置背景定位百分比,改变背景定位,从而计算得到图片相对容器的不同偏移值,以此实现了动画的效果。

background-size: 400% 400%;

这里给 background-position 属性设置了两个值,第一个值代表水平位置相对容器的偏移,第二个代表垂直位置相对容器的偏移。

使用百分比设置 background-position 值时,它会执行一个计算实际定位值公式 (container width - image width) * (position x%) = (x offset value),即容器和图片的宽度差乘上设置的百分比定位值,得到的结果就是实际的偏移值,将background-size 的宽度设置为 400% 的其中一个目的就是,这样就会和容器产生宽度差。

最后利用关键帧动画 keyframes,设置 background-positionx坐标 的值从 100%0%

@keyframes placeHolderShimmer {
    0% {background-position: 100% 50%;}
    100% {background-position: 0 50%;}
}

假设容器的宽度是 100px,那么背景图片的宽度就是 400px,利用上边的公式,第一帧的动画中,背景图相对容器偏移的真实值是 (100px-400px)*100% = -300px 最后一帧实际偏移 (100px-400px)*0% = 0 动画的过程实际就是一个 3倍 容器宽的线性背景图片相对于容器的偏移从 -300px0 的变化的过程。

纯css实现纸牌翻转抽奖,实用

平常活动专题中,产品经理会给我们提各种各样的需求,抽奖就是其中的一个主要功能,当然抽奖也分许多形式,纸牌翻转就是其中一个,看一下实际需求中用纯 css 实现的纸牌翻转效果(逻辑就是点击纸牌,请求后端接口,再根据返回的数据进行翻转显示)。

纸牌翻转
<div id="box" class="box viewport-flip">
    <a href="javascript:;" class="list flip in"><div class="img img-before"></div></a>
    <a href="javascript:;" class="list flip out"><div class="img img-back"></div></a>
</div>

关键css代码

.in{animation-timing-function:ease-out;animation-duration:350ms}
.out{animation-timing-function:ease-in;animation-duration:225ms}
.viewport-flip{-webkit-perspective:1000;perspective:1000;position:absolute}
.flip{backface-visibility:hidden;transform:translateX(0)}
.flip.out{transform:rotateY(-90deg) scale(.9);animation-name:flipouttoleft;animation-duration:175ms}
.flip.in{animation-name:flipintoright;animation-duration:225ms}
@keyframes flipouttoleft {
    from {transform: rotateY(0);}
    to {transform: rotateY(-90deg) scale(.9);}
}
@keyframes flipintoright {
    from {transform: rotateY(90deg) scale(.9);}
    to {transform: rotateY(0);}
}

纸牌翻转,通过 animation-name 来选择不同的 keyframes 规则:点击翻转的时候设置顶部的容器为 animation-name: flipouttoleft,底部的容器为 animation-name: flipintoright,来调用不同规则达到翻转抽奖效果;再次点击,则效果相反(可以相当于重新洗牌);纸牌翻转不仅可以用在抽奖,也可以用在一些内容介绍方面、翻书效果等等,可以根据项目需要,开发出适用于项目的效果。

翻书效果:就是做一个样式 .pape,再通过关键帧规则 keyframes,设置 rotateY 动画,如下代码所示:

@keyframes flip-to-left {
    from {transform: rotateY(0);}
    to {transform: rotateY(-180deg);}
}
.pape {
    transform-origin: left center;
    animation: flip-to-left 2s ease-in-out;
}

教你轻松使用css实现滚动列表

活动专题中经常有获奖列表名单的显示滚动等等,由于列表内容比较多,经常会用到滚动的方式来展示,看一下用纯 css 实现的滚动效果。

滚动列表
<div class="roll" id="roll">
    <ul>
        <li><span>06-09</span>[新闻]【重要公告】6月15日版本更新</li>
        <li><span>05-28</span>[新闻]【合服公告】6月9日合服公告</li>
        <li><span>05-25</span>[新闻]【重要公告】5月29日版本更新</li>
        <li><span>05-24</span>[新闻]【合服公告】5月26日合服公告</li>
        <li><span>05-24</span>[新闻]【合服公告】5月25日合服公告</li>
        <li><span>05-14</span>[新闻]【重要公告】5月15日版本更新</li>
        <li><span>05-12</span>[新闻]【合服公告】5月12日合服公告</li>
        <li><span>05-11</span>[新闻]【合服公告】5月10日合服公告</li>
        <li><span>06-09</span>[新闻]【重要公告】6月15日版本更新</li>
        <li><span>05-28</span>[新闻]【合服公告】6月9日合服公告</li>
    </ul>
</div>

关键css代码

.roll ul {
    list-style: none;
    animation: rollList 5s linear infinite;
}
@keyframes rollList {
    0% {margin-top: 0;}
    12.5% {margin-top: 0;}
    25% {margin-top: -40px;}
    37.5% {margin-top: -40px;}
    50% {margin-top: -80px;}
    62.5% {margin-top: -80px;}
    75% {margin-top: -120px;}
    87.5% {margin-top: -120px;}
    100% {margin-top: -160px;}
}

既然我们可以用 css 来实现列表的滚动,那也就是可以用实现图片的轮播滚动,在移动端越来越普及的现在,几句 css 代码实现,通过设置 animation-iteration-countinfinite ,再设置 keyframes 规则,来达到图片轮播滚动的效果:

img {
    animation: move 8s linear infinite normal;
    animation-fill-mode: forwards;
}
@keyframes move {
    0%{transform: translateX(0px);}
    100%{transform: translateX(-2400px);}
}

一个【开始游戏】动画的时代变迁

石器时代,官网开始游戏的动画效果都是通过 flash 的方式来进行交互,美术大佬写 flash 动画交互,前端小哥提供跳转链接,感觉天衣无缝;但是,运营大佬说这个跳转链接要改一下,美术大佬又要重新改 flash,导致这中间的流程非常的繁琐。
flash 制作的开始游戏按钮,可维护性也比较差,也会增加页面内容的大小和请求,影响页面加载。

css3时代,人类在进步,技术也在进步。前端小哥一个人就可以包办一切,从内容到动画到跳转,一个简单的官网开始游戏的动画简直是信手拈来,稳稳的提高了效率。

用animation制作开始游戏动画
<a class="start icon-start" href="#" target="_blank"><span></span></a>

关键css代码

.start:hover span{animation:linear icoBig 1.6s infinite;}
.start:hover span:after{opacity:.6;animation:linear icoBig2 1.6s infinite;}
@keyframes icoBig{
    0%{transform:scale(1)}
    20%{transform:scale(1.05)}
    30%{transform:scale(.93)}
    45%{transform:scale(1.04)}
    60%{transform:scale(1)}
}
@keyframes icoBig2{
    0%{transform:scale(1)}
    20%{transform:scale(1.3);opacity:0}
    100%{transform:scale(1);opacity:0}
}

so easy,上面这段代码实现了一个开始游戏按钮鼠标经过的动画效果;通过设置 过渡次数 animation-iteration-countinfinite(无限),过渡时间 animation-delay1.6s,过渡方式 animation-timing-functionlinear,再配合 keyframes 规则进行转换,在 hover 的时候触发效果。

纯div+css实现多功能又好玩的进度条

项目中我们可能会显示用户vip的晋升等级,或活动专题记录用户领取礼包的天数进度等等,都可以进度条来显示当前的进度。或者,上传图片的进度,下载图片的进,都会用到进度条。看一下用纯 css 的如何模拟进度条效果。

1、正常加载效果

使用 animation-fill-mode 设置为 forwards,再通过 keyframes 规则把动画状态设置到最后一帧的状态,让整个进度条达到 100%

纯css制作进度条
div {
    height: 10px;border: 1px solid;
    background: linear-gradient(rgb(0, 255, 157), #0ff);
    background-repeat: no-repeat;
    background-size: 0;
    animation: move 2s linear forwards;
}
@keyframes move {
    100% {background-size: 100%;}
}

2、延迟效果图

使用 animation-delay 设置延迟时间,延迟可以为负数,延迟表示动画仿佛开始前就已经运行过了那么长时间。
上述进度条例子,原动画用了 2s 是从 0% 加载到 100% 的。如果设置延迟为 -1s。这动画会从 50% 加载到 100%,仿佛已经运行了 1s 一样:

带延迟的进度条
div {
    height: 10px;border: 1px solid;
    background: linear-gradient(#0ff, #0ff);
    background-repeat: no-repeat;
    background-size: 0;
    animation: move 2s -1s linear forwards;
}

3、暂停效果图

css 动画是可以暂停的。属性 animation-play-state 表示动画播放状态,默认值 running 表示播放, paused 表示暂停(鼠标经过模拟暂停):

带暂停的进度条
.div3 {
    height: 10px;border: 1px solid;
    background: linear-gradient(#0ff, #0ff);
    background-repeat: no-repeat;
    background-size: 0;
    animation: move 2s linear forwards;
}
.div3:hover {
    animation-play-state: paused;
}
@keyframes move {
    100% {background-size: 100%;}
}

animation-play-state 还可以与负延迟一起实现特殊的效果,通过设置 --percent,来达到进度的显示,比如进度条插件:

.div4 {
    height: 10px;border: 1px solid;
    background: linear-gradient(#0ff, #0ff);
    background-repeat: no-repeat;
    background-size: 0;
    animation: move 100s calc(var(--percent) * -1s) linear paused;
    --percent: 30;
}
@keyframes move {
    100% {background-size: 100%;}
}

结论

经过以上几个例子的学习,大概的了解了 animation 的用法,随着对 animation 的深入理解,在日常开发中,是可以做一些更有有趣的动画,增加页面的交互性;你可以看看自己之前用 js 写的各种动画,尝试着用 animation 进行修改,相信你一定会有所收获的。

ps:一个不懂的人锤太简单了,吹牛的永远不累

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,001评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,210评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,874评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,001评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,022评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,005评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,929评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,742评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,193评论 1 309
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,427评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,583评论 1 346
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,305评论 5 342
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,911评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,564评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,731评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,581评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,478评论 2 352