在你的 Python 遊戲中模擬引力

學習如何使用 Python 的 Pygame 模塊編程電腦遊戲,並開始操做引力。html

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

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

當你添加一個敵人到你的遊戲中時,你實現了一些碰撞檢測,可是這篇文章要添加更多的東西,由於引力須要碰撞檢測。想一想爲何引力可能涉及碰撞。若是你不能想到任何緣由,不要擔憂 —— 它會隨着你開發示例代碼工做並且顯然。git

在真實世界中的引力是有質量的物體來相互吸引的傾向性。物體(質量)越大,它施加越大的引力做用。在電腦遊戲物理學中,你沒必要建立質量足夠大的物體來證實引力的正確;你能夠在電腦遊戲世界自己中僅編程一個物體落向假設的最大的對象的傾向。程序員

添加一個引力函數

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

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

在真實的世界中,引力影響一切。然而,在平臺遊戲中,引力是有選擇性的 —— 若是你添加引力到你的整個遊戲世界,你的全部平臺都將掉到地上。反之,你能夠僅添加引力到你的玩家和敵人精靈中。bash

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

    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):
    ''' Spawn an enemy '''
    def __init__(self,x,y,img):
        pygame.sprite.Sprite.__init__(self)
        self.image = pygame.image.load(os.path.join('images',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):
        ''' enemy movement '''
        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],'yeti.png') # 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,'ground.png')
                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,'ground.png')
                    plat_list.add(plat)
                    j=j+1
                print('run' + str(i) + str(ploc[i]))
                i=i+1

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

        return plat_list

''' Setup '''
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('images','stage.png')).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 )

''' Main loop '''
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('a'):
                print("LEFT")
                player.control(-steps,0)
            if event.key == pygame.K_RIGHT or event.key == ord('d'):
                print("RIGHT")
                player.control(steps,0)
            if event.key == pygame.K_UP or event.key == ord('w'):
                print('jump')

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

            if event.key == ord('q'):
                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: opensource.com/article/19/…

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

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

相關文章
相關標籤/搜索