EVA.js学习笔记(五)实践fappy-bird游戏(中)

221 阅读4分钟

EVA.js学习笔记(五)实践fappy-bird游戏(中)

# EVA.js学习笔记(四)实践fappy-bird游戏(上)

# EVA.js学习笔记(五)实践fappy-bird游戏(中)

# VA.js学习笔记(六)实践fappy-bird游戏(下)

创建小鸟

1、创建小鸟

创建小鸟的时候用到了SpriteAnimation这个组件

这组件在官方精灵图教程中并未提及,在API文档里有,但是没有写描述。

speed表示图片播放间隔的毫秒数。

 createBird() {
    const bird = new GameObject("bird", {
        size: { width: 88, height: 66 },
        position: {
          x: 100,
          y: this.sceneHeight / 2,
        },
        anchor: {
          x: 0,
          y: 0,
        },
        origin: {
          x: 0.5,
          y: 0.5,
        },
      });

      const anim = bird.addComponent(
        new SpriteAnimation({
          resource: "bird",
          speed: 500,
        })
      );
      const physics = "";

      return { bird, physics };
    },
   
 }

2、小鸟添加物理属性

Eva.js 基于 Matterjs 的物理引擎,目前只接入了小部分内容,在官方文档中有一定的描述

在使用的时候需要安装

npm install @eva/plugin-matterjs

然后在game中配置, gravity代表重力

import {Physics} from '@eva/plugin-matterjs';
// 1.安装物理引擎后引入
import {PhysicsSystem, Physics, PhysicsType} from '@eva/plugin-matterjs';

// 2.在Eva.js中注册插件
const game = new Game({
  ...
  systems: [
  ...
  
    new PhysicsSystem({
      resolution: 2, // 保持RendererSystem的resolution一致
      // isTest: true, // 是否开启调试模式
      // element: document.getElementById('game-container'), // 调试模式下canvas节点的挂载点
      world: {
        gravity: {
          y: 2, // 重力
        },
      },
      mouse: {
        open: true,
      },
    }),
  
  ],
});

\

添加到小鸟对象上

bodyOptions中的内容都来自于matter.js的body属性,可以去看matter.js的文档。

const physics = bird.addComponent(
        new Physics({
          type: PhysicsType.RECTANGLE,
          bodyOptions: {
            isStatic: false, //是否是静态,如果是静态,则质量和密度是无限大
            // restitution: 0, //回复系数
            frictionAir: 0.1,//空气阻力
            friction: 0.06, 
            frictionStatic: 0.3,
            force: {
              x: 0,
              y: 0,
            },
          },
          stopRotation: true,
        })

添加成功后,小鸟就会落下,因为之前在地面上也增加了物理组件,所以小鸟会停在地面上,这时候会触发一个碰撞事件

目前支持的碰撞事件 collisionStart collisionActive collisionEnd
当小鸟和地面接触的时候,就会触发下面方法。

一旦发生碰撞,我们就把方法抛出去,以便我们做后面的操作。

physics.on("collisionStart", () => {
    this.game.emit("game_over");
});

\

由于我们刚开始不需要小鸟,要点击了开始按钮后小鸟才创建,所以个将这段代码封装在方法里,等到调用开始游戏再添加小鸟。

3、点击屏幕小鸟Jump

思路: 使小鸟往上跳一下,避免落地,如果使用了gameObject的Tranform来改变Y的位移,是可以做到的,但是小鸟会瞬间移动到上面一个位置,并不会有一个从上到下的效果,所以需要配合过度动画来实现,比较麻烦。

思路2: 我们用另一种方法,因为物体添加了物理属性,则存在了一个力的属性,我们可以向物体施加一个向上的力,从而是小鸟跳一下。

既然需要点击屏幕,则需要在屏幕上增加点击方法,然后获取到小鸟对象里的物理对象physics

  const evt = this.game.scene.addComponent(
        new Event({
          type: HIT_AREA_TYPE.Rect,
        })
      );
  //点击屏幕跳跃
  evt.on("tap", () => {
     if (this.over) return;
     if (this.start && this.birdObject.physics.body) {
       //-0.62表示往上增加0.62的力,力的大小与物体的SIZE和密度有关
          this.birdObject.physics.body.force.y = -0.62;
      }
  });

创建游戏结束画面

这里和创建readyBox一样

1、创建overBox

   const overBox = new GameObject("overBox", {
        size: { width: 320, height: 80 },
        position: {
          x: 128,
          y: this.sceneHeight / 3,
        },
      });

2、创建GameOver标题、得分、重来

	 const overScore = new GameObject("overScore", {
        size: { width: 179, height: 79 },
        position: { x: 180, y: 320 },
        anchor: {
          x: 0,
          y: 0,
        },
      });

      overScore.addComponent(
        new Text({
          text: "得分:0",
          style: {
            fontFamily: "Arial",
            fontSize: 36,
            fontStyle: "italic",
            fontWeight: "bold",
            fill: "#fff",
            stroke: "#000",
            strokeThickness: 5,
            dropShadow: true,
            dropShadowColor: "#000000",
            dropShadowBlur: 4,
            dropShadowAngle: Math.PI / 6,
            dropShadowDistance: 6,
            wordWrap: true,
            wordWrapWidth: 400,
            breakWords: true,
          },
        })
      );


      const overTitle = new GameObject("overTitle", {
        size: { width: 483, height: 101 },
        position: {
          x: 0,
          y: 0,
        },
      });
      overTitle.addComponent(
        new Sprite({
          resource: "over",
          spriteName: "game_over.png",
        })
      );

      const play = new GameObject("gameOverPlay", {
        size: { width: 126, height: 86 },
        position: {
          x: 187,
          y: 188,
        },
      });

      const spritePlay = new Sprite({
        resource: "over",
        spriteName: "play.png",
      });

      play.addComponent(spritePlay);

      const evt = play.addComponent(
        new Event({
          type: HIT_AREA_TYPE.Rect,
        })
      );

      evt.on("tap", () => {
        this.game.emit("on-game-ready");
      });

      overBox.addChild(play);
      overBox.addChild(overTitle);
      overBox.addChild(overScore);

      const animation = overBox.addComponent(new Transition());
      const render = overBox.addComponent(
        new Render({
          alpha: 1,
          zIndex: 10,
        })
      );

      animation.group = {
        hidden: [
          {
            name: "alpha",
            component: render,
            values: [
              {
                time: 0,
                value: 1,
                tween: "linear",
              },
              {
                time: 20,
                value: 0,
                tween: "linear",
              },
            ],
          },
        ],
        show: [
          {
            name: "alpha",
            component: render,
            values: [
              {
                time: 0,
                value: 0,
                tween: "linear",
              },
              {
                time: 20,
                value: 1,
                tween: "linear",
              },
            ],
          },
        ],
      };

3、对"重新开始"添加事件

这里也把重新开始的事件抛出去。

 	 const evt = play.addComponent(
        new Event({
          type: HIT_AREA_TYPE.Rect,
        })
      );

      evt.on("tap", () => {
        this.game.emit("on-game-ready");
      });

\

创建从游戏开始,到结束,到重新开始流程

到这里后我们的游戏开始,结束,重新开始,三个事件都已经定义了,分别是:

game_start//开始

game_over//结束

on-game-ready//重新开始

我们分别对上面三个方法进行监听,需要执行的思路先写在里面,到现在为止就可以实现基本的开始结束流程了,小鸟可以原地跳跃,落地后死亡,重新开始。

		  //游戏开始
      this.game.on("start_game", () => {
       //1隐藏readyBOx
       //2添加小鸟(首次)
       //3添加小鸟的物理属性
      
      });
      //游戏结束
      this.game.on("game_over", () => {
      	//显示overBox
        
      });
      //回到标题
      this.game.on("on-game-ready", () => {
    		//隐藏overBox
        //小鸟回到中间并且取消物理属性
        //显示readyBox
      });