pygame 精靈的行走及二段跳實現方法

来源:http://www.cnblogs.com/is-ztx/archive/2017/07/10/7143771.html
-Advertisement-
Play Games

不得不承認《Python游戲編程入門》這本書翻譯、排版非常之爛,但是裡面的demo還是很好的,之前做了些改編放到這裡。 先是素材: 背景 精靈 所有素材均取自此書 接下來就是精靈類的創建了: 將精靈類“放置”到游戲屏幕上,並加上背景 這樣的話精靈就在畫布上了,我們得讓它能左右移動: 然後實現跳躍及二 ...


不得不承認《Python游戲編程入門》這本書翻譯、排版非常之爛,但是裡面的demo還是很好的,之前做了些改編放到這裡。

先是素材:

背景

精靈

所有素材均取自此書

接下來就是精靈類的創建了:

 

class MySprite(pygame.sprite.Sprite):
    def __init__(self, target):
        pygame.sprite.Sprite.__init__(self)
        self.master_image = None
        self.frame = 0
        self.old_frame = -1
        self.frame_width = 1
        self.frame_height = 1
        self.first_frame = 0
        self.last_frame = 0
        self.columns = 1
        self.last_time = 0

    #  使用property方法,讓精靈類對坐標操作更方便
    def _getx(self):
        return self.rect.x

    def _setx(self, value):
        self.rect.x = value

    X = property(_getx, _setx)

    def _gety(self):
        return self.rect.y

    def _sety(self, value):
        self.rect.y = value

    Y = property(_gety, _sety)

    def _getpos(self):
        return self.rect.topleft

    def _setpos(self, pos):
        self.rect.topleft = pos

    position = property(_getpos, _setpos)


  #  load方法中定義了圖片位置,長寬和幀的列數,由此來將素材切成一幀一幀
def load(self, filename, width, height, columns): self.master_image = pygame.image.load(filename).convert_alpha() self.frame_width = width self.frame_height = height self.rect = Rect(0, 0, width, height) self.columns = columns rect = self.master_image.get_rect() self.last_frame = (rect.width // width) * (rect.height // height) - 1 def update(self, current_time, rate=30): #  更新幀數 if current_time > self.last_time + rate: self.frame += 1 if self.frame > self.last_frame: self.frame = self.first_frame self.last_time = current_time #  當幀數發生改變時,創建新的圖片 if self.frame != self.old_frame: frame_x = (self.frame % self.columns) * self.frame_width frame_y = (self.frame // self.columns) * self.frame_height rect = Rect(frame_x, frame_y, self.frame_width, self.frame_height) self.image = self.master_image.subsurface(rect) self.old_frame = self.frame

 

 將精靈類“放置”到游戲屏幕上,並加上背景

pygame.init()
screen = pygame.display.set_mode((800, 600))
font = pygame.font.Font(None, 24)
framerate = pygame.time.Clock()

bg = pygame.image.load("background.png").convert_alpha()
pl = pygame.image.load('caveman.png').convert_alpha()
# 創建精靈組
group = pygame.sprite.Group()


player = MySprite(screen)
player.load("caveman.png", 50, 64, 8)
player.first_frame = 1
player.last_frame = 7
player.position = 400, 303
group.add(player)

while True:
    for event in pygame.event.get():
        if event.type == QUIT:
            sys.exit()
    #   設置幀數
    framerate.tick(30)
    ticks = pygame.time.get_ticks()

 

 這樣的話精靈就在畫布上了,我們得讓它能左右移動:

 

keys = pygame.key.get_pressed()
    if keys[K_ESCAPE]:
        sys.exit()
    if keys[K_RIGHT]:
        player.X += 8
    if keys[K_LEFT]:
        if player.X > 0:
            player.X -= 8

 

 

 

然後實現跳躍及二段跳躍

這裡需要說下二段跳躍的註意點:

1.直到落地前,只能跳兩次,也就是說精靈進行二次跳躍後不能再跳了

2.按下空格後,精靈的加速度重置

,這需要修改前面的代碼:

jump_vel = 0.0
#   設置一個記錄跳躍次數的變數
space_number = 0
#   跳躍判斷
player_jumping = False
player_start_y = player.Y

while True:
    for event in pygame.event.get():
        if event.type == QUIT:
            sys.exit()
        if event.type == KEYDOWN:
            if event.key == K_SPACE:
                #   跳躍次數小於2次時,
                if space_number < 2:
                    jump_vel = -15.0
                    space_number += 1
                    player_jumping = True

    keys = pygame.key.get_pressed()
    if keys[K_ESCAPE]:
        sys.exit()
    if keys[K_RIGHT]:
        player.X += 8
    if keys[K_LEFT]:
        if player.X > 0:
            player.X -= 8
    #   設置幀數
    framerate.tick(30)
    ticks = pygame.time.get_ticks()

    #   當按下空格後,jump_vel變數不斷變大,直到接觸地面
    if player_jumping:
        player.Y += jump_vel
        jump_vel += 2
        #   落地後,重置跳躍速度和其他判斷變數
        if player.Y >= player_start_y:
                player_jumping = False
                player.Y = player_start_y
                jump_vel = 0
                space_number = 0
                #   創建背景
    screen.blit(bg, (0, 0))

    # 精靈組更新
    group.update(ticks, 50)
    group.draw(screen)

    pygame.display.update()

 

所有代碼:

import sys, time, random, math, pygame
from pygame.locals import *


class MySprite(pygame.sprite.Sprite):
    def __init__(self, target):
        pygame.sprite.Sprite.__init__(self)
        self.master_image = None
        self.frame = 0
        self.old_frame = -1
        self.frame_width = 1
        self.frame_height = 1
        self.first_frame = 0
        self.last_frame = 0
        self.columns = 1
        self.last_time = 0

    #   使用property方法,讓精靈類對坐標操作更方便
    def _getx(self):
        return self.rect.x

    def _setx(self, value):
        self.rect.x = value

    X = property(_getx, _setx)

    def _gety(self):
        return self.rect.y

    def _sety(self, value):
        self.rect.y = value

    Y = property(_gety, _sety)

    def _getpos(self):
        return self.rect.topleft

    def _setpos(self, pos):
        self.rect.topleft = pos

    position = property(_getpos, _setpos)


    def load(self, filename, width, height, columns):
        self.master_image = pygame.image.load(filename).convert_alpha()
        self.frame_width = width
        self.frame_height = height
        self.rect = Rect(0, 0, width, height)
        self.columns = columns
        rect = self.master_image.get_rect()
        self.last_frame = (rect.width // width) * (rect.height // height) - 1


    def update(self, current_time, rate=30):
        #   更新幀數
        if current_time > self.last_time + rate:
            self.frame += 1
            if self.frame > self.last_frame:
                self.frame = self.first_frame
            self.last_time = current_time

        # 當幀數發生改變時,創建新的圖片
        if self.frame != self.old_frame:
            frame_x = (self.frame % self.columns) * self.frame_width
            frame_y = (self.frame // self.columns) * self.frame_height
            rect = Rect(frame_x, frame_y, self.frame_width, self.frame_height)
            self.image = self.master_image.subsurface(rect)
            self.old_frame = self.frame

pygame.init()
screen = pygame.display.set_mode((800, 600))
font = pygame.font.Font(None, 24)
framerate = pygame.time.Clock()

bg = pygame.image.load("background.png").convert_alpha()
pl = pygame.image.load('caveman.png').convert_alpha()
# 創建精靈組
group = pygame.sprite.Group()


player = MySprite(screen)
player.load("caveman.png", 50, 64, 8)
player.first_frame = 1
player.last_frame = 7
player.position = 400, 303
group.add(player)


jump_vel = 0.0
#   設置一個記錄跳躍次數的變數
space_number = 0
#   跳躍判斷
player_jumping = False
player_start_y = player.Y

while True:
    for event in pygame.event.get():
        if event.type == QUIT:
            sys.exit()
        if event.type == KEYDOWN:
            if event.key == K_SPACE:
                #   跳躍次數小於2次時,
                if space_number < 2:
                    jump_vel = -15.0
                    space_number += 1
                    player_jumping = True

    keys = pygame.key.get_pressed()
    if keys[K_ESCAPE]:
        sys.exit()
    if keys[K_RIGHT]:
        player.X += 8
    if keys[K_LEFT]:
        if player.X > 0:
            player.X -= 8
    #   設置幀數
    framerate.tick(30)
    ticks = pygame.time.get_ticks()

    #   當按下空格後,jump_vel變數不斷變大,直到接觸地面
    if player_jumping:
        player.Y += jump_vel
        jump_vel += 2
        #   落地後
        if player.Y >= player_start_y:
                player_jumping = False
                player.Y = player_start_y
                jump_vel = 0
                space_number = 0
                rush_number = 0

    #   創建背景
    screen.blit(bg, (0, 0))

    # 精靈組更新
    group.update(ticks, 50)
    group.draw(screen)

    pygame.display.update()

 這樣,一個粗糙的、會二段跳的精靈就完成了。

很感謝這本書提供單次跳躍的思路,讓我有思考二段跳的想法。其實像二段跳這類看上去容易,但實現其實還是需要思考一番的。

 


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

-Advertisement-
Play Games
更多相關文章
  • 今天編程時,JavaScript 程式報了這樣的錯誤:Cannot use 'in' operator to search for...,具體錯誤信息如下: 坦白說,這樣的錯誤最難調試。因為它並不指向你所寫的具體代碼,而是泛泛指向了 lib.js 文件(該文件通常是第三方的打包壓縮庫),你幾乎無法依 ...
  • 一、http方法 二、http常用狀態碼 1. 100~199信息狀態碼 2. 200~299成功狀態碼 3. 300 ~ 399重定向狀態碼 4. 400~499錯誤狀態碼 5. 500~599狀態碼 ...
  • 剛開始做NDK 開發的時候,Android Studio 還沒提供了 native C/C++ 設置斷點 調試,我們都是通過輸出 日誌來調試,這樣費時耗力。Android Studio 應該是在 2.2 版本才提供的設置斷點 debug 功能,同時在該版本也提供了 cmake 編譯。 我目前在做 N ...
  • 傳統機器學習依賴良好的特征工程。深度學習解決有效特征難人工提取問題。無監督學習,不需要標註數據,學習數據內容組織形式,提取頻繁出現特征,逐層抽象,從簡單到複雜,從微觀到巨集觀。 稀疏編碼(Sparse Coding),基本結構組合。自編碼器(AutoEncoder),用自身高階特征編碼自己。期望輸入/ ...
  • 換了四種黑蘋果,最終成功了 步驟: 1、升級vs2017, 2、安裝XCODE 8.3 3、安裝vs2017 for mac 企業版 4、啟動vs2017 for mac ,設置xcode 位置 5、打開遠程登錄與屏幕共用 6、打開WINDOWS中的VS2017,在 選項中設置XCODE位置,使用I... ...
  • 詮釋: 1. 破解VIP登陸限制 2.去後門 (自查) 下載地址 :https://pan.baidu.com/s/1eR2rUOM 查毒地址:http://a.virscan.org/a3983f36d31d08a51486501965d04cb5 Xise_V20.0.exe 更新日誌 生成內頁 ...
  • 一、進程、線程及多線程的概念 什麼是多線程呢?不理解。 那什麼是線程呢?說到線程就不得不說說進程。我在網上搜索也搜索了一些資料,大部分所說的進程其實是很抽象的東西。通俗的來講,進程就是一個應用程式開始運行,那麼這個應用程式就會存在一個屬於這個應用程式的進程。 那麼線程就是進程中的基本執行單元,每個進 ...
  • AWT概述 GUI全稱是Graphical User Interface,即圖形用戶界面,即應用程式提供給用戶操作的圖形界面,包括視窗、菜單、按鈕、工具欄和其它各種圖形界面元素 GUI設計提供了豐富的類庫,這些類分別位於java.awt和javax.swing包中,簡稱為AWT和Swing Swin ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...