Android实战打飞机游戏之子弹生成与碰撞以及爆炸效果(5)

Android实战打飞机游戏子弹生成,新建子弹类

public class Bullet {
 // 子弹图片资源
 public Bitmap bmpBullet;
 // 子弹的坐标
 public int bulletX, bulletY;
 // 子弹的速度
 public int speed;
 // 子弹的种类以及常量
 public int bulletType;
 // 主角的
 public static final int BULLET_PLAYER = -1;
 // 鸭子的
 public static final int BULLET_DUCK = 1;
 // 苍蝇的
 public static final int BULLET_FLY = 2;
 // Boss的
 public static final int BULLET_BOSS = 3;
 // 子弹是否超屏, 优化处理
 public boolean isDead;

 // Boss疯狂状态下子弹相关成员变量
 private int dir;// 当前Boss子弹方向
 // 8方向常量
 public static final int DIR_UP = -1;
 public static final int DIR_DOWN = 2;
 public static final int DIR_LEFT = 3;
 public static final int DIR_RIGHT = 4;
 public static final int DIR_UP_LEFT = 5;
 public static final int DIR_UP_RIGHT = 6;
 public static final int DIR_DOWN_LEFT = 7;
 public static final int DIR_DOWN_RIGHT = 8;

 // 子弹当前方向
 public Bullet(Bitmap bmpBullet, int bulletX, int bulletY, int bulletType) {
  this.bmpBullet = bmpBullet;
  this.bulletX = bulletX;
  this.bulletY = bulletY;
  this.bulletType = bulletType;
  // 不同的子弹类型速度不一
  switch (bulletType) {
  case BULLET_PLAYER:
   speed = 4;
   break;
  case BULLET_DUCK:
   speed = 3;
   break;
  case BULLET_FLY:
   speed = 4;
   break;
  case BULLET_BOSS:
   speed = 5;
   break;
  }
 }

 public void draw(Canvas canvas, Paint paint) {
  canvas.drawBitmap(bmpBullet, bulletX, bulletY, paint);
 }

 // 子弹的逻辑
 public void logic() {
  // 不同的子弹类型逻辑不一
  // 主角的子弹垂直向上运动
  switch (bulletType) {
  case BULLET_PLAYER:
   bulletY -= speed;
   if (bulletY < -50) {
    isDead = true;
   }
   break;
  // 鸭子和苍蝇的子弹都是垂直下落运动
  case BULLET_DUCK:
  case BULLET_FLY:
   bulletY += speed;
   if (bulletY > MySurfaceView.screenH) {
    isDead = true;
   }
   break;
  case BULLET_BOSS:
   // Boss疯狂状态下的子弹逻辑待实现
   // 边界处理
   if (bulletY > MySurfaceView.screenH || bulletY <= -40
     || bulletX > MySurfaceView.screenW || bulletX <= -40) {
    isDead = true;
   }
   break;
  }
 }

}

在在MySurfacview里面调用 生成子弹主角的和自己的

package com.gsf;

import java.util.Random;
import java.util.Vector;

import android.content.Context;
import android.content.res.Resources;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.view.KeyEvent;
import android.view.MotionEvent;
import android.view.SurfaceHolder;
import android.view.SurfaceHolder.Callback;
import android.view.SurfaceView;

public class MySurfaceView extends SurfaceView implements Callback, Runnable {
 private SurfaceHolder sfh;
 private Paint paint;
 private Thread th;
 private boolean flag;
 private Canvas canvas;

 // 1 定义游戏状态常量
 public static final int GAME_MENU = 0;// 游戏菜单
 public static final int GAMEING = 1;// 游戏中
 public static final int GAME_WIN = 2;// 游戏胜利
 public static final int GAME_LOST = 3;// 游戏失败
 public static final int GAME_PAUSE = -1;// 游戏菜单
 // 当前游戏状态(默认初始在游戏菜单界面)
 public static int gameState = GAME_MENU;
 // 声明一个Resources实例便于加载图片
 private Resources res = this.getResources();
 // 声明游戏需要用到的图片资源(图片声明)
 private Bitmap bmpBackGround;// 游戏背景
 private Bitmap bmpBoom;// 爆炸效果
 private Bitmap bmpBoosBoom;// Boos爆炸效果
 private Bitmap bmpButton;// 游戏开始按钮
 private Bitmap bmpButtonPress;// 游戏开始按钮被点击
 private Bitmap bmpEnemyDuck;// 怪物鸭子
 private Bitmap bmpEnemyFly;// 怪物苍蝇
 private Bitmap bmpEnemyBoos;// 怪物猪头Boos
 private Bitmap bmpGameWin;// 游戏胜利背景
 private Bitmap bmpGameLost;// 游戏失败背景
 private Bitmap bmpPlayer;// 游戏主角飞机
 private Bitmap bmpPlayerHp;// 主角飞机血量
 private Bitmap bmpMenu;// 菜单背景
 public static Bitmap bmpBullet;// 子弹
 public static Bitmap bmpEnemyBullet;// 敌机子弹
 public static Bitmap bmpBossBullet;// Boss子弹
 public static int screenW;
 public static int screenH;

 // 声明一个敌机容器
 private Vector<Enemy> vcEnemy;
 // 每次生成敌机的时间(毫秒)
 private int createEnemyTime = 50;
 private int count;// 计数器
 // 敌人数组:1和2表示敌机的种类,-1表示Boss
 // 二维数组的每一维都是一组怪物
 private int enemyArray[][] = { { 1, 2 }, { 1, 1 }, { 1, 3, 1, 2 },
   { 1, 2 }, { 2, 3 }, { 3, 1, 3 }, { 2, 2 }, { 1, 2 }, { 2, 2 },
   { 1, 3, 1, 1 }, { 2, 1 }, { 1, 3 }, { 2, 1 }, { -1 } };
 // 当前取出一维数组的下标
 private int enemyArrayIndex;
 // 是否出现Boss标识位
 private boolean isBoss;
 // 随机库,为创建的敌机赋予随即坐标
 private Random random;

 //
 private GameMenu gameMenu;
 private GameBg gameBg;

 private Player player;

 // 敌机子弹容器
 private Vector<Bullet> vcBullet;
 // 添加子弹的计数器
 private int countEnemyBullet;
 // 主角子弹容器
 private Vector<Bullet> vcBulletPlayer;
 // 添加子弹的计数器
 private int countPlayerBullet;

 /**
  * SurfaceView初始化函数
  */
 public MySurfaceView(Context context) {
  super(context);
  sfh = this.getHolder();
  sfh.addCallback(this);
  paint = new Paint();
  paint.setColor(Color.WHITE);
  paint.setAntiAlias(true);
  setFocusable(true);
 }

 /**
  * SurfaceView视图创建,响应此函数
  */
 @Override
 public void surfaceCreated(SurfaceHolder holder) {
  screenW = this.getWidth();
  screenH = this.getHeight();
  initGame();
  flag = true;
  // 实例线程
  th = new Thread(this);
  // 启动线程
  th.start();
 }

 /**
  * 加载游戏资源
  */
 private void initGame() {
  // 加载游戏资源
  bmpBackGround = BitmapFactory
    .decodeResource(res, R.drawable.background);
  bmpBoom = BitmapFactory.decodeResource(res, R.drawable.boom);
  bmpBoosBoom = BitmapFactory.decodeResource(res, R.drawable.boos_boom);
  bmpButton = BitmapFactory.decodeResource(res, R.drawable.button);
  bmpButtonPress = BitmapFactory.decodeResource(res,
    R.drawable.button_press);
  bmpEnemyDuck = BitmapFactory.decodeResource(res, R.drawable.enemy_duck);
  bmpEnemyFly = BitmapFactory.decodeResource(res, R.drawable.enemy_fly);
  bmpEnemyBoos = BitmapFactory.decodeResource(res, R.drawable.enemy_pig);
  bmpGameWin = BitmapFactory.decodeResource(res, R.drawable.gamewin);
  bmpGameLost = BitmapFactory.decodeResource(res, R.drawable.gamelost);
  bmpPlayer = BitmapFactory.decodeResource(res, R.drawable.player);
  bmpPlayerHp = BitmapFactory.decodeResource(res, R.drawable.hp);
  bmpMenu = BitmapFactory.decodeResource(res, R.drawable.menu);
  bmpBullet = BitmapFactory.decodeResource(res, R.drawable.bullet);
  bmpEnemyBullet = BitmapFactory.decodeResource(res,
    R.drawable.bullet_enemy);
  bmpBossBullet = BitmapFactory
    .decodeResource(res, R.drawable.boosbullet);

  // 菜单类实例化
  gameMenu = new GameMenu(bmpMenu, bmpButton, bmpButtonPress);
  // 实例游戏背景
  gameBg = new GameBg(bmpBackGround);
  // 实例主角
  player = new Player(bmpPlayer, bmpPlayerHp);
  //敌机子弹容器实例
  vcBullet = new Vector<Bullet>();
  //主角子弹容器实例
  vcBulletPlayer = new Vector<Bullet>();
  // 实例敌机容器
  vcEnemy = new Vector<Enemy>();
  // 实例随机库
  random = new Random();
 }

 /**
  * 游戏绘图
  */
 public void myDraw() {
  try {
   canvas = sfh.lockCanvas();
   if (canvas != null) {
    canvas.drawColor(Color.WHITE);
    // 绘图函数根据游戏状态不同进行不同绘制

    switch (gameState) {
    case GAME_MENU:

     gameMenu.draw(canvas, paint);
     break;
    case GAMEING:
     gameBg.draw(canvas, paint);
     player.draw(canvas, paint);
     if (isBoss == false) {
      // 敌机绘制
      for (int i = 0; i < vcEnemy.size(); i++) {
       vcEnemy.elementAt(i).draw(canvas, paint);
      }
      //敌机子弹绘制
      for (int i = 0; i < vcBullet.size(); i++) {
       vcBullet.elementAt(i).draw(canvas, paint);
      }

     } else {
      // boss 绘制

     }
     // 处理主角子弹绘制
     for (int i = 0; i < vcBulletPlayer.size(); i++) {
      vcBulletPlayer.elementAt(i).draw(canvas, paint);
     }

     break;

    case GAME_WIN:

     break;
    case GAME_LOST:

     break;
    case GAME_PAUSE:

     break;
    default:
     break;
    }

   }
  } catch (Exception e) {
   // TODO: handle exception
  } finally {
   if (canvas != null)
    sfh.unlockCanvasAndPost(canvas);
  }
 }

 /**
  * 触屏事件监听
  */
 @Override
 public boolean onTouchEvent(MotionEvent event) {
  switch (gameState) {
  case GAME_MENU:

   gameMenu.onTouchEvent(event);
   break;
  case GAMEING:

   break;

  case GAME_WIN:

   break;
  case GAME_LOST:

   break;
  case GAME_PAUSE:

   break;
  }
  return true;
 }

 /**
  * 按键事件监听
  */
 @Override
 public boolean onKeyDown(int keyCode, KeyEvent event) {
  switch (gameState) {
  case GAME_MENU:

   break;
  case GAMEING:
   player.onKeyDown(keyCode, event);
   break;

  case GAME_WIN:

   break;
  case GAME_LOST:

   break;
  case GAME_PAUSE:
   break;
  }
  return super.onKeyDown(keyCode, event);
 }

 @Override
 public boolean onKeyUp(int keyCode, KeyEvent event) {
  switch (gameState) {
  case GAME_MENU:

   break;
  case GAMEING:
   player.onKeyUp(keyCode, event);
   break;

  case GAME_WIN:

   break;
  case GAME_LOST:

   break;
  case GAME_PAUSE:
   break;
  }
  return super.onKeyUp(keyCode, event);
 }

 /**
  * 游戏逻辑
  */
 private void logic() {
  switch (gameState) {
  case GAME_MENU:

   break;
  case GAMEING:
   gameBg.logic();
   player.logic();
   // 敌机逻辑
   if (isBoss == false) {
    // 敌机逻辑
    for (int i = 0; i < vcEnemy.size(); i++) {
     Enemy en = vcEnemy.elementAt(i);
     // 因为容器不断添加敌机 ,那么对敌机isDead判定,
     // 如果已死亡那么就从容器中删除,对容器起到了优化作用;
     if (en.isDead) {
      vcEnemy.removeElementAt(i);
     } else {
      en.logic();
     }
    }
    // 生成敌机
    count++;
    if (count % createEnemyTime == 0) {
     for (int i = 0; i < enemyArray[enemyArrayIndex].length; i++) {
      // 苍蝇
      if (enemyArray[enemyArrayIndex][i] == 1) {
       int x = random.nextInt(screenW - 100) + 50;
       vcEnemy.addElement(new Enemy(bmpEnemyFly, 1, x, -50));
       // 鸭子左
      } else if (enemyArray[enemyArrayIndex][i] == 2) {
       int y = random.nextInt(20);
       vcEnemy.addElement(new Enemy(bmpEnemyDuck, 2, -50,
         y));
       // 鸭子右
      } else if (enemyArray[enemyArrayIndex][i] == 3) {
       int y = random.nextInt(20);
       vcEnemy.addElement(new Enemy(bmpEnemyDuck, 3,
         screenW + 50, y));
      }
     }
     // 这里判断下一组是否为最后一组(Boss)
     if (enemyArrayIndex == enemyArray.length - 1) {
      isBoss = true;
     } else {
      enemyArrayIndex++;
     }
    }

    // 每2秒添加一个敌机子弹
    countEnemyBullet++;
    if (countEnemyBullet % 40 == 0) {
     for (int i = 0; i < vcEnemy.size(); i++) {
      Enemy en = vcEnemy.elementAt(i);
      // 不同类型敌机不同的子弹运行轨迹
      int bulletType = 0;
      switch (en.type) {
      // 苍蝇
      case Enemy.TYPE_FLY:
       bulletType = Bullet.BULLET_FLY;
       break;
      // 鸭子
      case Enemy.TYPE_DUCKL:
      case Enemy.TYPE_DUCKR:
       bulletType = Bullet.BULLET_DUCK;
       break;
      }
      vcBullet.add(new Bullet(bmpEnemyBullet, en.x + 10,
        en.y + 20, bulletType));
     }
    }
    // 处理敌机子弹逻辑
    for (int i = 0; i < vcBullet.size(); i++) {
     Bullet b = vcBullet.elementAt(i);
     if (b.isDead) {
      vcBullet.removeElement(b);
     } else {
      b.logic();
     }
    }

    //每1秒添加一个主角子弹
    countPlayerBullet++;
    if (countPlayerBullet % 20 == 0) {
     vcBulletPlayer.add(new Bullet(bmpBullet, player.x + 15, player.y - 20, Bullet.BULLET_PLAYER));
    }

    // 处理敌机与主角的碰撞
    for (int i = 0; i < vcEnemy.size(); i++) {
     if (player.isCollsionWith(vcEnemy.elementAt(i))) {
      // 发生碰撞,主角血量-1
      player.setPlayerHp(player.getPlayerHp() - 1);
      // 当主角血量小于0,判定游戏失败
      if (player.getPlayerHp() <= -1) {
       gameState = GAME_LOST;
      }
     }
    }

    //处理主角子弹逻辑
    for (int i = 0; i < vcBulletPlayer.size(); i++) {
     Bullet b = vcBulletPlayer.elementAt(i);
     if (b.isDead) {
      vcBulletPlayer.removeElement(b);
     } else {
      b.logic();
     }
    }
   }
   break;

  case GAME_WIN:

   break;
  case GAME_LOST:

   break;
  case GAME_PAUSE:
   break;
  }

 }

 @Override
 public void run() {
  while (flag) {
   long start = System.currentTimeMillis();
   myDraw();
   logic();
   long end = System.currentTimeMillis();
   try {
    if (end - start < 50) {
     Thread.sleep(50 - (end - start));
    }
   } catch (InterruptedException e) {
    e.printStackTrace();
   }
  }
 }

 /**
  * SurfaceView视图状态发生改变,响应此函数
  */
 @Override
 public void surfaceChanged(SurfaceHolder holder, int format, int width,
   int height) {
 }

 /**
  * SurfaceView视图消亡时,响应此函数
  */
 @Override
 public void surfaceDestroyed(SurfaceHolder holder) {
  flag = false;
 }
}

现在子弹碰撞主角 并没有发生事情 需要子弹和主角的碰撞检测

//修改Player类 添加碰撞检测
 // // 判断碰撞(主角与敌机子弹)
 public boolean isCollsionWith(Bullet bullet) {
  // 是否处于无敌时间
  if (isCollision == false) {
   int x2 = bullet.bulletX;
   int y2 = bullet.bulletY;
   int w2 = bullet.bmpBullet.getWidth();
   int h2 = bullet.bmpBullet.getHeight();
   if (x >= x2 && x >= x2 + w2) {
    return false;
   } else if (x <= x2 && x + bmpPlayer.getWidth() <= x2) {
    return false;
   } else if (y >= y2 && y >= y2 + h2) {
    return false;
   } else if (y <= y2 && y + bmpPlayer.getHeight() <= y2) {
    return false;
   }
   // 碰撞即进入无敌状态
   isCollision = true;
   return true;
   // 处于无敌状态,无视碰撞
  } else {
   return false;
  }
 }

在主界面 MySrufaceView 的逻辑函数中添加 主角和敌机子弹的碰撞

 //处理敌机子弹与主角碰撞
    for (int i = 0; i < vcBullet.size(); i++) {
     if (player.isCollsionWith(vcBullet.elementAt(i))) {
      //发生碰撞,主角血量-1
      player.setPlayerHp(player.getPlayerHp() - 1);
      //当主角血量小于0,判定游戏失败
      if (player.getPlayerHp() <= -1) {
       gameState = GAME_LOST;
      }
     }
    }

当然 主角的子弹碰撞了敌机 也需要进行碰撞检测 修改 敌机类 Enemy

 //判断碰撞(敌机与主角子弹碰撞)
 public boolean isCollsionWith(Bullet bullet) {
  int x2 = bullet.bulletX;
  int y2 = bullet.bulletY;
  int w2 = bullet.bmpBullet.getWidth();
  int h2 = bullet.bmpBullet.getHeight();
  if (x >= x2 && x >= x2 + w2) {
   return false;
  } else if (x <= x2 && x + frameW <= x2) {
   return false;
  } else if (y >= y2 && y >= y2 + h2) {
   return false;
  } else if (y <= y2 && y + frameH <= y2) {
   return false;
  }
  //发生碰撞,让其死亡
  isDead = true;
  return true;
 }

在主视图中添加逻辑 主角子弹和敌机碰撞的逻辑

 //处理主角子弹与敌机碰撞
    for (int i = 0; i < vcBulletPlayer.size(); i++) {
     //取出主角子弹容器的每个元素
     Bullet blPlayer = vcBulletPlayer.elementAt(i);
     for (int j = 0; j < vcEnemy.size(); j++) {
      //添加爆炸效果 待完成
     }
    }

上面完成 的效果图

下面完成爆炸的效果

新建一个爆炸类 Boom

/**
 * 瞬间爆炸 类
 * @author liuml
 * @time 2016-6-1 上午11:32:56
 */
public class Boom {
 //爆炸效果资源图
 private Bitmap bmpBoom;
 //爆炸效果的位置坐标
 private int boomX, boomY;
 //爆炸动画播放当前的帧下标
 private int cureentFrameIndex;
 //爆炸效果的总帧数
 private int totleFrame;
 //每帧的宽高
 private int frameW, frameH;
 //是否播放完毕,优化处理
 public boolean playEnd;

 //爆炸效果的构造函数
 public Boom(Bitmap bmpBoom, int x, int y, int totleFrame) {
  this.bmpBoom = bmpBoom;
  this.boomX = x;
  this.boomY = y;
  this.totleFrame = totleFrame;
  frameW = bmpBoom.getWidth() / totleFrame;
  frameH = bmpBoom.getHeight();
 }

 //爆炸效果绘制
 public void draw(Canvas canvas, Paint paint) {
  canvas.save();
  canvas.clipRect(boomX, boomY, boomX + frameW, boomY + frameH);
  canvas.drawBitmap(bmpBoom, boomX - cureentFrameIndex * frameW, boomY, paint);
  canvas.restore();
 }

 //爆炸效果的逻辑
 public void logic() {
  if (cureentFrameIndex < totleFrame) {
   cureentFrameIndex++;
  } else {
   playEnd = true;
  }
 }
}

下面就还是老套路了 在主界面 声明爆炸容器 然后绘制 然后是逻辑实现

 package com.gsf;

import java.util.Random;
import java.util.Vector;

import android.content.Context;
import android.content.res.Resources;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.view.KeyEvent;
import android.view.MotionEvent;
import android.view.SurfaceHolder;
import android.view.SurfaceHolder.Callback;
import android.view.SurfaceView;

public class MySurfaceView extends SurfaceView implements Callback, Runnable {
 private SurfaceHolder sfh;
 private Paint paint;
 private Thread th;
 private boolean flag;
 private Canvas canvas;

 // 1 定义游戏状态常量
 public static final int GAME_MENU = 0;// 游戏菜单
 public static final int GAMEING = 1;// 游戏中
 public static final int GAME_WIN = 2;// 游戏胜利
 public static final int GAME_LOST = 3;// 游戏失败
 public static final int GAME_PAUSE = -1;// 游戏菜单
 // 当前游戏状态(默认初始在游戏菜单界面)
 public static int gameState = GAME_MENU;
 // 声明一个Resources实例便于加载图片
 private Resources res = this.getResources();
 // 声明游戏需要用到的图片资源(图片声明)
 private Bitmap bmpBackGround;// 游戏背景
 private Bitmap bmpBoom;// 爆炸效果
 private Bitmap bmpBoosBoom;// Boos爆炸效果
 private Bitmap bmpButton;// 游戏开始按钮
 private Bitmap bmpButtonPress;// 游戏开始按钮被点击
 private Bitmap bmpEnemyDuck;// 怪物鸭子
 private Bitmap bmpEnemyFly;// 怪物苍蝇
 private Bitmap bmpEnemyBoos;// 怪物猪头Boos
 private Bitmap bmpGameWin;// 游戏胜利背景
 private Bitmap bmpGameLost;// 游戏失败背景
 private Bitmap bmpPlayer;// 游戏主角飞机
 private Bitmap bmpPlayerHp;// 主角飞机血量
 private Bitmap bmpMenu;// 菜单背景
 public static Bitmap bmpBullet;// 子弹
 public static Bitmap bmpEnemyBullet;// 敌机子弹
 public static Bitmap bmpBossBullet;// Boss子弹
 public static int screenW;
 public static int screenH;

 // 声明一个敌机容器
 private Vector<Enemy> vcEnemy;
 // 每次生成敌机的时间(毫秒)
 private int createEnemyTime = 50;
 private int count;// 计数器
 // 敌人数组:1和2表示敌机的种类,-1表示Boss
 // 二维数组的每一维都是一组怪物
 private int enemyArray[][] = { { 1, 2 }, { 1, 1 }, { 1, 3, 1, 2 },
   { 1, 2 }, { 2, 3 }, { 3, 1, 3 }, { 2, 2 }, { 1, 2 }, { 2, 2 },
   { 1, 3, 1, 1 }, { 2, 1 }, { 1, 3 }, { 2, 1 }, { -1 } };
 // 当前取出一维数组的下标
 private int enemyArrayIndex;
 // 是否出现Boss标识位
 private boolean isBoss;
 // 随机库,为创建的敌机赋予随即坐标
 private Random random;

 //
 private GameMenu gameMenu;
 private GameBg gameBg;

 private Player player;

 // 敌机子弹容器
 private Vector<Bullet> vcBullet;
 // 添加子弹的计数器
 private int countEnemyBullet;
 // 主角子弹容器
 private Vector<Bullet> vcBulletPlayer;
 // 添加子弹的计数器
 private int countPlayerBullet;

 //爆炸效果容器
 private Vector<Boom> vcBoom;

 /**
  * SurfaceView初始化函数
  */
 public MySurfaceView(Context context) {
  super(context);
  sfh = this.getHolder();
  sfh.addCallback(this);
  paint = new Paint();
  paint.setColor(Color.WHITE);
  paint.setAntiAlias(true);
  setFocusable(true);
 }

 /**
  * SurfaceView视图创建,响应此函数
  */
 @Override
 public void surfaceCreated(SurfaceHolder holder) {
  screenW = this.getWidth();
  screenH = this.getHeight();
  initGame();
  flag = true;
  // 实例线程
  th = new Thread(this);
  // 启动线程
  th.start();
 }

 /**
  * 加载游戏资源
  */
 private void initGame() {
  // 加载游戏资源
  bmpBackGround = BitmapFactory
    .decodeResource(res, R.drawable.background);
  bmpBoom = BitmapFactory.decodeResource(res, R.drawable.boom);
  bmpBoosBoom = BitmapFactory.decodeResource(res, R.drawable.boos_boom);
  bmpButton = BitmapFactory.decodeResource(res, R.drawable.button);
  bmpButtonPress = BitmapFactory.decodeResource(res,
    R.drawable.button_press);
  bmpEnemyDuck = BitmapFactory.decodeResource(res, R.drawable.enemy_duck);
  bmpEnemyFly = BitmapFactory.decodeResource(res, R.drawable.enemy_fly);
  bmpEnemyBoos = BitmapFactory.decodeResource(res, R.drawable.enemy_pig);
  bmpGameWin = BitmapFactory.decodeResource(res, R.drawable.gamewin);
  bmpGameLost = BitmapFactory.decodeResource(res, R.drawable.gamelost);
  bmpPlayer = BitmapFactory.decodeResource(res, R.drawable.player);
  bmpPlayerHp = BitmapFactory.decodeResource(res, R.drawable.hp);
  bmpMenu = BitmapFactory.decodeResource(res, R.drawable.menu);
  bmpBullet = BitmapFactory.decodeResource(res, R.drawable.bullet);
  bmpEnemyBullet = BitmapFactory.decodeResource(res,
    R.drawable.bullet_enemy);
  bmpBossBullet = BitmapFactory
    .decodeResource(res, R.drawable.boosbullet);

  // 菜单类实例化
  gameMenu = new GameMenu(bmpMenu, bmpButton, bmpButtonPress);
  // 实例游戏背景
  gameBg = new GameBg(bmpBackGround);
  // 实例主角
  player = new Player(bmpPlayer, bmpPlayerHp);
  //敌机子弹容器实例
  vcBullet = new Vector<Bullet>();
  //主角子弹容器实例
  vcBulletPlayer = new Vector<Bullet>();
  // 实例敌机容器
  vcEnemy = new Vector<Enemy>();
  // 实例随机库
  random = new Random();
  //爆炸效果容器实例
  vcBoom = new Vector<Boom>();
 }

 /**
  * 游戏绘图
  */
 public void myDraw() {
  try {
   canvas = sfh.lockCanvas();
   if (canvas != null) {
    canvas.drawColor(Color.WHITE);
    // 绘图函数根据游戏状态不同进行不同绘制

    switch (gameState) {
    case GAME_MENU:

     gameMenu.draw(canvas, paint);
     break;
    case GAMEING:
     gameBg.draw(canvas, paint);
     player.draw(canvas, paint);
     if (isBoss == false) {
      // 敌机绘制
      for (int i = 0; i < vcEnemy.size(); i++) {
       vcEnemy.elementAt(i).draw(canvas, paint);
      }
      //敌机子弹绘制
      for (int i = 0; i < vcBullet.size(); i++) {
       vcBullet.elementAt(i).draw(canvas, paint);
      }
      //爆炸效果绘制
      for (int i = 0; i < vcBoom.size(); i++) {
       vcBoom.elementAt(i).draw(canvas, paint);
      }
     } else {
      // boss 绘制

     }
     // 处理主角子弹绘制
     for (int i = 0; i < vcBulletPlayer.size(); i++) {
      vcBulletPlayer.elementAt(i).draw(canvas, paint);
     }

     break;

    case GAME_WIN:

     break;
    case GAME_LOST:

     break;
    case GAME_PAUSE:

     break;
    default:
     break;
    }

   }
  } catch (Exception e) {
   // TODO: handle exception
  } finally {
   if (canvas != null)
    sfh.unlockCanvasAndPost(canvas);
  }
 }

 /**
  * 触屏事件监听
  */
 @Override
 public boolean onTouchEvent(MotionEvent event) {
  switch (gameState) {
  case GAME_MENU:

   gameMenu.onTouchEvent(event);
   break;
  case GAMEING:

   break;

  case GAME_WIN:

   break;
  case GAME_LOST:

   break;
  case GAME_PAUSE:

   break;
  }
  return true;
 }

 /**
  * 按键事件监听
  */
 @Override
 public boolean onKeyDown(int keyCode, KeyEvent event) {
  switch (gameState) {
  case GAME_MENU:

   break;
  case GAMEING:
   player.onKeyDown(keyCode, event);
   break;

  case GAME_WIN:

   break;
  case GAME_LOST:

   break;
  case GAME_PAUSE:
   break;
  }
  return super.onKeyDown(keyCode, event);
 }

 @Override
 public boolean onKeyUp(int keyCode, KeyEvent event) {
  switch (gameState) {
  case GAME_MENU:

   break;
  case GAMEING:
   player.onKeyUp(keyCode, event);
   break;

  case GAME_WIN:

   break;
  case GAME_LOST:

   break;
  case GAME_PAUSE:
   break;
  }
  return super.onKeyUp(keyCode, event);
 }

 /**
  * 游戏逻辑
  */
 private void logic() {
  switch (gameState) {
  case GAME_MENU:

   break;
  case GAMEING:
   gameBg.logic();
   player.logic();
   // 敌机逻辑
   if (isBoss == false) {
    // 敌机逻辑
    for (int i = 0; i < vcEnemy.size(); i++) {
     Enemy en = vcEnemy.elementAt(i);
     // 因为容器不断添加敌机 ,那么对敌机isDead判定,
     // 如果已死亡那么就从容器中删除,对容器起到了优化作用;
     if (en.isDead) {
      vcEnemy.removeElementAt(i);
     } else {
      en.logic();
     }
    }
    // 生成敌机
    count++;
    if (count % createEnemyTime == 0) {
     for (int i = 0; i < enemyArray[enemyArrayIndex].length; i++) {
      // 苍蝇
      if (enemyArray[enemyArrayIndex][i] == 1) {
       int x = random.nextInt(screenW - 100) + 50;
       vcEnemy.addElement(new Enemy(bmpEnemyFly, 1, x, -50));
       // 鸭子左
      } else if (enemyArray[enemyArrayIndex][i] == 2) {
       int y = random.nextInt(20);
       vcEnemy.addElement(new Enemy(bmpEnemyDuck, 2, -50,
         y));
       // 鸭子右
      } else if (enemyArray[enemyArrayIndex][i] == 3) {
       int y = random.nextInt(20);
       vcEnemy.addElement(new Enemy(bmpEnemyDuck, 3,
         screenW + 50, y));
      }
     }
     // 这里判断下一组是否为最后一组(Boss)
     if (enemyArrayIndex == enemyArray.length - 1) {
      isBoss = true;
     } else {
      enemyArrayIndex++;
     }
    }

    // 每2秒添加一个敌机子弹
    countEnemyBullet++;
    if (countEnemyBullet % 40 == 0) {
     for (int i = 0; i < vcEnemy.size(); i++) {
      Enemy en = vcEnemy.elementAt(i);
      // 不同类型敌机不同的子弹运行轨迹
      int bulletType = 0;
      switch (en.type) {
      // 苍蝇
      case Enemy.TYPE_FLY:
       bulletType = Bullet.BULLET_FLY;
       break;
      // 鸭子
      case Enemy.TYPE_DUCKL:
      case Enemy.TYPE_DUCKR:
       bulletType = Bullet.BULLET_DUCK;
       break;
      }
      vcBullet.add(new Bullet(bmpEnemyBullet, en.x + 10,
        en.y + 20, bulletType));
     }
    }
    // 处理敌机子弹逻辑
    for (int i = 0; i < vcBullet.size(); i++) {
     Bullet b = vcBullet.elementAt(i);
     if (b.isDead) {
      vcBullet.removeElement(b);
     } else {
      b.logic();
     }
    }

    //每1秒添加一个主角子弹
    countPlayerBullet++;
    if (countPlayerBullet % 20 == 0) {
     vcBulletPlayer.add(new Bullet(bmpBullet, player.x + 15, player.y - 20, Bullet.BULLET_PLAYER));
    }

    // 处理敌机与主角的碰撞
    for (int i = 0; i < vcEnemy.size(); i++) {
     if (player.isCollsionWith(vcEnemy.elementAt(i))) {
      // 发生碰撞,主角血量-1
      player.setPlayerHp(player.getPlayerHp() - 1);
      // 当主角血量小于0,判定游戏失败
      if (player.getPlayerHp() <= -1) {
       gameState = GAME_LOST;
      }
     }
    }

    //处理主角子弹逻辑
    for (int i = 0; i < vcBulletPlayer.size(); i++) {
     Bullet b = vcBulletPlayer.elementAt(i);
     if (b.isDead) {
      vcBulletPlayer.removeElement(b);
     } else {
      b.logic();
     }
    }

    //处理敌机子弹与主角碰撞
    for (int i = 0; i < vcBullet.size(); i++) {
     if (player.isCollsionWith(vcBullet.elementAt(i))) {
      //发生碰撞,主角血量-1
      player.setPlayerHp(player.getPlayerHp() - 1);
      //当主角血量小于0,判定游戏失败
      if (player.getPlayerHp() <= -1) {
       gameState = GAME_LOST;
      }
     }
    }

    //处理主角子弹与敌机碰撞
    for (int i = 0; i < vcBulletPlayer.size(); i++) {
     //取出主角子弹容器的每个元素
     Bullet blPlayer = vcBulletPlayer.elementAt(i);
     for (int j = 0; j < vcEnemy.size(); j++) {
      //添加爆炸效果
      //取出敌机容器的每个元与主角子弹遍历判断
      if (vcEnemy.elementAt(j).isCollsionWith(blPlayer)) {
       vcBoom.add(new Boom(bmpBoom, vcEnemy.elementAt(j).x, vcEnemy.elementAt(j).y, 7));
      }
     }
    }
   }
   break;

  case GAME_WIN:

   break;
  case GAME_LOST:

   break;
  case GAME_PAUSE:
   break;
  }

 }

 @Override
 public void run() {
  while (flag) {
   long start = System.currentTimeMillis();
   myDraw();
   logic();
   long end = System.currentTimeMillis();
   try {
    if (end - start < 50) {
     Thread.sleep(50 - (end - start));
    }
   } catch (InterruptedException e) {
    e.printStackTrace();
   }
  }
 }

 /**
  * SurfaceView视图状态发生改变,响应此函数
  */
 @Override
 public void surfaceChanged(SurfaceHolder holder, int format, int width,
   int height) {
 }

 /**
  * SurfaceView视图消亡时,响应此函数
  */
 @Override
 public void surfaceDestroyed(SurfaceHolder holder) {
  flag = false;
 }
}

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

时间: 2016-07-23

Android实现消水果游戏代码分享

消水果游戏大家都玩过吧,今天小编给大家分享实现消水果游戏的代码,废话不多说了,具体代码如下所示: #include "InGameScene.h" #include "PauseLayer.h" #include "ScoreScene.h" #include "AppDelegate.h" extern "C" { void showAds() { } void hideAds() { } } using

Android开发之经典游戏贪吃蛇

前言 这款游戏实现的思路和源码参考了Google自带的Snake的例子,其中修改了一些个人认为还不够完善的地方,加入了一些新的功能,比如屏幕上的方向操作盘,暂停按钮,开始按钮,退出按钮.另外,为了稍微增加些用户体验,除了游戏的主界面,本人自己新增了5个界面,分别是登陆界面,菜单界面,背景音乐设置界面,难度设置界面,还有个关于游戏的介绍界面.个人觉得在新手阶段,参考现成的思路和实现方式是难以避免的.重要的是我们需要有自己的理解,读懂代码之后,需要思考代码背后的实现逻辑,形成自己的思维.这样在下次开

Android游戏开发之碰撞检测(矩形碰撞、圆形碰撞、像素碰撞)

本文为大家分享了Android游戏开发之碰撞检测,供大家参考,具体内容如下 矩形碰撞 原理: 两个矩形位置 的四种情况 不是这四中情况 则碰撞 圆形碰撞 原理: 利用两个圆心之间的距离进行判定.当两个圆心的距离小于半径之和则碰撞. 像素碰撞 原理:不适用 遍历所有像素 检测 太多了 多矩形碰撞 原理:设置多个矩形碰撞检测区域 检测碰撞矩形数组 与另一碰撞矩形数组之间的位置关系. 矩形碰撞 代码: public class MySurfaceView extends SurfaceView imp

Android 重力传感器在游戏开发中的应用

手势操作可以说是智能手机的一种魅力所在,前两节给大家讲解了两种有趣的手势操作,将它们置于游戏当中,大大提升了游戏的可玩性和趣味性.本节将继续介绍智能手机的另一种神奇之处:传感器.    一.何为传感器 所谓传感器就是能够探测如光.热.温度.重力.方向等等的装置.    二.Android提供了哪些传感器 1.加速度传感器(重力传感器) 2.陀螺仪传感器 3.光传感器 4.恒定磁场传感器 5.方向传感器 6.恒定的压力传感器 7.接近传感器 8.温度传感器 今天我们给大家介绍的是游戏开发中最最常见

Android实战打飞机游戏之怪物(敌机)类的实现(4)

先看看效果图: 分析: 根据敌机类型区分 敌机 运动逻辑 以及绘制 /** * 敌机 * * @author liuml * @time 2016-5-31 下午4:14:59 */ public class Enemy { // 敌机的种类标识 public int type; // 苍蝇 public static final int TYPE_FLY = 1; // 鸭子(从左往右运动) public static final int TYPE_DUCKL = 2; // 鸭子(从右往左运

打飞机游戏终极BOSS Android实战打飞机游戏完结篇

本文实例为大家分享了打飞机游戏BOSS以及胜利失败页面设计的Android代码,具体内容如下 修改子弹类: public class Bullet { //子弹图片资源 public Bitmap bmpBullet; //子弹的坐标 public int bulletX, bulletY; //子弹的速度 public int speed; //子弹的种类以及常量 public int bulletType; //主角的 public static final int BULLET_PLAYE

Android实战打飞机游戏之无限循环的背景图(2)

首先分析下游戏界面内的元素: 无限滚动的背景图, 可以操作的主角,主角的子弹, 主角的血量,两种怪物(敌机),一个boss, boss的爆炸效果. 先看效果图 1.首先实现无限滚动的背景图 原理: 定义两个位图对象 当第一个位图到末尾是 第二个位图从第一个位图的末尾跟上. public class GameBg { // 游戏背景的图片资源 // 为了循环播放,这里定义两个位图对象, // 其资源引用的是同一张图片 private Bitmap bmpBackGround1; private B

Android实战打飞机游戏之菜单页面设计(1)

本文目标实现控制小飞机的左右移动.躲避子弹.打boss. 本节实现 开始菜单界面 1.首先 资源文件拷过来 2.划分游戏状态 public static final int GAME_MENU = 0;// 游戏菜单 public static final int GAMEING = 1;// 游戏中 public static final int GAME_WIN = 2;// 游戏胜利 public static final int GAME_LOST = 3;// 游戏失败 public

Android实战打飞机游戏之实现主角以及主角相关元素(3)

先看效果图 新建player 类 public class Player { private int playerHp = 3; private Bitmap bmpPlayerHP; // 主角坐标以及位图 private int x, y; private Bitmap bmpPlayer; // 主角移动速度 private int speed = 5; // 主角移动标识 private boolean isUp, isDown, isLeft, isRight; // 主角的构造函数

JavaScript仿微信打飞机游戏

首先实现微信打飞机游戏,首先会有自己和敌机,采用canvas绘图来生成自己和敌人. 1.生成自己,且可以通过左右键来进行左右移动. //生成自己,且可以左右移动 //控制飞机向右移动的函数 function moveRight(event){ context.clearRect(aligh,100,47,47); //防止飞机移除背景外 if(aligh < 260){ var img = new Image(); img.src = "../images/self.png";

JavaScript 小型打飞机游戏实现原理说明

玩法说明:上下左右控制移动,空格发弹. 每打中一个敌机就加100分,每提升5000分,玩家的飞机的一次发弹数就加一,最多四,被敌机撞到或者让敌机飞到底部就算输.... 演示代码:http://demo.jb51.net/js/FlyBeat/index.html 游戏目前的功能还是比较简单的....貌似就贴个源码不太好,所以这次还是写写思路... 游戏主要分为4个js文件,4个js文件分别包含4个类. 1:飞机类---Flyer 复制代码 代码如下: //飞机对应的dom元素 this.dom

C语言实现打飞机游戏

本文实例为大家分享了C语言实现打飞机游戏的具体代码,供大家参考,具体内容如下 #include<stdio.h> #include<stdlib.h> #include<windows.h> #include<time.h> #define U 1 #define D 2 #define L 3 #define R 4 typedef struct node //己方飞机的节点 { int x; int y; struct node *next; }node

python实现微信打飞机游戏

本文实例为大家分享了python实现微信打飞机游戏的具体代码,供大家参考,具体内容如下 import pygame import random import sys #初始化 pygame.init() pygame.display.set_caption('飞机火拼')#设置窗口标题 screen= pygame.display.set_mode((320, 570), 0, 32) pygame.mouse.set_visible(False)#隐藏光标 #加载图片 boom1=pygame