html
飞机大战
js
(function() {
var canvas = document.getElementById("canFly"); //获取canvas元素
//创建画布对象
var context = canvas.getContext("2d");
canvas.width = document.documentElement.clientWidth;
canvas.height = document.documentElement.clientHeight;
$('body').on('touchmove', function (event) {
event.preventDefault();
});
//游戏的基本数据
var gameData = {
state: this.START,
//游戏状态
START: 0,
STARTING: 1,
RUNNING: 2,
PAUSED: 3,
GAMEOVER: 4,
//英雄机生命
heroLife: 3,
//得分
score: 0,
//画布宽高
WIDTH: canvas.width,
HEIGHT: canvas.height
}
/*-- 加载游戏图片 -------------------------------------------------------------------*/
//背景图片
var bgImg = new Image();
bgImg.src = "images/background_1.jpg";
//logo图片
var startLogo = new Image();
startLogo.src = "images/开始背景.png";
//加载飞机入场动画
var loadings = [];
loadings[0] = new Image();
loadings[0].src = "Images/game_loading1.png";
loadings[1] = new Image();
loadings[1].src = "Images/game_loading2.png";
loadings[2] = new Image();
loadings[2].src = "Images/game_loading3.png";
loadings[3] = new Image();
loadings[3].src = "Images/game_loading4.png";
//加载英雄机图片
var heros = [];
heros[0] = new Image();
heros[0].src = "images/me.png";
heros[1] = new Image();
heros[1].src = "images/me.png";
//英雄机爆破动画图片
heros[2] = new Image();
heros[2].src = "images/me.png";
heros[3] = new Image();
heros[3].src = "images/me.png";
heros[4] = new Image();
heros[4].src = "images/me.png";
heros[5] = new Image();
heros[5].src = "images/me.png";
//加载子弹的图片
var bullet = [];
bullet[0] = new Image();
bullet[0].src = "images/bullet1.png";
//加载敌方飞机
//加载小飞机图片
var enemies1 = []; //小飞机
enemies1[0] = new Image();
enemies1[0].src = "images/enemy1_fly_1.png";
enemies1[1] = new Image();
enemies1[1].src = "images/enemy1_fly_1.png";
enemies1[2] = new Image();
enemies1[2].src = "images/enemy1_fly_1.png";
enemies1[3] = new Image();
enemies1[3].src = "images/enemy1_fly_1.png";
enemies1[4] = new Image();
enemies1[4].src = "images/enemy1_fly_1.png";
var enemies2 = []; //中飞机
enemies2[0] = new Image();
enemies2[0].src = "images/enemy3_fly_1.png";
enemies2[1] = new Image();
enemies2[1].src = "images/enemy3_fly_1.png";
enemies2[2] = new Image();
enemies2[2].src = "images/enemy3_fly_1.png";
enemies2[3] = new Image();
enemies2[3].src = "images/enemy3_fly_1.png";
enemies2[4] = new Image();
enemies2[4].src = "images/enemy3_fly_1.png";
var enemies3 = []; //大飞机
enemies3[0] = new Image();
enemies3[0].src = "images/enemy2_fly_1.png";
enemies3[1] = new Image();
enemies3[1].src = "images/enemy2_fly_1.png";
enemies3[2] = new Image();
enemies3[2].src = "images/enemy2_fly_1.png";
enemies3[3] = new Image();
enemies3[3].src = "images/enemy2_fly_1.png";
enemies3[4] = new Image();
enemies3[4].src = "images/enemy2_fly_1.png";
enemies3[5] = new Image();
enemies3[5].src = "images/enemy2_fly_1.png";
enemies3[6] = new Image();
enemies3[6].src = "images/enemy2_fly_1.png";
enemies3[7] = new Image();
enemies3[7].src = "images/enemy2_fly_1.png";
//加载暂停图片
var paused = new Image();
paused.src = "images/game_pause_nor.png";
/*-- 初始化游戏内容相关数据 --*/
//初始化游戏背景图片数据
var SKY = {
imgs: bgImg, //背景图片
width: 360, //图片宽度
height: 852 //图片高度
}
//初始化英雄机入场动画图片数据
var LOADING = {
imgs: loadings,
width: 186, //图片宽度
height: 38, //图片高度
sum: loadings.length //图片个数
}
//初始化英雄机的数据
var HERO = {
imgs: heros,
width: 60,
height: 124,
sum: heros.length,
length: 2 //我方飞机正常图片个数
}
//初始化子弹的数据
var BULLET = { //默认子弹
imgs: bullet,
width: 2,
height: 17,
sum: bullet.length
}
//初始化敌方飞机的数据内容
var ENEMY1 = {
imgs: enemies1,
width: 57,
height: 51,
type: 0, //敌方飞机类型
sum: enemies1.length,
length: 1, //正常图片的个数
reviseH: 12, //碰撞宽度修正
reviseW: 6, //碰撞高度修正
life: 1, //飞机生命值
score: 1 //得分
}
var ENEMY2 = { //中飞机
imgs: enemies2,
width: 69,
height: 95,
type: 1,
sum: enemies2.length,
length: 1,
reviseH: 8, //碰撞宽度修正
reviseW: 2, //碰撞高度修正
life: 3,
score: 3
}
var ENEMY3 = { //大飞机
imgs: enemies3,
width: 169,
height: 258,
type: 2,
sum: enemies3.length,
length: 2,
reviseH: 8, //碰撞宽度修正
reviseW: 2, //碰撞高度修正
life: 10,
score: 10
}
/*-- 通用构造器对象 前端代码尽量地使用通用代码 -------------------------------------------------------------------------------------------------------*/
function Compant(config) {
//加载图片
this.imgs = config.imgs;
//图片的宽度和高度
this.width = config.width;
this.height = config.height;
this.sum = config.sum;
this.length = config.length;
// 敌方飞机具有以下属性
this.type = config.type; //敌机类型
this.life = config.life; //敌机声明值
this.score = config.score; //敌机分数
// 设置相对速度
this.time = 0;
// 设置图片的索引值
this.index = 0;
// 是否执行爆破动画的标识
this.down = false;
// 是否删除标识
this.canDelete = false;
//绘图坐标
this.x = 0;
this.y = 0;
// 绘制方法
this.paint = function() {
context.drawImage(this.imgs[this.index], this.x, this.y);
}
// 移动方法
this.step = function() {}
// 执行撞击后的逻辑方法
this.bang = function() {}
}
/*-- 游戏各个内容构造器 ------------------------------------------------------------------------------------*/
//---背景
//创建背景图片的构造器
function BgSky(config) {
//调用通用构造器初始化
Compant.call(this, config);
//图片绘制高度变量
this.y1 = -this.height;
this.y2 = 0;
//定义绘制方法
this.paint = function() {
context.drawImage(this.imgs, 0, this.y1); //第一张图片
context.drawImage(this.imgs, 0, this.y2); //第二张图片
}
//背景heigth运动方法
this.step = function() {
this.time++;
if(this.time % 3 == 0) { //控制背景图片height值的增加
this.y1++; //图片运动下一帧
this.y2++;
//图片移动处画布后将y坐标重置为-height 实现图片衔接滚动
this.y1 > this.height && (this.y1 = -this.height);
this.y2 > this.height && (this.y2 = -this.height);
this.time = 1; //重置移动时间
}
}
}
//创建图片对象
var sky = new BgSky(SKY);
//---英雄机入场动画构造器
function Loading(config) {
Compant.call(this, config);
//定义绘制
this.paint = function() {
//绘制飞机入场动画图片
context.drawImage(this.imgs[this.index], 0, gameData.HEIGHT - this.height);
}
//定义入场动画
this.step = function() {
this.time++;
if(this.time % 20 == 0) { //实现动画的播放速度
this.index++; //下一帧动画
if(this.index == this.sum) { //判断动画结束后,更改游戏的状态,进入第三阶段游戏阶段
gameData.state = gameData.RUNNING;
this.time = 0; //重置动画时间
}
}
}
}
//创建飞机入场动画的对象
var loading = new Loading(LOADING);
//---定义我方飞机的构造函数
function Hero(config) {
Compant.call(this, config);
//定义子弹速度
this.btTime = 0;
//定义我方飞机的坐标
this.x = (gameData.WIDTH - this.width) / 2;
this.y = gameData.HEIGHT - this.height - 10;
//定义绘制方法
this.paint = function() {
context.drawImage(this.imgs[this.index], this.x, this.y);
}
//定义飞机动画
this.step = function() {
this.time++;
if(this.down) {
if(this.time % 15 == 0) {
this.index++;
//爆破动画执行完毕
if(this.index >= this.sum) {
if(gameData.heroLife > 0) { //我方飞机生命值大于零
hero = new Hero(HERO);
this.down = false;
} else { //生命值为零
gameData.state = gameData.GAMEOVER;
}
this.index = this.length;
}
}
} else {
if(this.time % 15 == 0) {
this.index++;
//控制this.index的值 - 0和1之间切换
//this.index==this.sum&&(this.index=0);
this.index = this.index % this.length; //飞机图片两张 可以
this.time = 0; //重置动画间隔时间
}
}
}
//增加飞机的射击方法
this.shoot = function() {
//
this.btTime++;
if(this.btTime % 11 == 0) {
bullets[bullets.length] = new Bullet(BULLET);; //向子弹数组中添加子弹
this.btTime = 0;
}
}
//我方飞机撞机后
this.bang = function() {
if(!this.down) {
//减生命值
gameData.heroLife--;
//进入爆破动画
this.down = true;
this.index = this.length;
}
}
}
//创建我方飞机对象
var hero = new Hero(HERO);
//---创建子弹的构造器
function Bullet(config) {
Compant.call(this, config);
//子弹的坐标
this.x = hero.x + hero.width / 2 - this.width / 2;
this.y = hero.y - this.height;
//定义绘制子弹的方法
this.paint = function() {
context.drawImage(this.imgs[this.index], this.x, this.y);
}
//定义移动子弹的方法
this.step = function() {
this.y -= 6; //子弹移动的速度
}
//设置删除子弹
this.bang = function() {
this.canDelete = true;
}
}
//定义子弹存储数组
var bullets = [];
//定义绘制所有子弹的函数
function paintBullets() {
for(var i = 0, length = bullets.length; i < length; i++) {
bullets[i].paint(); //绘制当前子弹
if(gameData.state == gameData.RUNNING) { //游戏运行中时移动子弹
bullets[i].step(); //移动子弹
}
}
}
//清除超出屏幕的子弹
function clearStep() {
for(var i = bullets.length - 1; i >= 0; i--) {
if(bullets[i].y <= -bullets[i].height || (bullets[i].canDelete)) {
bullets.splice(i, 1); //删除当前超出屏幕的子弹和撞机的子弹
}
}
}
//---
//敌方飞机构造器
function Enemy(config) {
Compant.call(this, config);
//撞机修正
this.reviseH = config.reviseH;
this.reviseW = config.reviseW;
//敌方飞机的坐标
this.x = Math.floor(Math.random() * (gameData.WIDTH - this.width));
this.y = -this.height;
//大飞机动画
this.bigTime = 0;
//定义绘制方法
this.paint = function() {
context.drawImage(this.imgs[this.index], this.x, this.y);
}
//移动方法
this.step = function() {
//判断this.down的值
if(this.down) {
//切换爆破动画
this.time++;
if(this.time % 15 == 0) {
this.index++;
}
//判断爆破动画是否执行完毕
if(this.index >= this.sum) {
gameData.score += this.score; //得分
this.canDelete = true; //删除敌机
}
} else {
//敌方飞机自上而下运动
this.time++;
switch(this.type) {
//小飞机
case 0:
this.y++;
break;
//中飞机
case 1:
this.time % 2 == 0 && this.y++;
break;
//大飞机
case 2:
this.time % 3 == 0 && this.y++;
break;
}
//大飞机动画设置
this.bigTime++;
if(this.type == 2 && (this.bigTime % 10 == 0)) {
this.index = this.index == 0 ? 1 : 0; //大飞机动画
this.bigTime = 0; //重置
}
}
}
//敌机是否被撞机判断
this.hit = function(bh) {
return(bh.x + bh.width > this.x - this.reviseW) && (bh.y < this.y + this.height - this.reviseH) && (bh.x < this.x + this.width + this.reviseW) && (bh.y + bh.height > this.y + this.reviseH);
}
//敌方飞机被撞
this.bang = function() {
if(!this.down && !this.canDelete) {
//生命值减一
this.life--;
//生命值为0
if(this.life <= 0) {
this.down = true; //执行爆破
this.index = this.length; //开始执行爆破动画
}
}
}
}
//创建存储敌方飞机的数组
var enemies = [];
//定义敌方飞机出现速度
var enemyTime = 10;
//创建用于创建敌方飞机的函数
function createEnemies() {
/*创建敌方飞机 - 小,中,大*/
var num = Math.floor(Math.random() * 100);
if(num < 80) { //小飞机
enemies[enemies.length] = new Enemy(ENEMY1);
} else if(num < 90) { //中飞机
enemies[enemies.length] = new Enemy(ENEMY2);
} else {
//大飞机只能存在一个
if(enemies.length > 0 && enemies[0].type != 2) {
enemies.unshift(new Enemy(ENEMY3)); //将大飞机添加到数组开头
}
}
}
/*--- 绘制敌方所有的飞机 及 删除超出屏幕和打掉的飞机和判断敌方飞机是否被撞机 ---------------------------------------------------------------------------------------------*/
function paintEnemiesAndCheckHit() {
for(var i = 0; i < enemies.length; i++) { //遍历敌机
//
var enemy = enemies[i]; //敌机
//检测敌机和英雄机是否碰撞
if((enemy.y > gameData.HEIGHT) || (enemy.canDelete)) {
enemies.splice(i, 1); //删除当前超出屏幕的飞机
continue;
}
enemy.paint(); //绘制飞机
if(gameData.state == gameData.RUNNING) { //游戏运行中时才移动飞机
enemy.step(); //移动飞机
}
//判断是否和我方飞机碰撞
if(enemy && enemy.hit(hero)) { //敌机和我方飞机相撞
enemy.bang();
hero.bang(); //飞机销毁
}
//判断子弹
for(var j = 0; j < bullets.length; j++) { //子弹遍历
var bullet = bullets[j]; //子弹
if(enemy.hit(bullet)) { //子弹撞机敌方飞机
enemy.bang(); //删除敌机
bullet.bang(); //删除子弹
}
}
}
}
/*--- 绘制游戏得分 ----------------------------------------------------------------------------------------------*/
function painText() {
context.font = "bold 24px 微软雅黑";
context.fillText("分数: " + gameData.score, 10, 30);
context.fillText("生命值: " + gameData.heroLife, 200, 30);
}
/*-- 绘制暂停图标 ----------------------------------------------------------------------------------------------*/
function paintPaused() {
context.drawImage(paused, gameData.WIDTH / 2 - 30, gameData.HEIGHT / 2 - 22.5);
}
/*-- 游戏结束 ----------------------------------------------------------------------------------------------*/
function gameover() {
context.font = "bold 35px 微软雅黑";
context.fillText("GAME OVER", gameData.WIDTH / 2 - 110, gameData.HEIGHT / 2);
}
function next() {
context.font = "bold 30px 微软雅黑";
context.fillText("再来一把", gameData.WIDTH / 2 - 60, gameData.HEIGHT / 1.5);
}
/*-- 事件绑定 ------------------------------------------------------------------------------------------------*/
//画布事件绑定
canvas.onclick = function() {
if(gameData.state == gameData.START) { //在游戏开始状态下单击,进入游戏过渡阶段
gameData.state = gameData.STARTING; //改变游戏状态
}
}
//为