抽奖动画 – 鲤鱼跳龙门


pageClass: home-page-class

鲤鱼跳龙门动画

1. 需求

年中618营销活动要求做一个鲤鱼跳龙门的动画,产品参考了支付宝上的一个动画,要求模仿这个来做一个类似的动画。产品提供的截屏视频如下:
抽奖动画 - 鲤鱼跳龙门

图1
从这个视频里得到的信息,我们可以把动画分解一下:

  • 321倒计时结束,动画开始播放。
  • 小河背景向下滚动,看上去小鱼在不停的向上游动,其实小鱼固定在屏幕中间位置。
  • 金币从屏幕顶部掉落,掉入小鱼的嘴里的时候金币消失,金币在掉落同时金币在旋转。
  • 用户点击“狂点”按钮,该按钮四周会出现一个光晕,并且变大变小。
  • 金币掉落完毕,出现龙门,小鱼跑到龙门上方。
  • 播放动画同时顶部有一个时钟倒计时,从6.18倒数到0。

从视频上看,有一部分用css动画实现起来比较麻烦,例如,金币掉落完成之后,小鱼要转身,从背对观众变成面向观众,同时大小在变化,这些常见的css动画没法完全复原,初步判断这些是使用其他动画库来实现的,普通的css动画无法实现。
我们事先要把这些告知产品,不然最后实现起来非常麻烦,因为本身活动项目开发时间非常短。

2. 整体思路

2.1 三二一倒计时

三二一倒计时这个很简单,直接用文字显示的话不太美观,UI提供了三个4个张图片,我们可以按照数字分别命名3.png,2.png,1.png,0.png,然后使用setTimeInterval给变量做递减就可以了。倒计时结束后静态的小鱼变成一个游泳的小鱼,这里是一个gif图片,所以直接使用切换图片就可以了。

2.2 河流

小鱼向下游动,相对而言可以让小河向上滚动,在游戏背景上让河流绝对定位,设置position,初始bottom为0,播放动画,变为top为0,这样看上去是小鱼向上游动。

2.3 金币坠落

金币坠落也是使用绝对定位的方式,初始状态top是负值,隐藏在屏幕最上方,下落过程中逐渐变小,并且有旋转的动作,这里使用rotateY来控制旋转。待金币坠落到小鱼嘴的位置的时候,金币消失,模拟小鱼吃掉金币,这里设置大小为0,使用scale来缩放图片实现。

2.4 “狂点”按钮

用户点击狂点按钮时,小鱼的背后出现一个光晕,它由大变小,再由小变大,看上去小鱼是在加速,这个交互可以让动画更加生动。点击狂点按钮是,这个按钮自己本身也有一个由小变大,再由大变小的过程。

2.5 跳龙门

整个跳龙门的时间控制在6.18秒内,也就是河流滚动的时间也是6.18秒,结束后背景上面出现一个龙门图片,小鱼跳出屏幕。龙门图片最初设置opacity是0,跳出后是1,这样自然过度,如果使用显示&影藏来控制,看上去有点突兀。

2.6 时钟

最后顶部的倒计时时钟就很简单了,只要控制一个数字从6.18递减到0就满足需求了。

3. 实现过程

3.1 布局

整个布局思路是绝对定位,整个背景fix定位在整个屏幕上,其他的元素使用absolute定位来固定位置。注意背景内的元素是absolute定位,都是居中显示,这里使用常用的方式left: 50%; margin-left: -(width/2);来设置左右居中。布局如下图1:
抽奖动画 - 鲤鱼跳龙门

图2 布局
初始状态是这样,注意狂点按钮覆盖在小鱼上方,这个可以使用不同的z-index来实现,还有一些隐藏的元素,例如:金币图片,龙门图片,动画未开始的时候他么是隐藏的。

html代码如下:

<!-- 跃龙门游戏 --> <div class="dragon-gate-game" @touchmove.prevent.stop @mousewheel.prevent>   <!-- 321倒计时 -->   <mask-dialog ref="refCountdown">     <div class="count-down">       <img v-show="countDown == 3" class="coupon-btn" :src="require('../assets/images/animation/3.png')" alt="" />       <img v-show="countDown == 2" class="coupon-btn" :src="require('../assets/images/animation/2.png')" alt="" />       <img v-show="countDown == 1" class="coupon-btn" :src="require('../assets/images/animation/1.png')" alt="" />       <img v-show="countDown == 0" class="coupon-btn" :src="require('../assets/images/animation/0.png')" alt="" />     </div>   </mask-dialog>    <!-- 跳龙门 -->   <div class="jump">     <!-- 时钟倒计时 -->     <div class="clock">{{ game.clock }}</div>     <!-- 福字 -->     <img v-for="(img, i) in game.blessing" :key="i" :src="img" class="blessing" alt="" />     <!-- 小鱼 -->     <div :class="[fish.name]" id="fish">       <img :src="fish.src" alt="" class="img-fish"/>       <img src="../assets/images/animation/bg-aureole.png" alt="" class="backdrop">     </div>     <!-- 狂点按钮 -->     <img src="../assets/images/animation/btn1.png" :data-name="fish.name" alt="" class="btn-click" @click="jump" />     <!-- 龙门 -->     <img src="../assets/images/animation/bg-door.png" alt="" class="door" />     <!-- 河 -->     <img src="../assets/images/animation/bg-animation.jpg" alt="" class="river" />   </div> </div> 

给背景div设置禁止滚轮滚动,禁止拖放,防止它出现滚动条,配合fix定位,固定在屏幕上。其他的元素使用absolute定位,这里有两个兼容性问题要注意:

  • 注意元素定位使用bottom,不能使用top,防止部分浏览器底部工具栏遮挡"狂点"按钮,其他的元素也使用bottom。
  • 注意321倒计时不能使用js动态切换图片的路径,而是使用v-show判断,否则切换浏览器的时候在低端浏览器上会出现屏幕闪烁的现象,估计是造成页面重绘了。
    css代码如下:
.dragon-gate-game {   position: fixed;   top: 0;   left: 0;   bottom: 0;   right: 0;   z-index: 1;   .loading,  .dragon-gate-game, .count-down, .jump {     width: 100%;     height: 100vh;   }   .count-down  {     @include flex(center, center, row, nowrap);     .coupon-btn {       width: 400px;     }   }   .jump {     position: relative;     overflow: hidden;     .river, .clock, .water, .fish, .swim-fish, .btn-click, .door, .blessing {       position: absolute;       bottom: 0;       left: 0;       right: 0;     }     .clock {       width: 239px;       height: 64px;       background: 34px center / 44px 44px no-repeat url("../assets/images/animation/icon-clock.png"), #000000;       opacity: 0.4;       border-radius: 32px;       top: 120px;       left: 50%;       margin-left: -119px;       z-index: 3;       font-size: 36px;       font-weight: 400;       color: #FFFFFF;       line-height: 64px;       text-indent: 44px;     }     .river {       z-index: 1;       width: 750px;       // height: 14039px;     }     .door {       width: 750px;       height: 960px;       z-index: 3;       opacity: 0;     }     .water {       z-index: 2;       width: 750px;       height: 467px;     }     .fish, .swim-fish {       position: relative;       z-index: 3;       left: 50%;       img {         position: absolute;         width: 100%;         left: 50%;         margin-left: -50%;       }       .img-fish {         z-index: 3;       }       .backdrop {         position: absolute;         z-index: 2;         left: 50%;         margin-left: -50%;         opacity: 0;       }     }     .fish {       width: 259px;       margin-left: -129px;       top: 700px;     }     .swim-fish {       width: 259px;       margin-left: -129px;       top: 600px;     }      .btn-click {       z-index: 4;       width: 240px;       left: 50%;       margin-left: -120px;       bottom: 200px;       // top: 1000px;       // animation: .4s linear 1s infinite alternate btnZoom;     }     @keyframes btnZoom {       from {         transform: scale(0.8);       }       to {         transform: scale(1.1);       }     }      .blessing {       width: 80px;       margin-left: -40px;       z-index: 3;       left: 50%;       top: -140px;     }   } } 

3.2 倒计时

data中定义变量countDown,初始值是3,使用setInterval来递减这个变量,这个逻辑相对来说比较简单,代码如下:

//倒计时 countDownClock() {   this.$refs.refCountdown && this.$refs.refCountdown.show()   this.timerInterval = null   this.timerInterval = setInterval(() => {     this.countDown--     if (this.countDown < 0) {       clearInterval(this.timerInterval)       this.timerInterval = null       this.$refs.refCountdown &&this.$refs.refCountdown.hidden()       this.countDown = 3       // 切换动画鱼       // this.fish = this.game.swimFish       // 播放动画       // this.playAnime()     }   }, 1100) } 

倒计时我们也放在一个透明蒙层里,最后两句切换动画鱼和静态鱼图片和播放小河,金币动画,暂时注释了,来看看效果:
抽奖动画 - 鲤鱼跳龙门

图3 倒计时

3.3 播放动画

开始播放动画时,首先把小鱼切换成那个gif图片,让小鱼动起来,这里在data数据中定义了一些数据。

data(){   return {     pageShow: '',                       //页面显示     percentage: '2%',                   //进度条变化     countDown: 3,                       //321倒计时     timerInterval: null,                //计时器,用于清除     fish: {},                           //当前显示小鱼     game: {       finish: false,                    //是否已完成,回调后不能再点       clock: 6.18,                      //时钟倒计时       duration: 6180,                   //动画持续时间       blessingOpacity: '1',             //显示金币       fish: {name: 'fish', src: require('../assets/images/animation/bg-fish.png')},               //小鱼图片       swimFish: {name: 'swim-fish', src: require('../assets/images/animation/fish-swim.gif')},    //游泳的小鱼       blessing: Array(20).fill(require('../assets/images/losing-lottery/text-blessing.png')),      //金币       clickCount: 0,                  //点击次数     }   } } 

切换小鱼只需要上面注释的那句就可以了:this.fish = this.game.swimFish,然后执行下面的this.playAnime()来播放动画。
这里还是使用anime.js动画库来播放,首先让小河向上滚动,同时让时钟从6.18倒数到0,同时让金币坠落,这三个动画前两个动画的时间是一致的,都是6.18秒,金币坠落的动画需要自己来估计,这里使用一个延迟,交错动画,延迟时间6.18*0.12,交错时间200毫秒,同时这个还和金币个数有关系,如果金币太少,动画后半部分没有金币坠落,金币太多6.18秒过了金币还没有落完,这都不是我们想要的结果,我们设置金币总共个数是20。
6.18秒结束时要让龙门浮出,小鱼跳出龙门,龙门浮出通过设置opacity来实现,小鱼跳出,通过translateY实现,最后看代码如下:

playAnime() {   let tl = anime.timeline()   //动画   tl.add({     //河流流动     targets: '.river',     easing: 'linear',     duration: this.game.duration,     top: 0,     complete: () => {       this.game.finish = true       this.$emit('animeFinish', this.game.clickCount)     }   }).add({     targets: this.game,     clock: 0,     easing: 'linear',     round: 100,     duration: this.game.duration,   }, 0).add({     //金币下落     targets: '.blessing',     easing: 'linear',     delay: anime.stagger(200, {start: this.game.duration * 0.12}),     keyframes: [       {top: '30%', opacity: '1', scale: 0.8},       {top: '45%', opacity: '0', scale: 0.5, rotateY: '360deg'}     ],   }, 0).add({     //龙门浮出     targets: '.door',     easing: 'linear',     delay: 200,     opacity: 1   }).add({     //鱼跳出去     targets: '#fish',     // translateY: -100,     translateY: -550,     duration: 1000   }) } 

结合data数据来看,前两个动画持续时间都是this.game.duration也就是6.18,金币坠落的动画需要我们调试,这里还使用了关键帧,动画进度是30%的时候,金币透明度是1,大小为原始大小的0.8倍,进度为45%的时候opacity是0,scale是0.5,沿Y轴旋转360度。金币坠落完成后龙门浮出,小鱼跳过龙门。这两个动画相对简单,一个是通过opacity来显示,一个通过translateY来隐藏。最后来看动画效果。
抽奖动画 - 鲤鱼跳龙门

图4 动画

3.4 用户点击

用户点击狂点按钮时有两个交互,一个是狂点按钮本身会有一个变大变小的过程,其次小鱼背后会出现一个光晕,这两个动画是每点击一次才播放一次的。每点击一次要纪录一下点击次数,这个调用抽奖接口的时候要用到,还有要判断动画是否已经结束,结束之后点击是没有什么效果的,当然这不是这里实现动画的关键。看下面的代码:

jump() {   let tl = anime.timeline()   if (this.game.finish) return   this.game.clickCount++   console.log('this.game.clickCount')   tl.add({     targets: '.backdrop',     duration: 1000,     keyframes: [       {opacity: 0.2},       {opacity: 0.5},       {opacity: 0.8},       {opacity: 1.2},       {opacity: 0.8},       {opacity: 0.5},       {opacity: 0.2},       {opacity: 0},     ]   }).add({     targets: '.btn-click',     easing: 'linear',     duration: 200,     keyframes: [       {scale: 0.9, opacity: 0.9},       {scale: 0.8, opacity: 0.8},       {scale: 0.7, opacity: 0.7},       {scale: 0.6, opacity: 0.6},       {scale: 0.8, opacity: 0.5},       {scale: 0.9, opacity: 0.4},       {scale: 1, opacity: 0.6},       {scale: 1.1, opacity: 0.8},       {scale: 1, opacity: 1}     ]   }, 0) } 

小鱼图片和它背后的光晕都是使用绝对定位,但是小鱼的z-index要比光晕大,这样看起来光晕是在小鱼的下方。这两个动画都使用了关键帧来增强效果。点击效果图如下:
抽奖动画 - 鲤鱼跳龙门
图5 按钮点击

最后就是调用接口,根据接口弹出中奖结果了,这和动画无关,只需要传一个参数,点击狂点按钮的次数。最后看一下整体效果,如下图6:
抽奖动画 - 鲤鱼跳龙门

图6 完整动画

4.总结

整个鲤鱼跳龙门动画已经介绍完,这个动画要考虑的元素很多,有小鱼,小鱼背后的光晕,龙门,金币,倒计时,小河等等,整个动画是由一个一个的小动画组合而成,只要把要考虑的细节考虑清楚,实现起来还是不难的。

5.参考

  1. animate https://www.animejs.cn/
发表评论

相关文章