UI设计师福利——SVG动态表情

先放上效果图,再说实现方法。

表情变化

实现的过程比较简单,不过大量用到了我在SVG变形动画里提到的虚拟曲线c0,0,0,0,0,0,需要补充相关知识的移步https://juejin.im/post/59195c22a0bb9f005ff711b2 这篇文章。
这个案例因为用到了一个元素叠加多个动画,所以还是具有代表性的,而且表情元素可以复用(见文章最后)。说一下实现的过程(思路解剖)

1.动静分离

无论何种形式的SVG动图,第一步自然离不开我们的“动静分离大法”。静态图分离出来如下:

静态底图

AI里放到命名为“base”的图层里。这个导出SVG后是可以不用理会的。
这个动效里,涉及的动效元素有三个,左眼、右眼和嘴巴。表情一共有四种,分别为初始的可爱cute,笑smile,悲伤sad以及晕faint,因此,为了获得路径,需要在AI里依次建立这四个表情的图层。对应眼睛的动画有四个,三个表情变化(需要路径变形动画完成)外加旋转动画。对应嘴巴的动画为三个变形动画,如下图所示。


理清动画原理,就可以开工了。
基础代码如下:

<g id="base">
…对应静态底图的代码…
</g>
<g class="emotion">
<path id="left"/>
<path id="right"/>
<path id="mouse"/>
</g>

"left"为左眼路径(由于悲伤sad的左右眼是对称关系,所以没有定义一组动画多次引用);"right"为右眼路径;"mouse"为嘴巴路径。为了方便定义描边属性,我把所有的表情元素放到了一个组里。这样我就可以通过定义CSS属性

.emotion path{fill:none;stroke:#615658;stroke-width:6;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:10}

来赋给所有的表情元素相同的描边效果了。

2.第一个动画的实现

先说眼睛从圆形变成弯弯曲线的过程。首先确定的是,肯定是用了变形动画。弯弯曲线只需要一段路径就可以实现,而圆形的组成路径有四段路径,因此我们需要改造的就是路径少的曲线,方法也很简单,只需要给其增加三个锚点,使其变成四段路径就可以了。另外圆形路径也要用剪刀剪成开放路径。其他要注意的细节调整方法不再详细说了,变形动画的三篇文章里有详细介绍。

眼睛动画1实现方法
@keyframes deformLeft1{
0% {d:path('')}  /*圆形路径*/
100%{d:path('')} /*笑时的弯弯路径*/
}  
#left{animation:deformLeft1 1s ease}

这样就实现了左眼的变形动画。
这个动画过程没有给路径少的曲线直接增加虚拟曲线是因为试过之后效果不好。
嘴巴实现的过程是直线变曲线的过程,AI直接导出的直线一定是下面这样的

<line  x1="" y1="" x2="" y2=""/>

而微微上翘的嘴巴是对应路径标签<path>。我们要做的,就是根据直线起点(x1,y1)终点(x2,y2),(水平直线的纵坐标y1与y2是等值的)将其转化成路径标签,方法也很简单,对应d值Mx1,y1c0,0,0,0,(x2-x1),0

嘴巴动画1实现方法

转化后我们就可以给嘴巴也定义同样的路径变形动画了

@keyframes deformMouse1{
0% {d:path('')} /*直线转化后的路径*/
100%{d:path('')} /*微微上翘的嘴巴的路径*/
}
#mouse{animation:deformMouse1 1s ease}

至此,完成了第一个动画过程。
在路径的改造过程中,请用任意一个方法保留未修改的路径,后面揭晓原因。

3.第二个动画的实现

在由微笑变伤心的过程中,因为表示伤心的眼睛是两条直线折线绘制的,这时导出的标签是<polyline points=""/>,当然了,如果你对贝塞尔曲线比较熟悉的话,完全可以根据坐标值直接来转化成小c开头的路径绘制方法,不熟悉的话就把直线改成带很少很少很少一点点手柄的曲线吧。
我们在上一个动画完成时,为了配合圆形,把眼睛弯弯的一条路径改成了四条,那么到了这里怎么办?难道再把伤心的眼睛也切成四份,好吧,即使这里可以这么操作,那最后晕晕眼睛的螺旋线?那可是N条路径组成的,难道都要以最终数量最多的路径为准,加锚点加到手软抽筋?!
所以说才要保留一份未修改的路径嘛。我们直接用原始的路径来进行改造,因为伤心的眼睛是两条路径,所以给原始路径增加一个锚点,使路径由一条变成两条就可以啦。无论是路径上添加了几个锚点,视觉上无差,所以动画才能实现无缝拼接。

动画2眼睛变形动画实现
@keyframes deformLeft2{
0% {d:path('')} /*变身两条路径的弯弯的眼睛*/
100%{d:path('')}/*悲伤的眼睛*/
}

我们是给一个元素附加多个动画,语法如下:

#left{animation:deformLeft1 1s ease,deformLeft2 1s ease 1s}

后面依次累加,用逗号,隔开就可以了,因为第一个动画我设置了1s的完成周期,所以第二段动画deformLeft2最后的1s表示延迟1s后执行。
之所以没有采用定义多个关键帧的方式来实现多个变形动画,一个是考虑到最后的螺旋线路径数量太多,当然,最重要的是每一个变形动画完成我都采用了ease函数,使每段动画增加一点停顿感。
对于第二个动画而言,嘴巴的变形很简单,就不再赘述了。

4.第三个动画的实现

这个动画变形过程路径的数量差异比较大,当然了,掌握了奇淫技巧的我们怎会被这个问题难住,悲伤的眼睛只有两段路径,而螺旋线眼睛足足有七段路径,差的五段路径让我们万能的虚拟曲线c0,0,0,0,0,0来实现就好啦,so easy!

动画3眼睛变形动画实现

这里因为圈圈太多,所以描边粗细stroke-width也顺便改一下。

@keyframes deformLeft3{
0% {d:path('')} /*后面补5个c0,0,0,0,0,0*/
100%{d:path('');stroke-width:4}
}

继续往后面追加定义的动画deformLeft3咯

#left{animation:deformLeft1 1s ease backwards 1s,deformLeft2 1s ease 1s,deformLeft3 1s ease 2s forwards}

这里唯一的不同是我加了 forwards,这个属性值对应的动画属性animation-fill-mode,用来定义结束后动画的状态,forwards为动画完成后保持最后一个属性值。
为什么要添加这个属性呢?因为从一开始,我们的眼睛和嘴巴元素在代码中是没有体现的,全凭动画属性绘制路径来添加,而眼睛的下一组动画为旋转,如果变成圈圈眼之后不定义结束后动画状态,在旋转动画时,动画元素为空。
嘴巴的变形动画也星对比较简单,只需要给表示悲伤的向下弯曲的路径增加锚点,补齐撇嘴对应的路径数量就好啦。

5.第四个动画的实现

眼睛的旋转动画实现起来很简单,不过要定义一下旋转的中心点transform-origin,不定义为默认以画布左上角原点为中心点。

@keyframes deformLeft4{
0%{transform:rotate(0); transform-origin:}
100%{transform:rotate(360deg); transform-origin:}
}

这个值直接通过AI获得。


旋转中心点

因为眼睛旋转的时候,嘴巴是不动的,所以在调用嘴巴的最后一个动画时也要叫上forwards属性。

#mouse{animation:deformMouse1 1s ease,deformMouse2 1s ease 2s,deformMouse3 1s ease 3s forwards}

好了,以上就是全部动画的实现过程,那么做完这么一套动态变化的表情能用在什么地方呢?这就是我们“动静分离大法”玄妙之处,你只需要改一下静态底图,就能获得各种不同效果。而且任意改颜色,简直爽歪歪。

6.不同底图的应用

比如我曾经做过一组方形的表情图标,是下面这种

方形表情图

现在,我想让第一排第二个小心心图标变身动态表情,只要把我们的动画效果叠加上,再通过SVG的width、height、viewBox属性值的设定进行缩放,就得到了下面这个方形表情:

方形表情

其他应用就不再一一举例了,就酱。

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

推荐阅读更多精彩内容