pygame 簡介 - LanKuDot/MLGame GitHub Wiki
MLGame 專案的遊戲都是使用 pygame 套件撰寫的,因此有 pygame 的知識有助於遊戲開發,本篇會概略介紹 pygame 常用的部分。
安裝套件
$ python -m pip install pygame==1.9.4
座標系統
在 pygame 中,場景原點位在畫面的左上方,往右為 X 正方向,往下為 Y 正方向。用來指定或取得的物件的 xy 座標也是在物件的左上角。
常用物件
以下大略介紹在開發遊戲中常用到的 pygame 物件。記得物件的用途即可,使用方式可以等到在後面的教學中遇到時再回來查詢,或是直接到 pygame 的 API 文件查詢。另外標註的常用函式中,可能會省略一些預設參數,如果需要完整的使用說明,請參考 pygame 的 API 文件。
pygame.display
pygame 的視窗,用來設置要顯示的內容。常用函式:
init()
:初始化視窗set_caption(title)
:設置視窗的標題名稱set_mode(size = (0, 0)) -> Surface
:設置視窗的大小,會回傳一個Surface
物件,為視窗的畫布flip()
:將繪製在視窗畫布上的內容顯示到視窗上
pygame.Surface
Surface
是 pygame 用來表示圖片的物件,同時可以是畫布。一個 surface 可以畫到另一個 surface 上。常用函式:
Surface((width, height))
:建立一個新的 surfacesurface.blit(source, dest) -> Rect
:將source
surface 畫到這個 surface 上。dest
為(x, y)
tuple,用來指定source
要畫在這個 surface 的哪個位置,要注意座標都是指物件的左上角。回傳的Rect
標記的是這個 surface 受影響的像素區域surface.fill(color, rect = None) -> Rect
:在rect
指定的區域填滿顏色color
是一個(r, g, b)
tuple,值為 0 ~ 255。如果沒有指定rect
,則填滿整個 surface。回傳的Rect
一樣是標記受影響的像素區域surface.convert() -> Surface
:將 surface 的像素格式轉換成與視窗畫布一樣,會加快繪製到display
的速度。回傳被轉換過的 surface,注意原本的 surface 並不會改變- 使用
pygame.image.load(filename)
讀取圖片檔案會回傳一個 surface,再使用convert()
轉換成符合視窗畫布的格式,就可以加快繪製的速度
- 使用
pygame.Rect
MLGame 專案中的遊戲經常使用 Rect
來定義遊戲物件的位置及大小。Rect
的四個屬性為 left
、top
、width
、height
,left
、top
即為該物件的左上角座標,同時也是物件的 xy 座標,width
、height
則可以定義出矩形的大小。例如:
>>> from pygame import Rect
>>> rect = Rect(0, 0, 5, 3)
>>> rect
<rect(0, 0, 5, 3)>
>>> print(rect.x, rect.y, rect.width, rect.height)
0 0 5 3
對應的矩形樣貌:
Rect
也提供虛擬屬性或是函式來方便取得或控制矩形的位置與大小:
>>> rect.bottomright
(5, 3)
>>> rect.center
(2, 1)
>>> rect.x = 2
>>> rect
<rect(2, 0, 5, 3)>
>>> rect.move_ip(5, 5)
>>> rect
<rect(7, 5, 5, 3)
另外 pygame.Rect
還有一系列檢測碰撞的函式,如:contains
、colliderect
。
pygame.sprite.Sprite
與 pygame.sprite.Group
MLGame 專案中的可視遊戲物件都會繼承自 Sprite
,主要是為了利用 Group
來將可視物件一次畫到 surface 上,不需個別呼叫 surface.blit()
。常用函式:
pygame.sprite.Group() -> Group
:建立一個新的 groupgroup.add(*sprites)
:將多個 sprite 加到 group 中。sprites
指定一系列的 sprite,例如:group.add(sprite_1, sprite_2, ...)
。只有尚未存在於該 group 中的 sprite 才會被加入group.remove(*sprites)
:將多個 sprite 從 group 中移除group.empty()
:清除所有在 group 中的 spritegroup.draw(surface)
:將 group 內的 sprite 畫到指定的surface
上,繪製 sprite 沒有先後順序。- 使用
draw()
被繪製 sprite 需要擁有兩個屬性:rect
、image
。rect
是一個 rect,代表該物件的位置,而image
是一個 surface,代表該物件的外觀
- 使用
一個簡單的 sprite 例子:
class Ball(pygame.sprite.Sprite):
def __init__(self, color, init_pos_x, init_pox_y):
super().__init__()
self.rect = pygame.rect.Rect(init_pos_x, init_pos_y, 5, 5)
self.image = pygame.surface.Surface(self.rect.w, self.rect.h)
self.image.fill(color)
pygame.sprite
提供檢測碰撞的函式:
pygame.sprite.spritecollide(sprite, group, dokill, collided = None) -> Sprite_list
:偵測sprite
有沒有與group
中的 sprite 碰撞。如果dokill
為True
則被碰撞的 sprite 會從group
中移除。collided
可指定函式用來檢測兩個 sprite 有沒有碰撞。回傳的Sprite_list
是一個 list,存放group
中被碰撞的 sprite。- 使用此函式的的
sprite
與group
中的 sprite 都必須有rect
屬性。
- 使用此函式的的
pygame.sprite.collide_rect(left, right) -> bool
:檢測left
、right
兩個 sprite 是否碰撞,兩個 sprite 必須有rect
屬性。
pygame.font
pygame 用來繪製文字的模組。常用函式:
pygame.font.init()
:初始化文字模組pygame.font.Font(filename, size) -> Font
:建立一個新的文字物件。filename
指定文字檔案,如果為None
則使用預設字體。size
是字體的高度,單位為 pixel。font.render(text, antialias, color, background = None) -> Surface
:建立已有指定文字的 surface。text
為繪製的文字內容,antialias
為True
則文字會較為平滑,color
為文字顏色,background
為文字背景顏色。
繪製文字到螢幕的例子:
import pygame
pygame.display.init()
pygame.font.init()
display_surface = pygame.display.set_mode((100, 100))
font = pygame.font.Font(None, 20)
font_surface = font.render("Hello", True, (255, 255, 255))
display_surface.blit(font_surface, (10, 10))
pygame.display.flip()