手记

VUE+WebPack游戏设计:实现两张扑克牌的逐对厮杀特效

本节,我们开始进行两张扑克牌对战时的情景设计,当用户从底部选中一张扑克牌后,系统需要产生一张对手扑克牌,这样两张牌才能相互较量,接下来我们先实现如何引入敌对扑克牌。

在gamescenecomponent.vue中,我们先添加敌对扑克牌的定义,在标签template中,添加如下代码:

<div class="card opponent" :class="{'out' : cardOpponentOut, 'in': !cardOpponentOut}"> 
       <div class="front face"></div>
       <div class="back face">back</div></div>

接着在script标签中添加相关变量的定义,以及控制敌对扑克牌的逻辑代码,首先添加几个变量定义:

<script>
  import Constant from './constant'
  export default {
    data () {      return {
      ....
      selectedCardPower: 0,      opponentCardPower: 100,      cardOpponentOut: true
      }
      ....
      methods: {
      ....
      cardAClick () {        this.cardASelected = true
        this.cardAFlipped = false
        this.cardBOut = true
        this.cardCOut = true

        this.beginBattleAnimation()
      },
      cardBClick () {        this.cardBSelected = true
        this.cardBFlipped = false
        this.cardAOut = true
        this.cardCOut = true

        this.beginBattleAnimation()
      },
      cardCClick () {        this.cardCSelected = true
        this.cardCFlipped = false
        this.cardBOut = true
        this.cardAOut = true

        this.beginBattleAnimation()
      },
      randomizerPower () {        return Math.round(Math.random() * 60) + 40
      },
      beginBattleAnimation () {        this.cardOpponentOut = false
      }
   }
}
  <script>

selectedCardPower用来表示选中扑克牌的能量值,opponentCardPower用来表示敌对扑克牌的能量值,cardOpponentOut用来控制敌对扑克牌是否在页面上出现,它的值设置成false的话,它就会出现在页面上。

beginBattleAnimation被调用时,他把变量cardOpponentOut设置成false,这样敌对扑克牌就会出现在页面上,当用户选择任意一张扑克牌时,该函数就会调用,也就是说用户选中一张扑克牌后,程序立马将敌对扑克牌显示在页面上,与用户选择的扑克牌并排放在一起,上面代码完成后加载页面,进入游戏场景,点击选择一张扑克牌后,情形如下:


这里写图片描述

右边是用户选中的扑克牌,左边是用户选择后,程序将敌对扑克牌显示到页面上,敌对扑克牌的显示具有动态效果,具体请参看视频。

实现对战特效
当两张对手牌出现在页面中间后,扑克牌间的厮杀就开始了,玩家选择的牌先发送一个冲击波打击对手牌,接着对手牌也发生一个冲击波打击玩家选择的牌。我们先看看这两个冲击波是如何实现的。

首先在html代码部分增加冲击波的定义:

<div class="card opponent" :class="{'out' : cardOpponentOut, 'in': !cardOpponentOut, 'shake' :opponentCardShake}"> 
       <div class="front face"></div>
       <div class="back face">back</div></div>
     <div class="blaze toward-left" :class="{'attack' : blazeAttackLeft}"></div><div class="blaze toward-right"></div>

冲击波本质上是两个静态图片在页面上实现的动态效果:


这里写图片描述

代码中blazeAttackLeft是定义在组件中的变量,如果这个值是true,那么上面的div控件就会具备attack属性,一旦具备这个属性,div控件就可以实现css定义的动画效果。

利用最新的CSS3标准,我们可以实现很多动画特效,在style标签中,我们先添加以下代码:

<style scoped>@keyframes shake {
    0%   {transform: translate3d(0, 0, 0);}
    20%  {transform: translate3d(-5%, 0, 0);}
    40%  {transform: translate3d(5%, 0, 0);}
    60%  {transform: translate3d(-5%, 0, 0);}
    80%  {transform: translate3d(5%, 0, 0);}
    100% {transofrm: translate3d(0, 0, 0);}
  }.card.shake{animation: shake 300ms ease-out}</style>

上面的CSS代码定义了一系列的变换,一系列的变换连接在一起就变成了动画,translate3d(x,y,z)定义了元素沿着x,y,z三个坐标轴上进行变换,在上面的一系列变换中,y,z两个坐标轴全是0,也就是说元素只会沿着x轴变换,translate3d(-5%,0,0)表示元素先向左挪动相应位置,translate3d(5%,0,0)表示元素向右移动相应位置,一系列沿着x轴方向上的作用移动变换会使得css所作用的元素产生一种左右颤抖的效果,具体请参看视频
当我们把变量opponentCardShake设置成true时,shake属性就会添加到'card opponent'这个div上,于是上面定义的shake动画就会作用到敌对扑克牌上。

接着继续添加冲击波的CSS定义,还是在style标签里,添加一下代码:

.blaze {    position: absolute;    bottom: 300px;    width: 50px;    height: 50px;    opacity: 0;    animation-timing-function: ease-out;    animation-duration: 1000ms;
  }  .blaze.toward-left {background-image: url(../../static/images/blaze-left.png);}  .blaze.toward-right {background-image: url(../../static/images/blaze-right.png);}

  @keyframes blaze-toward-left {
  0%, 20% {opacity: 1; transform: translate3d(300px, 0, 0)}
  80% {transform: translate3d(100px, 0, 0);}
  100% {opacity: 0; transform: translate3d(100px, 0, 0);}
  }  .blaze.toward-left.attack {animation-name: blaze-toward-left;}  .blaze.toward-right.attack {animation-name: blaze-toward-right;}

我们看看blaze-toward-left所定义的变化,它首先使得图片blaze-left.png出现在x坐标轴300px的地方,接着向左移动一直到100px的地方,然后他的透明度变成0,也就是消失看不到了,这个效果就类似于页面右边的扑克牌放出了一个指向左边的冲击波,冲击波从右向左移动,抵达左边扑克牌的位置后消失,此时左边扑克牌触发shake变化,于是扑克牌产生出一种被击打后左右颤抖的效果。

接下来我们需要使用js实现整个动画流程,这是整个项目的难点所在。我们要实现的效果是,用户从底部选择一张扑克牌后,选中的牌显示在界面的右边,然后敌对扑克牌出现在坐标,一旦敌对扑克牌出现后,右边扑克牌发出一个blaze-left.png表示的冲击波,冲击波打中左边扑克牌后,扑克牌产生一个左右摇摆的颤抖效果。

我们注意到,一个html元素能够产生两种特效,一种叫transform,一种叫animation,后者是由一系列前者组成的。由于我们现在需要做的是一种特效完成后,由程序接着触发另一种特效,这就需要我们的程序知道特效在哪个时刻完成,好在浏览器给我们提供了相应机制,当一个元素完成transform或animation之后,浏览器就可以通知我们的js程序。

当一个元素完成transform变换时,它会发出一个消息叫webkitTransitionEnd,当元素完成animation变换时,它会发出一个消息叫webkitAnimationEnd,我们只要监听这两个消息,然后才行相应动作就好。当前能产生相应变换的只有两个元素,一个是属性为'card opponent'的div,另一个是属性为'blaze toward-left'的div。所以我们的代码要监听这两个元素所发出的相应消息。

我们先在组件中添加相关变量定义:

<script>
  import Constant from './constant'
  export default {
    data () {      return {
      ....
        opponentCardObject: null,        blazeTowardLeftObject: null,        blazeAttackLeft: false,        opponentCardShake: false,        transitionState: ''
      }
     ....
  }

其中opponentCardObject将用来获得属性为'card opponent'的div实例,blazeTowardLeftObject将用来获得属性为'blaze toward-left'的div实例。我们在组件的mounted调用中添加如下代码:

mounted () {
      ....
      
      this.opponentCardObject = document.querySelector('.card.opponent')      this.blazeTowardLeftObject = document.querySelector('.blaze.toward-left')
    }

当敌对扑克牌要出现在页面左边时,代码需要把变量cardOpponentOut设置成false,然后'in'属性就会添加到属性为'card opponent'的div元素上,我们在看看css定义的in属性:

.card.player.in {    transform: translate3d(0, 0, 0);
  }

in属性对应的是一个transform变换,也就是说div具备了in属性后,就会执行上面定义的变换,div执行上面的变换后,它就出现在页面上了。前面我们提到过transform变换结束后,元素会发出一个webkitTransitionEnd消息,所以只要我们程序监听到div发出这个消息时,我们就可以判定左边扑克牌出现在页面上了。因此在组件的methods区域添加下面代码:

handleTransitionEnd (htmlObj) {        var listener = function (e) {
          e.target.removeEventListener('webkitTransitionEnd', listener)          this.handleTransitionEvent(e)
        }.bind(this)

        htmlObj.addEventListener('webkitTransitionEnd', listener)
      }

只要我们执行handleTransitionEnd(this.opponentCardObject),那么程序就可以监听div发出的webkitTransitionEnd消息,这个消息一旦监控到,里面定义的listener函数会被调用,然后他会调用组件的handleTransitionEvent接口来处理消息。我们以同样的方式来监控元素发出的webkitAnimationEnd消息,在组件中添加如下代码:

handleAnimationEnd (htmlObj) {        var listener = function (e) {
          e.target.removeEventListener('webkitAnimationEnd', listener)          this.handleTransitionEvent(e)
        }.bind(this)

        htmlObj.addEventListener('webkitAnimationEnd', listener)
      }

接着修改beginBattaleAnimation代码:

beginBattleAnimation () {        this.handleTransitionEnd(this.opponentCardObject)        this.transitionState = Constant.OPPONENT_CARD_TRANSITION_END        this.cardOpponentOut = false
      }

this.handleTransitionEnd(this.opponentCardObject)作用是监听opponentCardObject对象发出的webkitTransitionEnd消息,然后将变量transitionState设置成Constant.OPPONENT_CARD_TRANSITION_END,然后将cardOpponentOut设置成false,这样opponentCardObject对应的div元素会添加上in属性,于是他就会执行in属性定义的transform变换,一旦变换完成后,组件的handleTransitionEvent接口就会被调用,我们看看该接口的实现,在组件中添加如下代码:

handleTransitionEvent (e) {        switch (this.transitionState) {          case Constant.OPPONENT_CARD_TRANSITION_END:            if (this.cardOpponentOut === false) {              this.transitionState = Constant.BLAZE_TOWARD_LEFT_ANIMATION_END              this.blazeAttackLeft = true
              this.handleAnimationEnd(this.blazeTowardLeftObject)
            }            break
          case Constant.BLAZE_TOWARD_LEFT_ANIMATION_END:            this.opponentCardShake = true
            break
        }
      }

当handleTransitionEvent被调用,而且transitionState的值是Constant.OPPONENT_CARD_TRANSITION_END,这就表明opponentCardObject对应的div对象刚完成了一个transform变化,如果变量cardOpponentOut的值是false,我们就确定它刚完成了属性in定义的变换,也就是说敌对扑克牌在出现在页面上了。

然后我们把transitionState的值变为Constant.BLAZE_TOWARD_LEFT_ANIMATION_END,然后将blazeAttackLeft属性设置为true,于是attack属性就会添加到属性为'blaze toward-left'所对应的div上,于是该元素就会执行attack所定义的animation变换,this.handleAnimationEnd(this.blazeTowardLeftObject)让我们的代码监听blazeTowardLeftObject元素发出的webkitAnimationEnd消息,一旦这个消息被监控到后,handleTransitionEvent又再次被调用,它执行时发现transitionState的值是Constant.BLAZE_TOWARD_LEFT_ANIMATION_END,此时我们就可以确定blazeTowardLeftObject所对应的div元素,也就是属性为'blaze toward-left'的div元素完成了CSS属性attack所定义的动画特效,也就是说敌对扑克牌被冲击波打到了,因此我们就可以把oppoentCardShake设置成true,一旦设置后,shake属性就会添加到属性为'card opponent'的div元素上,因此该元素就会执行CSS属性shake所定义的动画特效,也就是左右颤抖的效果。

最后我们在Constant组件里添加如下代码:

script>  import Vue from 'vue'
  export default {
  ...
  OPPONENT_CARD_TRANSITION_END: 'opponent_card_transition_end',  BLAZE_TOWARD_LEFT_ANIMATION_END: 'blaze_left_animation_end'}

完成上面代码,然后加载到浏览器后,运行起来可以看到如下效果:

这里写图片描述


一个冲击波从右边发出,打到左边扑克牌后,左边的牌发出一个左右颤抖的动画效果,



作者:望月从良
链接:https://www.jianshu.com/p/3aaa95c61007


0人推荐
随时随地看视频
慕课网APP