Python 開發植物大戰僵屍游戲

来源:https://www.cnblogs.com/moonhmily/archive/2019/09/20/11553616.html
-Advertisement-
Play Games

本文使用python 的pygame 來編寫一個植物大戰僵屍的完整版游戲 ...


在這裡插入圖片描述

作者:楷楷
鏈接:https://segmentfault.com/a/1190000019418065

開發思路

完整項目地址:

https://github.com/371854496/pygame

更多好玩有趣的python,盡在公眾號「Python專欄」,後臺回覆「機器學習電子書」免費獲取100本機器學習電子書

在這裡插入圖片描述

  1. 引入需要的模塊,配置圖片路徑,設置界面寬高背景顏色,創建游戲主入口。
#1引入需要的模塊
import pygame
import random
#1配置圖片地址
IMAGE_PATH = 'imgs/'
#1設置頁面寬高
scrrr_width=800
scrrr_height =560
#1創建控制游戲結束的狀態
GAMEOVER = False
#1主程式
class MainGame():
     #1載入游戲視窗
    def init_window(self):
        #1調用顯示模塊的初始化
        pygame.display.init()
        #1創建視窗
        MainGame.window = pygame.display.set_mode([scrrr_width,scrrr_height])  #
    #1開始游戲
    def start_game(self):
        #1初始化視窗
        self.init_window()
        #1只要游戲沒結束,就一直迴圈
        while not GAMEOVER:
            #1渲染白色背景
            MainGame.window.fill((255, 255, 255))
            #1實時更新
            pygame.display.update()
#1啟動主程式
if __name__ == '__main__':
    game = MainGame()
    game.start_game()
  1. 文本繪製,創建要動態改變的屬性,渲染的位置
#2 創建關數,得分,剩餘分數,錢數
shaoguan = 1
score = 0
remnant_score = 100
money = 200
#2 文本繪製
def draw_text(self, content, size, color):
    pygame.font.init()
    font = pygame.font.SysFont('kaiti', size)
    text = font.render(content, True, color)
    return text

#2 載入幫助提示
def load_help_text(self):
    text1 = self.draw_text('1.按左鍵創建嚮日葵 2.按右鍵創建豌豆射手', 26, (255, 0, 0))
    MainGame.window.blit(text1, (5, 5))

#2 渲染的文字和坐標位置
        MainGame.window.blit(self.draw_text('當前錢數$: {}'.format(MainGame.money), 26, (255, 0, 0)), (500, 40))
        MainGame.window.blit(self.draw_text(
            '當前關數{},得分{},距離下關還差{}分'.format(MainGame.shaoguan, MainGame.score, MainGame.remnant_score), 26,
            (255, 0, 0)), (5, 40))
        self.load_help_text()
  1. 創建地圖類,初始化地圖和坐標
#3 創建地圖類
class Map():
    #3 存儲兩張不同顏色的圖片名稱
    map_names_list = [IMAGE_PATH + 'map1.png', IMAGE_PATH + 'map2.png']
    #3 初始化地圖
    def __init__(self, x, y, img_index):
        self.image = pygame.image.load(Map.map_names_list[img_index])
        self.position = (x, y)
        # 是否能夠種植
        self.can_grow = True
    #3 載入地圖
    def load_map(self):
         MainGame.window.blit(self.image,self.position)

    #3 存儲所有地圖坐標點
    map_points_list = []
    #3 存儲所有的地圖塊
    map_list = []

    #3 初始化坐標點
    def init_plant_points(self):
        for y in range(1, 7):
            points = []
            for x in range(10):
                point = (x, y)
                points.append(point)
            MainGame.map_points_list.append(points)
            print("MainGame.map_points_list", MainGame.map_points_list)

    #3 初始化地圖
    def init_map(self):
        for points in MainGame.map_points_list:
            temp_map_list = list()
            for point in points:
                # map = None
                if (point[0] + point[1]) % 2 == 0:
                    map = Map(point[0] * 80, point[1] * 80, 0)
                else:
                    map = Map(point[0] * 80, point[1] * 80, 1)
                # 將地圖塊加入到視窗中
                temp_map_list.append(map)
                print("temp_map_list", temp_map_list)
            MainGame.map_list.append(temp_map_list)
        print("MainGame.map_list", MainGame.map_list)

    #3 將地圖載入到視窗中
    def load_map(self):
        for temp_map_list in MainGame.map_list:
            for map in temp_map_list:
                map.load_map()

    #3 初始化坐標和地圖
    self.init_plant_points()
    self.init_map()

    #3 需要反覆載入地圖
    self.load_map()
  1. 創建植物類,圖片載入報錯處理,載入植物方法
#4 圖片載入報錯處理
LOG = '文件:{}中的方法:{}出錯'.format(__file__,__name__)
#4 植物類
class Plant(pygame.sprite.Sprite):
    def __init__(self):
        super(Plant, self).__init__()
        self.live=True

    # 載入圖片
    def load_image(self):
        if hasattr(self, 'image') and hasattr(self, 'rect'):
            MainGame.window.blit(self.image, self.rect)
        else:
            print(LOG)

#4 存儲所有植物的列表
plants_list = []

5.創建嚮日葵類

#5 嚮日葵類
class Sunflower(Plant):
    def __init__(self,x,y):
        super(Sunflower, self).__init__()
        self.image = pygame.image.load('imgs/sunflower.png')
        self.rect = self.image.get_rect()
        self.rect.x = x
        self.rect.y = y
        self.price = 50
        self.hp = 100
        #5 時間計數器
        self.time_count = 0

    #5 功能:生成陽光(生產錢)
    def produce_money(self):
        self.time_count += 1
        if self.time_count == 25:
            MainGame.money += 5
            self.time_count = 0
    #5 嚮日葵加入到視窗中
    def display_sunflower(self):
        MainGame.window.blit(self.image,self.rect)
  1. 創建豌豆射手類
    #6 豌豆射手類
class PeaShooter(Plant):
    def __init__(self,x,y):
        super(PeaShooter, self).__init__()
        # self.image 為一個 surface
        self.image = pygame.image.load('imgs/peashooter.png')
        self.rect = self.image.get_rect()
        self.rect.x = x
        self.rect.y = y
        self.price = 50
        self.hp = 200
        #6 發射計數器
        self.shot_count = 0

    #6 增加射擊方法
    def shot(self):
        #6 記錄是否應該射擊
        should_fire = False
        for zombie in MainGame.zombie_list:
            if zombie.rect.y == self.rect.y and zombie.rect.x < 800 and zombie.rect.x > self.rect.x:
                should_fire = True
        #6 如果活著
        if self.live and should_fire:
            self.shot_count += 1
            # 計數器到25發射一次
            if self.shot_count == 25:
                #6 基於當前豌豆射手的位置,創建子彈
                peabullet = PeaBullet(self)
                #6 將子彈存儲到子彈列表中
                MainGame.peabullet_list.append(peabullet)
                self.shot_count = 0

    #6 將豌豆射手加入到視窗中的方法
    def display_peashooter(self):
        MainGame.window.blit(self.image,self.rect)

    #6 增加豌豆射手發射處理
    def load_plants(self):
        for plant in MainGame.plants_list:
            #6 優化載入植物的處理邏輯
            if plant.live:
                if isinstance(plant, Sunflower):
                    plant.display_sunflower()
                    plant.produce_money()
                elif isinstance(plant, PeaShooter):
                    plant.display_peashooter()
                    plant.shot()
            else:
                MainGame.plants_list.remove(plant)
     #6 調用載入植物的方法
     self.load_plants()
  1. 創建子彈類
#7 豌豆子彈類
class PeaBullet(pygame.sprite.Sprite):
    def __init__(self,peashooter):
        self.live = True
        self.image = pygame.image.load('imgs/peabullet.png')
        self.damage = 50
        self.speed  = 10
        self.rect = self.image.get_rect()
        self.rect.x = peashooter.rect.x + 60
        self.rect.y = peashooter.rect.y + 15

    def move_bullet(self):
        #7 在屏幕範圍內,實現往右移動
        if self.rect.x < scrrr_width:
            self.rect.x += self.speed
        else:
            self.live = False

    #7 新增,子彈與僵屍的碰撞
    def hit_zombie(self):
        for zombie in MainGame.zombie_list:
            if pygame.sprite.collide_rect(self,zombie):
                #打中僵屍之後,修改子彈的狀態,
                self.live = False
                #僵屍掉血
                zombie.hp -= self.damage
                if zombie.hp <= 0:
                    zombie.live = False
                    self.nextLevel()
    #7闖關方法
    def nextLevel(self):
        MainGame.score += 20
        MainGame.remnant_score -=20
        for i in range(1,100):
            if MainGame.score==100*i and MainGame.remnant_score==0:
                    MainGame.remnant_score=100*i
                    MainGame.shaoguan+=1
                    MainGame.produce_zombie+=50

    def display_peabullet(self):
        MainGame.window.blit(self.image,self.rect)

    #7 存儲所有豌豆子彈的列表
    peabullet_list = []

    #7 載入所有子彈的方法
def load_peabullets(self):
    for b in MainGame.peabullet_list:
        if b.live:
            b.display_peabullet()
            b.move_bullet()
            #7  調用子彈是否打中僵屍的方法
            b.hit_zombie()
        else:
            MainGame.peabullet_list.remove(b)
    #7 調用載入所有子彈的方法
    self.load_peabullets()
  1. 事件處理
#8事件處理

def deal_events(self):
    #8 獲取所有事件
    eventList = pygame.event.get()
    #8 遍歷事件列表,判斷
    for e in eventList:
        if e.type == pygame.QUIT:
            self.gameOver()
        elif e.type == pygame.MOUSEBUTTONDOWN:
            # print('按下滑鼠按鍵')
            print(e.pos)
            # print(e.button)#左鍵1  按下滾輪2 上轉滾輪為4 下轉滾輪為5  右鍵 3

            x = e.pos[0] // 80
            y = e.pos[1] // 80
            print(x, y)
            map = MainGame.map_list[y - 1][x]
            print(map.position)
            #8 增加創建時候的地圖裝填判斷以及金錢判斷
            if e.button == 1:
                if map.can_grow and MainGame.money >= 50:
                    sunflower = Sunflower(map.position[0], map.position[1])
                    MainGame.plants_list.append(sunflower)
                    print('當前植物列表長度:{}'.format(len(MainGame.plants_list)))
                    map.can_grow = False
                    MainGame.money -= 50
            elif e.button == 3:
                if map.can_grow and MainGame.money >= 50:
                    peashooter = PeaShooter(map.position[0], map.position[1])
                    MainGame.plants_list.append(peashooter)
                    print('當前植物列表長度:{}'.format(len(MainGame.plants_list)))
                    map.can_grow = False
                    MainGame.money -= 50

                    #8 調用事件處理的方法
                    self.deal_events()
  1. 創建僵屍類
#9 僵屍類
class Zombie(pygame.sprite.Sprite):
    def __init__(self,x,y):
        super(Zombie, self).__init__()
        self.image = pygame.image.load('imgs/zombie.png')
        self.rect = self.image.get_rect()
        self.rect.x = x
        self.rect.y = y
        self.hp = 1000
        self.damage = 2
        self.speed = 1
        self.live = True
        self.stop = False
    #9 僵屍的移動
    def move_zombie(self):
        if self.live and not self.stop:
            self.rect.x -= self.speed
            if self.rect.x < -80:
                #8 調用游戲結束方法
                MainGame().gameOver()

    #9 判斷僵屍是否碰撞到植物,如果碰撞,調用攻擊植物的方法
    def hit_plant(self):
        for plant in MainGame.plants_list:
            if pygame.sprite.collide_rect(self,plant):
                #8  僵屍移動狀態的修改
                self.stop = True
                self.eat_plant(plant)
    #9 僵屍攻擊植物
    def eat_plant(self,plant):
        #9 植物生命值減少
        plant.hp -= self.damage
        #9 植物死亡後的狀態修改,以及地圖狀態的修改
        if plant.hp <= 0:
            a = plant.rect.y // 80 - 1
            b = plant.rect.x // 80
            map = MainGame.map_list[a][b]
            map.can_grow = True
            plant.live = False
            #8 修改僵屍的移動狀態
            self.stop = False

    #9 將僵屍載入到地圖中
    def display_zombie(self):
        MainGame.window.blit(self.image,self.rect)

    #9 新增存儲所有僵屍的列表
    zombie_list = []
    count_zombie = 0
    produce_zombie = 100     

    #9 新增初始化僵屍的方法
    def init_zombies(self):
        for i in range(1, 7):
            dis = random.randint(1, 5) * 200
            zombie = Zombie(800 + dis, i * 80)
            MainGame.zombie_list.append(zombie)

    #9將所有僵屍載入到地圖中
    def load_zombies(self):
        for zombie in MainGame.zombie_list:
            if zombie.live:
                zombie.display_zombie()
                zombie.move_zombie()
                # v2.0 調用是否碰撞到植物的方法
                zombie.hit_plant()
            else:
                MainGame.zombie_list.remove(zombie)

    #9 調用初始化僵屍的方法
        self.init_zombies()

    #9 調用展示僵屍的方法
            self.load_zombies()
            #9 計數器增長,每數到100,調用初始化僵屍的方法
            MainGame.count_zombie += 1
            if MainGame.count_zombie == MainGame.produce_zombie:
                self.init_zombies()
                MainGame.count_zombie = 0
            #9 pygame自己的休眠
            pygame.time.wait(10)
  1. 游戲結束方法
#10 程式結束方法
def gameOver(self):
    MainGame.window.blit(self.draw_text('游戲結束', 50, (255, 0, 0)), (300, 200))
    pygame.time.wait(400)
    global GAMEOVER
    GAMEOVER = True

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 場景 Docker-Compose簡介與Ubuntu Server 上安裝Compose: https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/100902301 Docker Compose基本使用-使用Compose啟動Tomcat ...
  • 每個人應該都訂閱了不少微信公眾號,那你有沒有註意到微信公眾號的消息呢?你訂閱的公眾號號主每發佈一篇文章,你都會主動的接收到文章的推送,並不需要你點開每個訂閱的公眾號一一查看有沒有更新,是不是覺得有點意思?感興趣?那就接著往下看吧,因為接下來我們要模擬公眾號群發的場景。 要模擬公眾號群發,首先需要簡單 ...
  • 最開始接觸AOP這個概念,是在大學Java課程中(具體哪本忘記了,JavaWeb?)接觸到的。當時的理解就是,一個請求過來,自上而下,突然從中間切一刀。從那個圖是這樣理解的,文字描述的都忘記了。關於AOP的博客有好多,在工作中需要用到,我也是看著博客,外加視頻學習來理解的。 http://wayfa ...
  • 一個對象的動作觸發多個對象的行為,通過觀察者可以去掉對象的依賴,支持各種自定義和擴展。 觀察者模式,還要從那隻申請的貓開始說起。 貓叫一聲之後觸發: Baby Cry()、Brother Trun()、Dog Wang()、Father Roar()、Mothor Whisper()、Mouse R ...
  • 適配器模式:解決重構的問題,新東西和舊系統不吻合,通過組合/繼承進行配 適配器:插座,電源適配器,做個轉接的 程式員已經確定好規範IHelper,新增了一個RedisHelper 第三方,二者規範不一致,就是沒實現介面 /// <summary> /// 數據訪問介面 /// </summary> ...
  • 抽象工廠:創建對象,創建一組密不可分的對象 創建產品簇,多個對象是個整體,不可分割 工廠+約束 傾斜的可擴展性設計,增加產品很麻煩 /// <summary> /// War3種族之一 /// </summary> public class Human : IRace { public Human( ...
  • 單例的基礎上升級了一下,把對象從記憶體層面複製了一下,然後返回。是個新對象,但是又不是new出來的。 ...
  • 適配器模式(Adapter Pattern): 將某個類的介面轉換成客戶端期望的另一個介面表示,主要的目的是相容性,讓原本因介面不匹配不能一起工作的兩個類可以協同工作。如讀卡器是作為記憶體卡和筆記本之間的適配器,需要將記憶體卡插入讀卡器,再將讀卡器插入筆記本,這樣筆記本就可以讀取記憶體卡了。 適配器模式的 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...