python-pygame實現飛機大戰-2-添加敵機以及碰撞爆炸


承接上一部,上一部講到實現了添加游戲框,游戲背景,以及玩家飛機的運動

 

 這一次的教程的目的在於實現生成敵機、敵機的移動、敵機與玩家的碰撞后,兩者的摧毀,重生等
如下:

 

 先准備資源,網絡上也可以下載到,我這里直接用的教程的資源,為什么呢(確實好看,自己找的慘不忍睹)

 

 然后不多說,直接上代碼,基本上每一行能進行的注釋以及方法結束都寫得很詳細了,如果有描述的不是很正確的地方,可以評論一下(我是小菜雞,請大佬輕點噴)
enemy.py(敵機類,包含敵機的屬性、運行、重置等)
其中的知識點是1.”基類的初始化“

import pygame
from random import *


# 小型飛機類
# pygame.sprite.Sprite就是Pygame里面用來實現精靈的一個類,使用時,並不需要對它實例化,只需要繼承他,然后按需寫出自己的類就好了,因此非常簡單實用
# 所有精靈在建立時都是從pygame.sprite.Sprite中繼承的
class SmallEnemy(pygame.sprite.Sprite):
    def __init__(self, bg_size):
        # 這里涉及到”基類的初始化“建用百度好好理解一下。
        # 通俗來說,SmallEnemy類繼承了父類pygame.sprite.Sprite,子類父類都有__init__()這個函數,如果子類不實現這個init()函數,
        # 那么初始化時直接調用父類的初始化函數,如果子類實現這個init()函數,就覆蓋了父類的這個函數,既然繼承父類,就要在這個函數里顯式調用一下父類的__init__()
        pygame.sprite.Sprite.__init__(self)
        # 敵機圖片
        self.image = pygame.image.load('images/enemy1.png').convert_alpha()
        # 敵機摧毀圖片
        self.destory_images = []
        self.destory_images.extend([
            pygame.image.load('images/enemy1_down1.png').convert_alpha(),
            pygame.image.load('images/enemy1_down2.png').convert_alpha(),
            pygame.image.load('images/enemy1_down3.png').convert_alpha(),
            pygame.image.load('images/enemy1_down4.png').convert_alpha()
        ])
        # 定義屏幕寬高
        self.width = bg_size[0]
        self.height = bg_size[1]

        # get_rect()是一個處理矩形圖像的方法,返回值包含矩形的各屬性,這里返回敵機圖片的位置,可以獲取圖片的寬高等屬性
        self.rect = self.image.get_rect()

        # 隨機生成飛機的位置,randint(a,b)即生成a<=n<=b,即在屏幕寬度,以及負5倍的高度下隨機生成
        self.rect.left = randint(0, self.width - self.rect.width)
        self.rect.top = randint(-5 * self.height, 0)

        # 小型敵機速度
        self.speed = 2
        # 敵機是否存活狀態
        self.active = True
        # 飛機碰撞檢測,會忽略掉圖片中白色的背景部分
        self.mask = pygame.mask.from_surface(self.image)

    # 小型敵機向下移動
    def samll_enemy_move(self):
        # 飛機還未飛出屏幕外,就向下運動(這里可以自己修改飛機的飛行軌跡,比如x軸的隨機左右飛行等,怎么浪怎么來)
        if self.rect.top < self.height:
            self.rect.top += self.speed
        else:
            self.reset()

    # 飛機飛出屏幕外后,別浪費,重置其位置(或者摧毀也行,反正得處理,不然內存會炸)
    def reset(self):
        # 這里寫不寫這個感覺都行,因為都是存活狀態,寫了更保險
        self.active = True
        # 隨機重置飛機的位置,跟上面生成得一樣
        self.rect.left = randint(0, self.width - self.rect.width)
        self.rect.top = randint(-5 * self.height, 0)

這里的mian是在上一部的基礎上添加的,即調用敵機類,生成小型敵機,然后敵機運動,如果與玩家飛機碰撞,則兩者都摧毀,且玩家生命減一
main.py
主要涉及的知識點:
1.pygame.sprite.Group()函數可以創建一個精靈組,從而統一管理,以及Group對應的方法
2.pygame.USEREVENT代表事件1,pygame.time.set_timer:就是每隔一段時間(這里是3毫秒 * 1000 = 3s),去執行一些動作,然后通過event.type == invincible_event去捕獲事件的發生
3.碰撞檢測,pygame.sprite.spritecollide(sprite,sprite_group,bool):一個組中的所有精靈都會逐個地對另外一個單個精靈進行沖突檢測

import pygame
import sys
import traceback
from pygame.locals import *
from random import *
import myplane
import enemy

# 初始化
pygame.init()
# 設置窗口大小
bg_size = width, height = 400, 700  # 實際上是元組
screen = pygame.display.set_mode(bg_size)  # 設置窗口
pygame.display.set_caption("飛機大戰")  # 窗口標題
# 加載背景圖片,對於普通圖像的顯示效果有沒有convert都是一樣的,但是 使用 convert 可以轉換格式,提高 blit 的速度
background = pygame.image.load("images/background.png").convert()

# 設置黑、綠、紅、百幾種顏色對應值,后面會用到
BLACK = (0, 0, 0)
GREEN = (0, 255, 0)
RED = (255, 0, 0)
WHITE = (255, 255, 255)

# 生成敵方小型飛機
def add_small_enemy(small_enemies, enemiesGroup, num):
    for i in range(num):
        smallenemy = enemy.SmallEnemy(bg_size)
        # 精靈組來實現多個圖像,很適合處理精靈列表,有添加,移除,繪制,更新等方法
        # Group.sprites 精靈組
        # Group.copy 復制
        # Group.add 添加
        # Group.remove 移除
        # Group.has 判斷精靈組成員
        # Group.update 更新
        # Group.draw 位塊顯示
        # Group.clear - 繪制背景
        # Group.empty 清空
        # 將這一組敵機都添加上小型飛機屬性,相當於統一處理,統一賦值
        small_enemies.add(smallenemy)
        enemiesGroup.add(smallenemy)

def main():
    # 創建時鍾對象(可以控制游戲循環頻率)
    clock = pygame.time.Clock()

    # 生成玩家飛機
    me = myplane.MyPlane(bg_size)

    # 存放所有敵方飛機,這個飛機組包含了小型飛機、中型飛機、大型飛機的各種屬性,只要用於處理碰撞
    # 當程序中有大量的實體的時候,操作這些實體將會是一件相當麻煩的事
    # 使用pygame.sprite.Group()函數可以創建一個精靈組,從而統一管理,這里創建了一個敵機組
    enemiesGroup = pygame.sprite.Group()

    # 生成地方小型飛機,敵方小型飛機也是一個組,進行統一處理
    small_enemies = pygame.sprite.Group()
    add_small_enemy(small_enemies, enemiesGroup, 15)

    # 玩家三條命
    life_num = 3
    # 設置無敵時間事件,pygame.USEREVENT代表事件1,pygame.USEREVENT+1代表事件2,以此類推,這里相當於定義了一個事件
    invincible_event = pygame.USEREVENT

    # 游戲暫停,默認為非暫停狀態
    paused = False

    # 控制玩家飛機圖片切換,展示突突突的效果
    switch_image = True
    # 切換延時
    delay = 100

    # 游戲分數
    score = 0

    # 飛機爆炸的圖片下標,依次為小型敵機,中型敵機,大型敵機,玩家飛機的爆炸的圖片的下標,切換下標來改變爆炸圖片
    e1_destory_index = 0
    e2_destory_index = 0
    e3_destory_index = 0
    me_destory_index = 0

    running = True
    while running:
        # 獲取事件
        for event in pygame.event.get():
            # 結束事件觸發結束操作
            if event.type == QUIT:
                pygame.quit()
                sys.exit()
            # 在觸發碰撞的時候,寫了pygame.time.set_timer(invincible_event, 3*1000)
            # 意思就是3秒后將會執行invincible_event事件,這里捕獲了invincible_event事件,執行后,將取消這個計時器,防止循環重復執行,等待下一次觸發
            if event.type == invincible_event:
                # 解除無敵狀態
                me.invincible = False
                pygame.time.set_timer(invincible_event, 0)


        # 檢測用戶鍵盤操作,分別為上下左右
        key_pressed = pygame.key.get_pressed()
        if key_pressed[K_w] or key_pressed[K_UP]:
            me.moveUp()
        if key_pressed[K_s] or key_pressed[K_DOWN]:
            me.moveDown()
        if key_pressed[K_a] or key_pressed[K_LEFT]:
            me.moveLeft()
        if key_pressed[K_d] or key_pressed[K_RIGHT]:
            me.moveRight()


        # 在屏幕上面繪制背景圖像,並指定位置
        screen.blit(background, (0, 0))

        # 繪制子彈補給、炸彈補給等各種元素
        # 未暫停且生命大於0
        if paused == False and life_num > 0:
            # 繪制小型敵機,這里是由於上面定義了小型飛機組,飛機組add了小型飛機屬性(速度、位置等),這時候地圖上就生成了飛機
            # 如果這些飛機屬於小型敵機類,即一起處理
            for ei in small_enemies:
                # 敵機是活的,未被擊毀
                if ei.active == True:
                    # 繪制小型敵機,並且敵機開始運動
                    screen.blit(ei.image, ei.rect)
                    ei.samll_enemy_move()
                # 小型敵機被摧毀(被玩家擊毀或者與玩家碰撞)
                else:
                    # 這里設置delay % 4是指爆炸畫面為4幀(個人猜測),理解為爆炸停留時間,可自行設置
                    if not (delay % 4):
                        # 用於播放爆炸聲音,每一架敵機只有一次
                        if e1_destory_index == 0:
                            print("播放敵機爆炸聲音")
                        # 繪制敵機撞擊爆炸畫面
                        screen.blit(ei.destory_images[e1_destory_index], ei.rect)
                        # 切換爆炸圖片下標,從而切換爆炸圖片
                        e1_destory_index = (e1_destory_index + 1) % 4
                        # 經歷完一輪爆炸的敵機,可以將其銷毀,也可以重生,都是不能不處理,不然會一直爆炸、爆炸
                        # 這里選擇將其重生
                        if e1_destory_index == 0:
                            ei.reset()
                            score += 1000
                            print("得分:", score)

            # 做碰撞檢測,pygame.sprite.spritecollide(sprite,sprite_group,bool):一個組中的所有精靈都會逐個地對另外一個單個精靈進行沖突檢測,發生沖突的精靈會作為一個列表返回。
            # 第一個參數就是單個精靈,第二個參數是精靈組,第三個參數是一個bool值,最后這個參數起了很大的作用。當為True的時候,會刪除組中所有沖突的精靈,False的時候不會刪除沖突的精靈
            # 第四個參數是:兩個精靈之間的像素遮罩檢測
            enemy_collide = pygame.sprite.spritecollide(me, enemiesGroup, False, pygame.sprite.collide_mask)
            # 碰撞處理,如果不是無敵狀態下發生碰撞
            if enemy_collide and not me.invincible:
                # 玩家飛機觸發摧毀狀態
                me.active = False
                # enemy_collide是一個列表,存儲所有跟玩家飛機發生碰撞的敵機,然后把碰撞的敵機狀態置為摧毀狀態
                for ei in enemy_collide:
                    ei.active = False

            # 繪制玩家飛機,如果飛機為激活狀態
            if me.active:
                # 在屏幕上繪制玩家飛機,switch_image為是否切換圖片
                if switch_image:
                    screen.blit(me.image1, me.rect)
                # 切換一下飛行圖片
                else:
                    screen.blit(me.image2, me.rect)
            # 代表飛機遭到碰撞,激活爆炸事件
            else:
                if not (delay % 4):
                    # 用於播放爆炸聲音,每一架敵機只有一次
                    if me_destory_index == 0:
                        print("玩家飛機爆炸聲音")
                    # 繪制玩家撞擊爆炸畫面
                    screen.blit(me.destory_image[me_destory_index], me.rect)
                    # 切換爆炸圖片下標,從而切換爆炸圖片
                    me_destory_index = (me_destory_index + 1) % 4
                    # 爆炸畫面播放完之后飛機重生
                    if me_destory_index == 0:
                        # 生命減一條,如果見到0,會自動跳過上一級循環
                        life_num -= 1
                        # 重置狀態
                        me.reset()
                        # 無敵時間設置為3秒,3秒后,觸發無敵時間事件,pygame.time.set_timer:就是每隔一段時間(這里是3毫秒 * 1000 = 3s),去執行一些動作
                        pygame.time.set_timer(invincible_event, 3 * 1000)


            delay -= 1
            if delay == 0:
                delay = 100
            # 每5幀切換一下飛行圖片樣式
            if delay % 5 == 0:
                switch_image = not switch_image


        # 更新整個待顯示的  Surface 對象到屏幕上,將內存中的內容顯示到屏幕上
        pygame.display.flip()
        # 通過時鍾對象指定循環頻率,每秒循環60次
        # 幀速率是指程序每秒在屏幕山繪制圖
        clock.tick(60)

if __name__ == "__main__":
    try:
        main()
    # 服務正常退出
    except SystemExit:
        print("游戲正常退出!")
        # pass忽略錯誤並繼續往下運行,其實這里以及退出了
        pass
    # 服務出現其他的異常
    except:
        # 直接將錯誤打印出來
        traceback.print_exc()
        pygame.quit()

上一部的myplane.py,這里沒有添加其他額外的東西了,這里也貼上去:

import pygame

# 玩家飛機類,pygame.sprite模塊里面包含了一個名為Sprite類,他是pygame本身自帶的一個精靈。
class MyPlane(pygame.sprite.Sprite):
    def __init__(self, bg_size):
        # convert_alpha()更改圖像的像素格式,包括每個像素的alpha,相當於圖片背景變為透明
        self.image1 = pygame.image.load('images/me1.png').convert_alpha()
        self.image2 = pygame.image.load('images/me2.png').convert_alpha()
        # 飛機摧毀圖片,以數字形式保存
        self.destory_image = []
        self.destory_image.extend([
            pygame.image.load('images/me_destroy_1.png').convert_alpha(),
            pygame.image.load('images/me_destroy_2.png').convert_alpha(),
            pygame.image.load('images/me_destroy_3.png').convert_alpha(),
            pygame.image.load('images/me_destroy_4.png').convert_alpha()
        ])
        # 定義屏幕寬高
        self.width = bg_size[0]
        self.height = bg_size[1]

        # get_rect()是一個處理矩形圖像的方法,返回值包含矩形的各屬性,這里返回飛機圖片1的位置,可以獲取圖片的寬高等屬性
        self.rect = self.image1.get_rect()

        # 飛機的初始化位置,//是整除,位置居中以及高度為圖片下框離屏幕最下方60
        self.rect.left = (self.width - self.rect.width)//2
        self.rect.top = self.height - self.rect.height - 60

        # 設置飛機的速度
        self.myPlaneSpeed = 10
        self.active = True

        # 設置飛機是否是無敵狀態(重生3秒內無敵)
        self.invincible = False

        # 飛機碰撞檢測,會忽略掉圖片中白色的背景部分,從指定 Surface 對象中返回一個 Mask
        # 用於快速實現完美的碰撞檢測,Mask 可以精確到 1 個像素級別的判斷。
        # Surface 對象中透明的部分設置為 1,不透明部分設置為 0。
        self.mask = pygame.mask.from_surface(self.image1)

    # 玩家飛機向上移動
    def moveUp(self):
        # 說明還沒定格,即還未到達游戲界面上邊界
        if self.rect.top > 0:
            self.rect.top -= self.myPlaneSpeed
        # 說明移動到達上邊界了
        else:
            self.rect.top = 0

    # 玩家飛機向下移動
    def moveDown(self):
        # 底部需要划出60的高度用來展示其他數據(炸彈數,生命數等)
        if self.rect.bottom < self.height - 60:
            # self.rect.bottom指的是飛機圖片下邊界
            self.rect.bottom += self.myPlaneSpeed
        else:
            self.rect.bottom = self.height - 60

    # 玩家飛機向左移動
    def moveLeft(self):
        if self.rect.left > 0:
            self.rect.left -= self.myPlaneSpeed
        else:
            self.rect.left = 0

    # 玩家飛機向右移動
    def moveRight(self):
        if self.rect.right < self.width:
            self.rect.right += self.myPlaneSpeed
        else:
            self.rect.right = self.width

    # 玩家飛機重生
    def reset(self):
        self.active = True
        # 重生時處於無敵狀態
        self.invincible = True
        # 重生飛機的初始化位置,//是整除,位置居中以及高度為圖片下框離屏幕最下方60
        self.rect.left = (self.width - self.rect.width) // 2
        self.rect.top = self.height - self.rect.height - 60

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM