#使用HTML5制作的打飞机游戏 ##教你如何“打飞机” 道友们看到这标题是不是心血来潮, 。
由于楼主最近晚上闲来无事,前端历程也不到一年,JS水平也是 ,又对游戏挺感兴趣的,所以心血来潮,用这个良辰写个小小游戏练练手。
好了,废话不多说,“打飞机”现在开始,游戏很简单,是仿微信上面的打飞机小游戏的,玩法很简单,自己控制的飞机随鼠标移动自动发射子弹,敌方飞机分为3种大小,种类不同血量即不同,当自己飞机中敌方子弹或碰到敌方飞机则少一命,最后命为0则game over,游戏会记录所获得的积分。 ##整个游戏分析 ###1.资源 整理好各种图片,飞机按类型分类命名,一组飞机(一只飞机正常状态到死亡会有好几张图片组成)命名相似。
敌机
enemy1.png
enemy1_down(n).png
enemy1_down(n).png
enemy1_down(n).png
enemy1_down(n).png
英雄(同敌机)...... ###2.游戏状态(state) START : 进入到游戏 , 即进入到游戏主界面,并未开始游戏页面初始化状态为 START
详解:
1、背景在动
2、显示飞机大战的图像
STARTING : 过渡,开始状态, 与游戏进行状态的过渡,用户点击鼠标左键时,由START --> STARTING
RUNNING : 游戏正式开始,实现飞机大战的过程
1、当准备图像中的最后一幅显示完毕时,STARTING --> RUNNING
2、当鼠标从canvas所在区域外移入时,并且状态为PAUSE是, PAUSE --> RUNNING
PAUSE : 暂停,鼠标移出canvas所在的区域 RUNNING --> PAUSE
GAME_OVER:结束,LIFE为0时(销毁过程已经结束) ###3.计时器 setInterval 整个游戏很多的东西都在使用计时器(英雄、敌机、子弹、天空等等),因此设置这么多计时器是不行的,这里想到了只用一个计时器来控制所有的模块。
这里有个问题:如何控制图像组件的速度
解决:计时器固定刷新频率 : 1000 / 100,每一个图像组件都有自己的速度 (20 --> 1000 / 20),让计时器测量图像组件是否到达运动时间,如果到达则运动,否则不动。 ###4.设计元素 1、数据对象:只负责封装基本数据 例:天空-宽度高度移动速度等等 var SKY={width:480,height:600,speed:20}
2、业务对象:负责业务逻辑的计算 例:处理天空运动var Sky=function(SKY){...}
##具体实现
###1.三种敌机(Enermy)
E1:小飞机
E2:中型飞机
E3:大型飞机
飞机图像数组
var e1 = [];//装载从开始到销毁的所有状态的图片
注意:e1与e2 一副基本图像,多幅销毁图像
e3:两幅基本图像,多幅销毁图像
####三种飞机的数据对象
E1:
type:类型 1-3 分别表示不同类型的飞机 : 1
score:分数 : 1
frames:对应的图像数组 :e1
baseFrameCount:基本图像帧数 : 1
life:生命,几枪被击落 :1
minSpeed:最小速度 :70
maxSpeed:最大速度 :100
width:宽度
height:高度
E2:
type:类型 1-3 分别表示不同类型的飞机
score:分数
frames:对应的图像数组
baseFrameCount:基本图像帧数 :1
life:生命,几枪被击落
minSpeed:最小速度 :50
maxSpeed:最大速度 :70
width:宽度
height:高度
E3:
type:类型 1-3 分别表示不同类型的飞机
score:分数
frames:对应的图像数组
baseFrameCount:基本图像帧数 :2
life:生命,几枪被击落
speed:10
width:宽度
height:高度
####三种飞机的业务对象 #####属性: 1、speed: //判断是小飞机、中型飞机 还是大飞机
if(config.minSpeed && config.maxSpeed){
this.speed = Math.floor((Math.random() * (config.maxSpeed-config.minSpeed)) + config.minSpeed);
}else {
this.speed = config.speed;
}
2、x
当前飞机的横坐标,随机,不能再屏幕中出现一半
3、y
当前飞机的纵坐标
-config.height
#####方法
1、step
1)、基本帧的切换
借助config.baseFrameCount 的值进行切换
2)、控制飞机的移动 : y++
2、move
y ++
3、paint
ctx.drawImage(this.frame,x,y);
###2.创建飞机
var enemies = []; 保存所有飞机的数组
数组中的第一个位置,永远都是大飞机
var lastTime = 0;
var interval = 800;
//创建飞机的全局函数,由RUNNING时调用
function componentEnter(){
if(时间判断){
//控制飞机出现的数量
//使用0-9的随机数控制
//0-7 : 出现的都是小飞机
new Enemy(E1)
enemies[enemies.length] = new Enemy(E1);
//8:出现的就是中型飞机
new Enemy(E2)
//9:可能出现的是大飞机
if(enemies[0].type != 3){
//创建大飞机,并且保存进第一个位置
enemies.splice(0,0,new Enemy(E3));
}
lastTime = new Date().getTime();
}
}
###3.移除超出边界的敌人飞机
function Enemy(){
this.outOfBounds=function(){
if(this.y >= HEIGHT){
return true;
}
return false;
}
}
for(var i=0;i<enemies.length;i++){
if(enemies[i].outOfBounds()){
enemies.splice(i,1);
}
}
###4.给敌人业务对象中添加业务方法
/**
* 检查敌人飞机(this)是否与其他元素进行相撞
* c:其他元素 --> Hero,Bullet
* 判断c的中心点是否出现在指定区域中
* c的中心点:
* x:c.x + c.width / 2
* y:c.y + c.heght / 2
* 横向范围:this.x - c.width / 2 ~ this.x+this.width+c.width/2
* 纵向范围:this.y - c.height /2 ~ this.y + this.height + c.height / 2
*/
function hit = function(c){
//相撞返回true
//否则返回false
}
###5.创建全局判断每个敌人是否与其他元素碰撞的方法
function checkHit(){
//遍历所有的敌人飞机
for(var i=0;i<enemies.length;i++){
var enemy = enemies[i];
//enemy是否与子弹相撞
//enemy是否与英雄相撞
if(enemy.hit(hero)){
//enemy与英雄相撞
}
}
}
###6.撞击后
为英雄、敌人飞机添加属性
this.down : 是否爆破,默认为false
一旦this.down为true时
英雄,敌人飞机的 frame 立即改为爆破的几幅图像
英雄this.down后判断还有几条命。如果>0 ,创建英雄对象接着玩
如果 = 0 ---> state = GAME_OVER
所有的爆破图像显示完毕后,this.canDelete=true
this.canDelete:是否删除,默认为false
function deleteComponent(){
//判断边界移除飞机
//判断canDelete属性删除飞机
}
这张图所画的是本游戏的核心,包含英雄和敌机碰撞以及子弹打中敌机、英雄中弹都是根据此图来的,请大家仔细研究下此图。
以上就是整个游戏的核心代码。
演示地址:进入游戏
具体代码(全程注释)在github上了,地址是:https://github.com/godnew/PLANE,欢迎star。