Linux中國

在你的 Python 遊戲中模擬引力

真實的世界充滿了運動和生活。物理學使得真實的生活如此忙碌和動態。物理學是物質在空間中運動的方式。既然一個電腦遊戲世界沒有物質,它也就沒有物理學規律,使用遊戲程序員不得不模擬物理學。

從大多數電腦遊戲來說,這裡基本上僅有兩個方面的物理學是重要的:引力和碰撞。

當你添加一個敵人到你的遊戲中時,你實現了一些碰撞檢測,但是這篇文章要添加更多的東西,因為引力需要碰撞檢測。想想為什麼引力可能涉及碰撞。如果你不能想到任何原因,不要擔心 —— 它會隨著你開發示例代碼工作而且顯然。

在真實世界中的引力是有質量的物體來相互吸引的傾向性。物體(質量)越大,它施加越大的引力作用。在電腦遊戲物理學中,你不必創建質量足夠大的物體來證明引力的正確;你可以在電腦遊戲世界本身中僅編程一個物體落向假設的最大的對象的傾向。

添加一個引力函數

記住你的玩家已經有了一個決定動作的屬性。使用這個屬性來將玩家精靈拉向屏幕底部。

Pygame 中,較高的數字更接近屏幕的底部邊緣。

在真實的世界中,引力影響一切。然而,在平台遊戲中,引力是有選擇性的 —— 如果你添加引力到你的整個遊戲世界,你的所有平台都將掉到地上。反之,你可以僅添加引力到你的玩家和敵人精靈中。

首先,在你的 Player 類中添加一個 gravity 函數:

    def gravity(self):
        self.movey += 3.2 # 玩家掉落的多快

這是一個簡單的函數。首先,不管你的玩家是否想運動,你設置你的玩家垂直運動。也就是說,你已經編程你的玩家總是在下降。這基本上就是引力。

為使引力函數生效,你必須在你的主循環中調用它。這樣,當每一個處理循環時,Python 都應用下落運動到你的玩家。

在這代碼中,添加第一行到你的循環中:

    player.gravity() # 檢查引力
    player.update()

啟動你的遊戲來看看會發生什麼。要注意,因為它發生的很快:你是玩家從天空上下落,馬上掉出了你的遊戲屏幕。

你的引力模擬是工作的,但是,也許太好了。

作為一次試驗,嘗試更改你玩家下落的速度。

給引力添加一個地板

你的遊戲沒有辦法發現你的角色掉落出世界的問題。在一些遊戲中,如果一個玩家掉落出世界,該精靈被刪除,並在某個新的位置重生。在另一些遊戲中,玩家會丟失分數或一條生命。當一個玩家掉落出世界時,不管你想發生什麼,你必須能夠偵測出玩家何時消失在屏幕外。

在 Python 中,要檢查一個條件,你可以使用一個 if 語句。

你必需查看你玩家是否正在掉落,以及你的玩家掉落的程度。如果你的玩家掉落到屏幕的底部,那麼你可以做一些事情。簡化一下,設置玩家精靈的位置為底部邊緣上方 20 像素。

使你的 gravity 函數看起來像這樣:

    def gravity(self):
        self.movey += 3.2 # 玩家掉落的多快

        if self.rect.y > worldy and self.movey >= 0:
            self.movey = 0
            self.rect.y = worldy-ty

然後,啟動你的遊戲。你的精靈仍然下落,但是它停在屏幕的底部。不過,你也許不能看到你在地面層之上的精靈。一個簡單的解決方法是,在精靈碰撞遊戲世界的底部後,通過添加另一個 -ty 到它的新 Y 位置,從而使你的精靈彈跳到更高處:

    def gravity(self):
        self.movey += 3.2 # 玩家掉落的多快

        if self.rect.y > worldy and self.movey >= 0:
            self.movey = 0
            self.rect.y = worldy-ty-ty

現在你的玩家在屏幕底部彈跳,恰好在你地面精靈上面。

你的玩家真正需要的是反抗引力的方法。引力問題是,你不能反抗它,除非你有一些東西來推開引力作用。因此,在接下來的文章中,你將添加地面和平台碰撞以及跳躍能力。在這期間,嘗試應用引力到敵人精靈。

到目前為止,這裡是全部的代碼:

#!/usr/bin/env python3
# draw a world
# add a player and player control
# add player movement
# add enemy and basic collision
# add platform
# add gravity

# GNU All-Permissive License
# Copying and distribution of this file, with or without modification,
# are permitted in any medium without royalty provided the copyright
# notice and this notice are preserved.  This file is offered as-is,
# without any warranty.

import pygame
import sys
import os

'''
Objects
'''

class Platform(pygame.sprite.Sprite):
    # x location, y location, img width, img height, img file    
    def __init__(self,xloc,yloc,imgw,imgh,img):
        pygame.sprite.Sprite.__init__(self)
        self.image = pygame.image.load(os.path.join('images',img)).convert()
        self.image.convert_alpha()
        self.rect = self.image.get_rect()
        self.rect.y = yloc
        self.rect.x = xloc

class Player(pygame.sprite.Sprite):
    '''
    Spawn a player
    '''
    def __init__(self):
        pygame.sprite.Sprite.__init__(self)
        self.movex = 0
        self.movey = 0
        self.frame = 0
        self.health = 10
        self.score = 1
        self.images = []
        for i in range(1,9):
            img = pygame.image.load(os.path.join('images','hero' + str(i) + '.png')).convert()
            img.convert_alpha()
            img.set_colorkey(ALPHA)
            self.images.append(img)
            self.image = self.images[0]
            self.rect  = self.image.get_rect()

    def gravity(self):
        self.movey += 3.2 # how fast player falls

        if self.rect.y > worldy and self.movey >= 0:
            self.movey = 0
            self.rect.y = worldy-ty-ty

    def control(self,x,y):
        '''
        control player movement
        '''
        self.movex += x
        self.movey += y

    def update(self):
        '''
        Update sprite position
        '''

        self.rect.x = self.rect.x + self.movex
        self.rect.y = self.rect.y + self.movey

        # moving left
        if self.movex < 0:
            self.frame += 1
            if self.frame > ani*3:
                self.frame = 0
            self.image = self.images[self.frame//ani]

        # moving right
        if self.movex > 0:
            self.frame += 1
            if self.frame > ani*3:
                self.frame = 0
            self.image = self.images[(self.frame//ani)+4]

        # collisions
        enemy_hit_list = pygame.sprite.spritecollide(self, enemy_list, False)
        for enemy in enemy_hit_list:
            self.health -= 1
            print(self.health)

        ground_hit_list = pygame.sprite.spritecollide(self, ground_list, False)
        for g in ground_hit_list:
            self.health -= 1
            print(self.health)

class Enemy(pygame.sprite.Sprite):
    &apos;&apos;&apos;
    Spawn an enemy
    &apos;&apos;&apos;
    def __init__(self,x,y,img):
        pygame.sprite.Sprite.__init__(self)
        self.image = pygame.image.load(os.path.join(&apos;images&apos;,img))
        #self.image.convert_alpha()
        #self.image.set_colorkey(ALPHA)
        self.rect = self.image.get_rect()
        self.rect.x = x
        self.rect.y = y
        self.counter = 0

    def move(self):
        &apos;&apos;&apos;
        enemy movement
        &apos;&apos;&apos;
        distance = 80
        speed = 8

        if self.counter >= 0 and self.counter <= distance:
            self.rect.x += speed
        elif self.counter >= distance and self.counter <= distance*2:
            self.rect.x -= speed
        else:
            self.counter = 0

        self.counter += 1

class Level():
    def bad(lvl,eloc):
        if lvl == 1:
            enemy = Enemy(eloc[0],eloc[1],&apos;yeti.png&apos;) # spawn enemy
            enemy_list = pygame.sprite.Group() # create enemy group
            enemy_list.add(enemy)              # add enemy to group

        if lvl == 2:
            print("Level " + str(lvl) )

        return enemy_list

    def loot(lvl,lloc):
        print(lvl)

    def ground(lvl,gloc,tx,ty):
        ground_list = pygame.sprite.Group()
        i=0
        if lvl == 1:
            while i < len(gloc):
                ground = Platform(gloc[i],worldy-ty,tx,ty,&apos;ground.png&apos;)
                ground_list.add(ground)
                i=i+1

        if lvl == 2:
            print("Level " + str(lvl) )

        return ground_list

    def platform(lvl,tx,ty):
        plat_list = pygame.sprite.Group()
        ploc = []
        i=0
        if lvl == 1:
            ploc.append((0,worldy-ty-128,3))
            ploc.append((300,worldy-ty-256,3))
            ploc.append((500,worldy-ty-128,4))

            while i < len(ploc):
                j=0
                while j <= ploc[i][2]:
                    plat = Platform((ploc[i][0]+(j*tx)),ploc[i][1],tx,ty,&apos;ground.png&apos;)
                    plat_list.add(plat)
                    j=j+1
                print(&apos;run&apos; + str(i) + str(ploc[i]))
                i=i+1

        if lvl == 2:
            print("Level " + str(lvl) )

        return plat_list

&apos;&apos;&apos;
Setup
&apos;&apos;&apos;
worldx = 960
worldy = 720

fps = 40 # frame rate
ani = 4  # animation cycles
clock = pygame.time.Clock()
pygame.init()
main = True

BLUE  = (25,25,200)
BLACK = (23,23,23 )
WHITE = (254,254,254)
ALPHA = (0,255,0)

world = pygame.display.set_mode([worldx,worldy])
backdrop = pygame.image.load(os.path.join(&apos;images&apos;,&apos;stage.png&apos;)).convert()
backdropbox = world.get_rect()
player = Player() # spawn player
player.rect.x = 0
player.rect.y = 0
player_list = pygame.sprite.Group()
player_list.add(player)
steps = 10 # how fast to move

eloc = []
eloc = [200,20]
gloc = []
#gloc = [0,630,64,630,128,630,192,630,256,630,320,630,384,630]
tx = 64 #tile size
ty = 64 #tile size

i=0
while i <= (worldx/tx)+tx:
    gloc.append(i*tx)
    i=i+1

enemy_list = Level.bad( 1, eloc )
ground_list = Level.ground( 1,gloc,tx,ty )
plat_list = Level.platform( 1,tx,ty )

&apos;&apos;&apos;
Main loop
&apos;&apos;&apos;
while main == True:
    for event in pygame.event.get():
        if event.type == pygame.QUIT:
            pygame.quit(); sys.exit()
            main = False

        if event.type == pygame.KEYDOWN:
            if event.key == pygame.K_LEFT or event.key == ord(&apos;a&apos;):
                print("LEFT")
                player.control(-steps,0)
            if event.key == pygame.K_RIGHT or event.key == ord(&apos;d&apos;):
                print("RIGHT")
                player.control(steps,0)
            if event.key == pygame.K_UP or event.key == ord(&apos;w&apos;):
                print(&apos;jump&apos;)

        if event.type == pygame.KEYUP:
            if event.key == pygame.K_LEFT or event.key == ord(&apos;a&apos;):
                player.control(steps,0)
            if event.key == pygame.K_RIGHT or event.key == ord(&apos;d&apos;):
                player.control(-steps,0)
            if event.key == pygame.K_UP or event.key == ord(&apos;w&apos;):
                print(&apos;jump&apos;)

            if event.key == ord(&apos;q&apos;):
                pygame.quit()
                sys.exit()
                main = False

    world.blit(backdrop, backdropbox)
    player.gravity() # check gravity
    player.update()
    player_list.draw(world)
    enemy_list.draw(world)
    ground_list.draw(world)
    plat_list.draw(world)
    for e in enemy_list:
        e.move()
    pygame.display.flip()
    clock.tick(fps)

這是仍在進行中的關於使用 Pygame 模塊來在 Python 3 在創建電腦遊戲的第七部分。先前的文章是:

via: https://opensource.com/article/19/11/simulate-gravity-python

作者:Seth Kenlon 選題:lujun9972 譯者:robsean 校對:wxy

本文由 LCTT 原創編譯,Linux中國 榮譽推出


本文轉載來自 Linux 中國: https://github.com/Linux-CN/archive

對這篇文章感覺如何?

太棒了
0
不錯
0
愛死了
0
不太好
0
感覺很糟
0
雨落清風。心向陽

    You may also like

    Leave a reply

    您的電子郵箱地址不會被公開。 必填項已用 * 標註

    此站點使用Akismet來減少垃圾評論。了解我們如何處理您的評論數據

    More in:Linux中國