飞机大战>背景/音效动起来

前面的准备工作已经做完,马上开始我们的第一站吧,游戏背景和游戏音乐的实现。

bg.gif
1.为了项目结构清晰和代码规范,我们把游戏拆解为层次结构来实现,在src文件夹下新建一个目录background,这个目录的内容就是本章实现的核心.
image.png
2.在目录下新建一个类BackgroundGame并且继承egret.DisplayObjectContainer,关于该类,自行查阅API,我的理解就是对象容器显示接口,每一个显示的对象都可以看做一个小模块,然后填充到eui.UILayer容器中,用于丰富我们的容器。
3.关于项目编译过程,查看官网文档
  • eui.UILayer:游戏的入口类就是Main.ts类下,继承于eui.UILayer
  • LoadingUI:游戏加载过程的进度显示,可以根据自己游戏风格更改
  • ThemeAdapter:皮肤类,关于UI上面的逻辑和开发可以纯代码实现,也可以通过布局文件来实现,我比较喜欢纯代码风格,所以整个项目我都不会使用皮肤方式。
  • 其他类都是项目自行生成,暂时没有用到,就不在多说,直接打开我们需要实现的类BackgroundGame
  • 每个类好像只能有一个构造方法,对于我这个java控来说有点诡异,如果有继承还需要在构造方法中实现super
class BackgroundGame extends egret.DisplayObjectContainer{
    public constructor() {
        super();
        this.addEventListener(egret.Event.ADDED_TO_STAGE,this.loadBackground,this);
    }
}
  • 构造方法中有一行代码addEventListener,用于监听填充到容器中完成后的事件,这样可以获取到正确的容器高宽。
  • addEventListener第一个参数是用于监听的事件,查看官方API;第二个参数是事件回调,这里就是填充容器完毕后执行改方法;第三个参数是事件作用域(当前类)。
4.实现背景逻辑
/**
    * 地图上半部分(隐藏)
     */
    private mBackgroundTop:egret.Bitmap;
    /**
     * 地图显示部分
     */
    private mBackgroundBottom:egret.Bitmap;
    /**
     * 屏幕的高度
     */
    private stageH:number;
    /**
     * 地图滚动速度,越小滚动越快。毫秒
     */
    private mBgSpeed = 4000;
  • 实现回调方法loadBackground
public loadBackground() : void{
        //游戏背景上半部分资源加载
        this.mBackgroundTop = ImageResUtils.createBitmapByName("background_jpg");
        this.addChild(this.mBackgroundTop);
        let stageW = this.stage.stageWidth;
        this.stageH = this.stage.stageHeight;
        this.mBackgroundTop.width = stageW;
        this.mBackgroundTop.height = this.stageH;
        this.mBackgroundTop.y = -this.stageH;
        /**
         * 游戏背景下半部分资源
         */
        this.mBackgroundBottom = ImageResUtils.createBitmapByName("background_jpg");
        this.mBackgroundBottom.width = stageW;
        this.mBackgroundBottom.height = this.stageH;
        this.addChild(this.mBackgroundBottom);
    }
  • 图片工具类ImageResUtils
class ImageResUtils {
    public constructor() {
    }
     /**
     * 根据name关键字创建一个Bitmap对象。name属性请参考resources/resource.json配置文件的内容。
     * Create a Bitmap object according to name keyword.As for the property of name please refer to the configuration file of resources/resource.json.
     */
    public static createBitmapByName(name:string) : egret.Bitmap{
        let result = new egret.Bitmap();
        let texture: egret.Texture = RES.getRes(name);
        result.texture = texture;
        return result;
    }
}
  • 把游戏背景图片拖入到resource下的assets文件夹下,并且IDE顶部会有一个资源新增提示,点击保存后资源会刷新,这样你在代码中才能获取到图片文件

    image.png

  • 方法内容就是加载游戏背景资源,参数很容易理解,游戏背景是给玩家一种无限滚动的视觉差,说一下实现思路:
    把同一张背景加载到模块中上下拼接起来做往下滚动,战机是往上飞行的。
    如果难以理解不需要钻牛角尖,后面做多了自己就能顿悟。

5.新增一个背景动画开始的方法,提供给外部调用
/**
     * 游戏背景动画开始
     */
    public startAnimationBg(){
        egret.Tween.get(this.mBackgroundTop,{loop:true}).to({y:0},this.mBgSpeed);
        egret.Tween.get(this.mBackgroundBottom,{loop: true}).to({y:this.stageH},this.mBgSpeed);
    }

egret.Tween缓存动画库,也是项目中用得最多的方法,可以省下很多代码,唯一就是感觉灵动性不好,查看官网API

最顶部的图片最开始是不可见的,执行动画就是让第一张图片从不可见运动到可见,第二张图片从可见运行到最底部不可见状态,以此循环达到无限滚动效果。

6.设计一个关卡类来加载相关的模块。
  • 新增父类
class Level {
    /**
     * 容器
     */
    private layour:eui.UILayer;
    /**
     * 游戏状态
     */
    public static GAME_READY:number = 1;
    public static GAME_ING:number = 2;
    public static GAME_PAUSE:number = 3;
    public static GAME_END:number = 4;
    public static GAME_STATE : number;

    public constructor(layour:eui.UILayer) {
        this.layour = layour;
        Level.GAME_STATE = Level.GAME_READY;
    }
    /**
     * 游戏开始
     */
    public startGame(){
        Level.GAME_STATE = Level.GAME_ING;
    }
    /**
     * 游戏暂停
     */
    public pauseGame(){
        Level.GAME_STATE = Level.GAME_PAUSE;
    }
    /**
     * 重新开始
     */
    public restartGame(){

    }
    /**
     * 游戏结束
     */
    public endGame(){
        Level.GAME_STATE = Level.GAME_END;
        this.layour.removeChildren();
    }
    /**
     * 添加元素
     */
    public addChild(child:egret.DisplayObject){
        this.layour.addChild(child);
    }
    public addChildAt(child:egret.DisplayObject,index:number){
        this.layour.addChildAt(child,index);
    }
    /**
     * 容器全局变量
     */
    public getContext():eui.UILayer{
        return this.layour;
    }
}
  • 新增我们的第一关,新建一个类TheFirstPass(类名很随意)继承Level,并且实现startGameendGamerestartGame方法
public startGame(){
        super.startGame();
        //加载背景
        this.mBgGame = new BackgroundGame();
        this.addChild(this.mBgGame);
        this.mBgGame.startAnimationBg();
        //游戏结束提示
        this.mGameTextMessage = new egret.TextField();
        this.mGameTextMessage.textColor = 0xfffffff;
        this.mGameTextMessage.size = 40;
        this.mGameTextMessage.textAlign = "center"
        this.mGameTextMessage.x = 0;
        //重新开始
        this.mGameRestart = new egret.TextField();
        this.mGameRestart.textColor = 0xfffffff;
        this.mGameRestart.size = 40;
        this.mGameRestart.textAlign = egret.HorizontalAlign.CENTER;
        this.mGameRestart.verticalAlign = egret.VerticalAlign.MIDDLE;
        this.mGameRestart.backgroundColor = 0x000000; 
        this.mGameRestart.text = "再来一次"
        this.mGameRestart.touchEnabled = true;
        this.mGameRestart.stroke = 1;
        this.mGameRestart.strokeColor = 0xffffff;
    }
/**
     * 游戏结束
     */
    public endGame(){
        super.endGame();
        this.addChild(this.mBgGame);
        this.mBgGame.loadBackground();
        this.mGameTextMessage.text="游戏结束,你输了!";
        this.addChild(this.mGameTextMessage);
        this.addChild(this.mGameRestart);
        this.mGameRestart.addEventListener(egret.TouchEvent.TOUCH_TAP,this.restartGame,this);
    }
/**
     * 游戏重新开始
     */
    public restartGame(){
        this.mGameRestart.removeEventListener(egret.TouchEvent.TOUCH_TAP,this.restartGame,this);
        super.endGame();
        this.startGame();
    }
  • startGame中加载游戏背景,调用背景类的startAnimationBg方法让我们的背景动起来。
7.打开入口类Main.ts,找到createGameScene方法,删除方法中原来的代码,新增自己的关卡代码。
/**
     * 创建场景界面
     * Create scene interface
     */
    protected createGameScene(): void {
        //第一关
        var mLevel = new TheFirstPass(this);
        mLevel.startGame();
    }

游戏关卡1中的逻辑就成功加载到容器中。

8.背景有了,音乐呢?
  • 把背景音乐资源拖入到resource>assets文件夹下并保存。
  • 新增一个音乐管理工具类BgSoundMusic
class BgSoundMusic {
    
    private mSoundChannel:egret.SoundChannel;

    public constructor() {
        
    }
    /**
     * 开始播放音乐
     */
    public playMusic(){
        var sound:egret.Sound = new egret.Sound();
        var url:string="resource/assets/gamemusic.mp3";
        sound.addEventListener(egret.Event.COMPLETE, this.onLoadComplete, this);
        sound.load(url);
    }
    private onLoadComplete(event:egret.Event):void {
        //获取加载到的 Sound 对象
        var sound:egret.Sound = <egret.Sound>event.target;
        //播放音乐
        this.mSoundChannel = sound.play(0,0);
    }
    /**
     * 停止
     */
    public stopMusic(){
        if(this.mSoundChannel){
            this.mSoundChannel.stop();
            this.mSoundChannel = null;
        }
    }
}
  • 内容很少,也很简单,首先要加载资源,通过addEventListener来监听资源是否加载完成,加载完成后直接播放。
    play参数解释:两个参数值,第一个是播放位置,第二个是播放次数,0表示循环播放,大于0就是指定播放次数。play方法可以得到一个新生成的对象SoundChannel ,通过它来控制音乐的播放/停止/进度等信息
  • 回到关卡类TheFirstPassstartGame方法中调用背景音乐
//加载游戏音乐
this.mBgSoundMusic = new BgSoundMusic();
this.mBgSoundMusic.playMusic();
我这里实现的都是最基本的动画以及效果,便于新手理解和实现,只要跟着官网文档和自己的设计思路开发,基本上不是什么问题。如果喜欢追求自身不断地进化,请关注我的文章,将持续更新,也会有更多的惊喜。

下一篇是主角出场

在线体验
如果是电脑端记得先把浏览器缩小后再打开地址,不然是横过来的。

源码地址

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,401评论 25 707
  • 虹 汉字,通红或古代把虹比作美人。 虹,红色在外、紫色在内,颜色鲜艳,又叫正虹。 副虹,红色在内,紫色在外,颜色较...
    景光相馆1951阅读 716评论 0 3
  • 五岁时看着哥哥手里拿着俄罗斯方块的游戏机心里羡慕的直痒痒。想着只要有个游戏机,每天自己只玩一分钟也高兴。小学时想着...
    长马阅读 371评论 0 2
  • (稻盛哲学学习会)打卡96天 姓名:张之深 公司:宁波万尚进出口有限公司 部门:行政人事部 【知~学习】 背诵《六...
    张之深Wellin阅读 138评论 0 0
  • 朱日和沙场点兵,另国人骄傲!!!在这值得庆祝的今天,我们雄鹰少年特种兵军事研学夏令营迎来了第五批新战友,从队员们神...
    不寂的鹰阅读 497评论 1 0