Android Canvas打飛機之被打中的主角

<a href="http://www.lxweimin.com/p/55d93ffc7138">上一篇</a>實現了主角打中敵機,敵機死亡并且顯示爆炸效果,今天這里主要來實現主角被敵機碰撞以及被敵機子彈打中的效果和邏輯處理。

device-2017-01-06-161922.png
1.打開主角類DrawPlayer
  • 先給主角定義一個血量值
/**
 * 主角血量
 */
private float mLife =20;
  • 主角被敵機擊中以后在一定時間內處于無敵狀態,不再受任何傷害。
/**
     * 是否被擊中,被擊中后閃爍 并且一定時間內無敵。
     */
    private boolean isCollision=true;
    private int noCollisionCount;```

- 新增兩個方法處理主角和敵機的碰撞以及和敵機子彈的碰撞

/**
* 和敵機碰撞
* @param en
* @return
*/
public boolean isCollisionWith(DrawEnemy en){
if(!isCollision){
//獲取到敵機的xy坐標
float ex = en.getEnemyX();
float ey = en.getEnemyY();
int ew = en.getWidth()/2;
int eh = en.getHeight()/2;
//得到主角當前位置,主角的高寬和敵機的坐標位置進行對比,如果坐標范圍出現重疊 說明敵機和主角碰撞了。
if(getPlayerX()+getWidth()/2<=ex||getPlayerX()-getWidth()/2>=ex+ew){
return false;
}
if(getPlayerY()>=ey+eh||getPlayerY()+getHeight()<=ey){
return false;
}
isCollision=true;
return true;
}else{
return false;
}
}```

/**
     * 和敵機子彈碰撞
     * @param bullet
     * @return
     */
    public boolean isCollisionWith(DrawEnemyBullet bullet){
        if(!isCollision){
            //獲取敵機子彈的xy坐標
            float bx = bullet.getBulletX();
            float by = bullet.getBulletY();
            float bw = bullet.getWidth();
            float bh=bullet.getHeight();
            //計算邏輯和上面主角敵機碰撞一樣,每一顆子彈也是一個對象。
            if(getPlayerX()+getWidth()/2<=bx||getPlayerX()>=bx+bw){
                return false;
            }
            if(getPlayerY()>=by+bh||getPlayerY()+getHeight()/2<=by){
                return false;
            }
            isCollision=true;
            return true;
        }else{
            return false;
        }
    }```

這里的實現邏輯和上一篇的碰撞邏輯一樣,通過獲取敵機每一顆子彈或者敵機的xy坐標以及主角的xy坐標判斷是否出現范圍重疊,如果重疊說明發生碰撞。

- 繪制主角的血槽在最頂部位置。

int screenW = ScreenUtils.getScreenWidth(getContext());
mPaint.setColor(Color.WHITE);
int lifeH = SizeUtils.dp2px(getContext(),20);
mPaint.setStrokeWidth(SizeUtils.dp2px(getContext(),3));
canvas.drawLine(screenW/6, lifeH,screenW/5+ mLife *50,lifeH,mPaint);```

  • 最后在updateGame方法中處理無敵狀態的刷新
@Override
    void updateGame() {
        if(isCollision){
            noCollisionCount++;
            if(noCollisionCount>=60){
                isCollision=false;
                noCollisionCount=0;
            }
        }
    }```

上面就是在主角類新增的內容,內容不多,主要新增一個血槽和被敵機擊中以及無敵狀態的處理。

######2.回到GameView類調用上面的方法。
- 在<b>onDrawEnemy</b>方法中循環繪制敵機的時候調用主角類的<b>isCollisionWith(DrawEnemy en)</b>方法判斷是否發生碰撞,主要判斷的是敵機和主角的碰撞。

![Paste_Image.png](http://upload-images.jianshu.io/upload_images/3982371-4a87187ddb1793f2.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

如果發生碰撞,主角掉三滴血,敵機死亡并且添加死亡爆炸效果。

- 在<b>dieEnemyBullet</b>方法中循環繪制敵機子彈的時候調用主角類的<b>isCollisionWith(DrawEnemyBullet bullet)</b>方法判斷是否發生碰撞,主要是判斷敵機子彈和主角的碰撞。


![Paste_Image.png](http://upload-images.jianshu.io/upload_images/3982371-41f63567414d6c47.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

主角和敵機子彈碰撞,敵機該子彈失效,主角掉一滴血。

- 新增一個游戲狀態

private final static int OVER=3;//主角死亡 游戲結束

- 在<b>onGameDraw</b>中需要每次都去判斷主角的血量,如果<=0說明主角已經死亡,游戲結束。


![Paste_Image.png](http://upload-images.jianshu.io/upload_images/3982371-02272b8b78709681.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240)

- 游戲結束后彈出提示框,因為繪制過程是在非主線程中完成的,但是彈出提示框需要通過主UI完成,所以我這里采用<a >Handler</a>顯示提示框。

Handler handler = new Handler(){
@Override
public void handleMessage(Message msg) {
super.handleMessage(msg);
if(mDialog==null){
mDialog = new AlertDialog.Builder(getContext());
mDialog.setMessage("戰機被摧毀,游戲結束");
mDialog.setTitle("提示");
mDialog.setPositiveButton("確認", new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
((Activity)getContext()).finish();
}
});
mDialog.setCancelable(true);
mDialog.show();
}
}
};```
<b>GameView</b>中新增內容不多,主要在之前已經實現的方法中新增一些邏輯處理來完善游戲思路。

Canvas學習過程差不多就快結束了,下一篇就是游戲中的大BOSS了。

<a >源碼</a>已經通過Git更新。

<a href="http://www.lxweimin.com/p/55d93ffc7138">上一篇</a>

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,182評論 6 543
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,489評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,290評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,776評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,510評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,866評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,860評論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,036評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,585評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,331評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,536評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,058評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,754評論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,154評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,469評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,273評論 3 399
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,505評論 2 379

推薦閱讀更多精彩內容