6

三种 Loading 制作方案

 3 years ago
source link: https://mp.weixin.qq.com/s?__biz=Mzg2NDAzMjE5NQ%3D%3D&%3Bmid=2247487621&%3Bidx=1&%3Bsn=c6c07d1056a34724158f0317753d7c74
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.

uArmemq.png!mobile

作者: JS_Even_JS

来源:https://segmentfault.com/a/1190000038692080

一、简介

Loading几乎是每个应用都会用到的一个组件。很多组件库都会提供相应的Loading组件,但是有的时候我们可能 需要自定义Loading效果 ,掌握Loading组件制作的基础知识将变得非常必要。Loading主要就是 一个旋转的圆环 ,而旋转部分则比较简单,直接通过CSS动画即可实现,所以 关键部分就是得到Loading的圆环

二、通过border-radius绘制圆环

我们通常让一个元素变成圆形是先 将一个元素设置为长和宽相等的正方形 ,然后 给这个元素设置一个border-radius值为50% 。需要注意的是, border-radius: 50%是让整个正方形元素都变成圆形 ,即 包括边框和内容区 。所以我们可以通过 控制元素边框和内容区的大小将元素的内容区域作为内圆将元素的边框区域作为外圆 ,从而绘制出一个圆环。

<div class="loading-css"></div>
.loading-css {
width: 50px; /*先将loading区域变成正方形*/
height: 50px;
display: inline-block; /*将loading区域变成行内元素,防止旋转的时候,100%宽度都在旋转*/
border: 3px solid #f3f3f3; /*设置四周边框大小,并将颜色设置为浅白色*/
border-top: 3px solid red; /*将上边框颜色设置为红色高亮,以便旋转的时候能够看到旋转的效果*/
border-radius: 50%; /*将边框和内容区域都变成圆形*/
}

此时效果如下:

V32Eru6.png!mobile

圆环效果已经出来了,接下来让圆环旋转起来即可,如:

@keyframes loading-360 {
0% {
transform: rotate(0deg); /*动画起始的时候旋转了0度*/
}
100% {
transform: rotate(360deg); /*动画结束的时候旋转了360度*/
}
}
.loading-css { /*在之前的CSS中加上动画效果即可*/
animation: loading-360 0.8s infinite linear; /*给圆环添加旋转360度的动画,并且是无限次*/
}

2eyyIjI.gif!mobile

二、通过svg来绘制圆环

SVG 意为 可缩放矢量图形Scalable Vector Graphics ),其 使用 XML 格式定义图像<circle> 标签可用来创建一个圆 ,同时 外面必须嵌套一个<svg>标签

<svg viewBox="0 0 50 50" class="loading-svg">
<circle cx="25" cy="25" r="20" fill="none" class="path"></circle>
</svg>
.loading-svg {
width: 50px; /*设置svg显示区域大小*/
height: 50px;
}

<svg>标签的width和height设置的是svg图形可显示区域大小。而viewBox表示的是截取图形的区域,因为矢量图的绘制区域可以是无限大的,具体绘制在哪里根据具体的设置而定 ,比如上面的circle就绘制在圆心坐标为(25,25),半径为20的圆形区域中,而viewBox设置为0 0 50 50,表示截图区域为左上角坐标为(0, 0),右下角坐标为(50,50)的矩形区域内,即 会截取这个区域内的矢量图,然后将截取的矢量图放到svg的可显示区域内,同时会根据svg可显示区域的大小等比例进行缩放,但是截取的图片必须在svg可显示区域内完整显示。


假如,现在讲svg的大小设置为60px,如:

.loading-svg {
width: 60px; /*设置svg显示区域大小*/
height: 60px;
}

如上分析,viewBox截图区域中, 绘制的圆的圆心正好在截图区域的中心 ,所以截图区域四周边框与绘制的圆之间有5px的距离,而圆的半径为20px,所以 比例为1:4 ,现在将svg显示区域变为60px,所以也需要 将截图区域等比例放大并占满整个svg显示区域 ,截图区域经过拉伸后, 圆心位置变为了(30,30) ,即半径变为了30,按1:4比例,半径变为24,外围变为了6,所以整个圆也会跟着变大。

需要注意的时候, <cicle>绘制的圆目前是看不到的 ,因为 没有给画笔设置上颜色 ,如:

.path {
stroke: #409eff; /*给画笔设置一个颜色*/
stroke-width: 2; /*设置线条的宽度*/
}

i6f26fv.png!mobile

此时可以看到绘制出的圆环了。为了给圆环添加转动效果,我们需要绘制带缺口的圆环,后面通过改变缺口的位置大小来实现转动效果,如:

.path {
stroke-dasharray: 95, 126; /*设置实现长95,虚线长126*/
stroke-dashoffset: 0; /*设置虚线的偏移位置*/
}

nQZZ7nB.png!mobile

如图所示,圆环的绘制起点是 在水平方向最右边的那个点 ,然后进行 顺时针 绘制。因为该圆环的周长为2 3.14 20=125.6,约等于126,stroke-dasharray设置了实线(可见部分)长为95,约等于圆的3/4,所以只能绘制到圆环的最高点位置,接下来是126的虚线,但是 圆环周长只有126 ,所以 只能显示31的虚线 。可以看做是一根 无限循环的水平线条 ,实线(-221,0)---虚线(-126,0)--- 目前起点为(0,0) ---实线(95,0)---虚线(221,0)---实线(316,0),然后 让水平线的起点(0,0)位置与圆环的起点位置重合 水平线顺时针沿着圆环绕 即可, 随着stroke-dashoffset起点位置的偏移 左侧的(-126,0)的虚线就可以慢慢显示出来 。当stroke-dashoffset值为 负数 的时候, 上面的线往右拉 ,当stroke-dashoffset值为 正数 的时候, 下面的线往右拉

UZjqEvr.png!mobile

接下来就是添加圆环的转动效果,分别设置三个动画状态,如:

// 0%
{
stroke-dasharray: 1, 126; /*实线部分1,虚线部分126*/
stroke-dashoffset: 0; /*前面1/126显示实线,后面125显示空白*/
}

从圆环最右边作为起点绘制1个像素的距离的实线,接下来绘制126像素的虚线(空白),因为圆周长为126,所以剩余部分全部为空白,如图所示,

ZvIzQrj.png!mobile

// 50%
{
stroke-dasharray: 95, 126; /*实线部分95,虚线部分126*/
stroke-dashoffset: -31px; /*顺时针偏移31/126,即前31/126显示空白,后面3/4显示线条*/
}

从圆环的最右边作为起点,并且顺时针移动31像素,即圆环的1/4,所以实线起点变为了圆环的最底部,实线长度为95像素,即圆环的3/4,如图所示,

ERnUNfB.png!mobile

// 100%
{
stroke-dasharray: 6, 120; /*实线部分6,虚线部分120*/
stroke-dashoffset: -120px; /*最后顺时针偏移120/126,即前120/126显示空白,后面6点显示线条部分*/
}

从圆环的最右边作为起点,并且顺时针移动120像素,所以实线长度仅剩下6像素了,如图所示,

ZZbqMra.png!mobile

给圆环加上动画效果,如:

.path {
animation: loading-dash 1.5s ease-in-out infinite;
}
@keyframes loading-dash {
0% {
stroke-dasharray: 1, 126; /*实线部分1,虚线部分126*/
stroke-dashoffset: 0; /*前面1/126显示实线,后面125显示空白*/
}


50% {
stroke-dasharray: 95, 126; /*实线部分95,虚线部分126*/
stroke-dashoffset: -31px /*顺时针偏移31/126,即前31/126显示空白,后面3/4显示线条*/
}


to {
stroke-dasharray: 6, 120; /*实线部分6,虚线部分120*/
stroke-dashoffset: -120px; /*最后顺时针偏移120/126,即前120/126显示空白,后面6点显示线条部分*/
}
}

QBNJBv7.gif!mobile

为了让Loading动画更加生动细腻,我们还可以给svg标签也加上一个旋转动画,如:

.loading-svg {
width: 50px; /*设置svg显示区域大小*/
height: 50px;
animation: loading-rotate 1.5s infinite ease-in-out; /*给svg也加上一个旋转动画*/
}
@keyframes loading-rotate {
to {
transform: rotate(1turn); // 旋转1圈
}
}

bmuIRjZ.gif!mobile

三、通过iconfont字体图标

我们可 以直接 通过iconfont字体图标代替圆环的绘制直接以字体的形式显示出圆环 ,然后给其加上旋转动画即可,如:

我们可以在iconfont网站上下载喜欢的Loading图案。字体图标下载后, 将解压后的内容拷贝到项目中 ,并 引入其中的iconfont.css到页面中给要显示字体图标的元素加上iconfont类样式 ,字体图标会有一个对应的unicode编码, 通过::before设置content为该unicode编码即可显示对应的字体图标了 ,或者直接在unicode码前加上\&#x,并作为元素内容。

<link rel="stylesheet" href="icon/iconfont.css">
<style>
.icon-loading {
display: inline-block; /*需要设置为行内块元素动画才会生效*/
font-size: 56px;
color: grey;
}
.icon-loading::before {
content: "\e65b"; /*显示字体图内容,值为\unicode*/
}
</style>
<i class="icon-loading iconfont"></i>
<!--或者-->
<i class="iconfont"></i><!--值为unicode-->

eyemiqr.png!mobile

接下来让字体图标旋转起来即可,如:

.icon-loading {
animation: rotating 2s infinite linear;
}
@keyframes rotating {
0% {
transform: rotate(0deg) /*动画起始位置为旋转0度*/
}


to {
transform: rotate(1turn) /*动画结束位置为旋转1圈*/
}
}

MvaqQjv.gif!mobile


About Joyk


Aggregate valuable and interesting links.
Joyk means Joy of geeK