用Pygame和Python做遊戲-從入門到精通10

pythontab發表於2012-12-20

用Python和Pygame寫遊戲-從入門到精通(10)

有時候無聊在網上翻翻小說看看,絕大多數那叫一個無聊。比如說修煉的境界分幾種,都有個名字,然後每種境界再有幾層,這不就是變相的打怪練級麼?文筆也不咋樣,故事情節的駕馭能力更是讓我瞠目結舌,想到這些類小說盛行,不覺感到悲從中來。感覺看這些小說,就想在看別人打遊戲一般,崩潰到極點。遊戲和小說的最大區別,除了聲色以外,最不同的就是玩家可以沉入進去,通過自己的雙手來參與;而好的遊戲,更是可以通過玩家的選擇,完全掌控遊戲的發展,這是傳統的故事媒介無法做到的事情。

很自然的,我們講述了遊戲中視覺上的種種,現在開始就要學習一下游戲中的使用者輸入。同樣我們也要探討一下如何讓使用者的輸入更為順暢,換個詞就是,如果讓遊戲的手感更好一些。

遊戲裝置

玩過遊戲的都知道滑鼠和鍵盤是遊戲的不可或缺的輸入裝置。鍵盤可以控制有限的方向和諸多的命令操作,而滑鼠更是提供了全方位的方向和位置操作。不過這兩個裝置並不是為遊戲而生,專業的遊戲手柄給玩家提供了更好的操作感,加上力反饋等技術,應該說遊戲裝置越來越豐富,玩家們也是越來越幸福。

鍵盤裝置

我們先從最廣泛的鍵盤開始講起。

現在使用的鍵盤,基本都是QWERTY鍵盤(看看字幕鍵盤排布的左上就知道了),儘管這個世界上還有其他種類的鍵盤,比如AZERTY啥的,反正我是沒見過,如果你能在寫遊戲的時候考慮到這些特殊使用者自然是最好,個人感覺是問題不大吧。

以前第二部分也稍微使用了一下鍵盤,那時候是用了pygame.event.get()獲取所有的事件,當event.type == KEYDOWN的時候,在判斷event.key的種類,而各個種類也使用K_aK_b……等判斷。這裡再介紹一個pygame.key.get_pressed()來獲得所有按下的鍵值,它會返回一個元組。這個元組的索引就是鍵值,對應的就是是否按下,比如說:

1
2
3
4
    pressed_keys = pygame.key.get_pressed()
    if pressed_keys[K_SPACE]:
        # Space key has been pressed
        fire()pressed_keys = pygame.key.get_pressed()

當然key模組下還有很多函式:

  • key.get_focused —— 當前啟用的pygame視窗
  • key.get_pressed —— 剛剛解釋過了
  • key.get_mods —— 按下的組合鍵(Alt, Ctrl, Shift)
  • key.set_mods —— 你也可以模擬按下組合鍵的效果(KMOD_ALT, KMOD_CTRL, KMOD_SHIFT)
  • key.set_repeat —— 設定允許pygame接受重複按鍵
  • key.name —— 接受鍵值返回鍵名

使用鍵盤控制方向

有了上一章向量的基礎,只需一幅圖就能明白鍵盤如何控制方向:

很多遊戲也使用ASDW當做方向鍵來移動,我們來看一個實際的例子:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
background_image_filename = 'sushiplate.jpg'
sprite_image_filename = 'fugu.png'
 
import pygame
from pygame.locals import *
from sys import exit
from gameobjects.vector2 import Vector2
 
pygame.init()
 
screen = pygame.display.set_mode((640, 480), 0, 32)
 
background = pygame.image.load(background_image_filename).convert()
sprite = pygame.image.load(sprite_image_filename).convert_alpha()
 
clock = pygame.time.Clock()
 
sprite_pos = Vector2(200, 150)
sprite_speed = 300.
 
while True:
 
    for event in pygame.event.get():
        if event.type == QUIT:
            exit()
 
    pressed_keys = pygame.key.get_pressed()
 
    key_direction = Vector2(0, 0)
    if pressed_keys[K_LEFT]:
        key_direction.x = -1
    elif pressed_keys[K_RIGHT]:
        key_direction.x = +1
    if pressed_keys[K_UP]:
        key_direction.y = -1
    elif pressed_keys[K_DOWN]:
        key_direction.y = +1
 
    key_direction.normalize()
 
    screen.blit(background, (0,0))
    screen.blit(sprite, sprite_pos)
 
    time_passed = clock.tick(30)
    time_passed_seconds = time_passed / 1000.0
 
    sprite_pos+= key_direction * sprite_speed * time_passed_seconds
 
    pygame.display.update()

這個例子很簡單,就是使用方向鍵移動小魚。使用的知識也都講過了,相信大家都可以理解。不過這裡並不是單純的判斷按下的鍵來獲得方向,而是通過對方向的加減來獲得最終的效果,這樣可能會更簡短一些,也需要一些技術;如果把方向寫入程式碼,效率更高,不過明顯通用性就要低一些。記得把力氣花在刀刃上!當然這個例子也不是那麼完美,看程式碼、實踐一下都能看到,左方向鍵的優先順序大於右方向鍵,而上則優於下,我們是否有更好的方法?……有興趣的自己考慮~

這個例子我們可以看到,小魚只能在八個方向移動,如何做到全方向?如果你遊戲經驗足一點或許可以想到,是的,先轉向,再移動,儘管不是那麼快捷,但畢竟達到了目標。我們看一下這樣的程式碼怎麼寫:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
background_image_filename = 'sushiplate.jpg'
sprite_image_filename = 'fugu.png'
 
import pygame
from pygame.locals import *
from sys import exit
from gameobjects.vector2 import Vector2
from math import *
 
pygame.init()
 
screen = pygame.display.set_mode((640, 480), 0, 32)
 
background = pygame.image.load(background_image_filename).convert()
sprite = pygame.image.load(sprite_image_filename).convert_alpha()
 
clock = pygame.time.Clock()
 
sprite_pos = Vector2(200, 150)   # 初始位置
sprite_speed = 300.     # 每秒前進的畫素數(速度)
sprite_rotation = 0.      # 初始角度
sprite_rotation_speed = 360. # 每秒轉動的角度數(轉速)
 
while True:
 
    for event in pygame.event.get():
        if event.type == QUIT:
            exit()
 
    pressed_keys = pygame.key.get_pressed()
 
    rotation_direction = 0.
    movement_direction = 0.
 
    # 更改角度
    if pressed_keys[K_LEFT]:
        rotation_direction = +1.
    if pressed_keys[K_RIGHT]:
        rotation_direction = -1.
    # 前進、後退
    if pressed_keys[K_UP]:
        movement_direction = +1.
    if pressed_keys[K_DOWN]:
        movement_direction = -1.
 
    screen.blit(background, (0,0))
 
    # 獲得一條轉向後的魚
    rotated_sprite = pygame.transform.rotate(sprite, sprite_rotation)
    # 轉向後,圖片的長寬會變化,因為圖片永遠是矩形,為了放得下一個轉向後的矩形,外接的矩形勢必會比較大
    w, h = rotated_sprite.get_size()
    # 獲得繪製圖片的左上角(感謝pltc325網友的指正)
    sprite_draw_pos = Vector2(sprite_pos.x-w/2, sprite_pos.y-h/2)
    screen.blit(rotated_sprite, sprite_draw_pos)
 
    time_passed = clock.tick()
    time_passed_seconds = time_passed / 1000.0
 
    # 圖片的轉向速度也需要和行進速度一樣,通過時間來控制
    sprite_rotation += rotation_direction * sprite_rotation_speed * time_passed_seconds
 
    # 獲得前進(x方向和y方向),這兩個需要一點點三角的知識
    heading_x = sin(sprite_rotation*pi/180.)
    heading_y = cos(sprite_rotation*pi/180.)
    # 轉換為單位速度向量
    heading = Vector2(heading_x, heading_y)
    # 轉換為速度
    heading *= movement_direction
 
    sprite_pos+= heading * sprite_speed * time_passed_seconds
 
    pygame.display.update()

我們通過上下控制前進/後退,而左右控制轉向。我們通過pygame.transform.rotate()來獲得了轉向後的圖片,具體引數可以參考程式碼。各條語句的作用也可以參考註釋。

下次講解使用滑鼠控制遊戲。


相關文章