Pygame: Difference between revisions

From ICO wiki
Jump to navigationJump to search
Lvosandi (talk | contribs)
Lvosandi (talk | contribs)
No edit summary
 
(5 intermediate revisions by the same user not shown)
Line 1: Line 1:
Pygame is a library for programming games with Python. It handles event loop and simplifies drawing the scene.
Pygame is a library for programming games with Python. It handles event loop and simplifies drawing the scene.


==Installation==
=Installation=


Ubuntu users can simply apt-get the package:
Ubuntu users can simply apt-get the package:
Line 11: Line 11:


   pip3 install pygame-1.9.2a0-cp35-none-win_amd64.whl
   pip3 install pygame-1.9.2a0-cp35-none-win_amd64.whl
=Flappy Bird clones=


==Code example==
==Code example==
Line 36: Line 38:
             elif event.type == KEYDOWN and event.key == K_ESCAPE:
             elif event.type == KEYDOWN and event.key == K_ESCAPE:
                 return
                 return
         screen.blit(background, (50, 50), (0, 0, 200, 300))
         screen.blit(background, (50, 50))
         screen.blit(bird, (100, 100), (0, 0, 17, 18))
         screen.blit(bird, (100, 100))
         pygame.display.flip()
         pygame.display.flip()


Line 54: Line 56:
* Implement panning of background
* Implement panning of background
* Draw the pipes on the screen, use [http://www.pygame.org/docs/ref/transform.html#pygame.transform.flip pygame.transform.flip] to flip the pipe image
* Draw the pipes on the screen, use [http://www.pygame.org/docs/ref/transform.html#pygame.transform.flip pygame.transform.flip] to flip the pipe image


==Improved example==
==Improved example==
Line 121: Line 122:
* Add text overlays and scoring using [http://www.pygame.org/docs/tut/chimp/ChimpLineByLine.html pygame.font]
* Add text overlays and scoring using [http://www.pygame.org/docs/tut/chimp/ChimpLineByLine.html pygame.font]
* Attempt to get fixed FPS
* Attempt to get fixed FPS
==Almost there!==
<source lang="python">
import os, sys
import pygame
from pygame.locals import *
def main_loop(screen):
    pipe = pygame.image.load("img/pipe.png")
    bird = pygame.image.load("img/bird.png")
    background = pygame.image.load("img/bg.png")
    pygame.display.set_caption('Floppy Bird')
    pygame.mouse.set_visible(0)
    print "Pipe dimensions:", pipe.get_size()   
    print "Background dimensions:", background.get_size()
    print "Bird dimensions:", bird.get_size()   
    bird_left = background.get_width() / 3
    bird_top = background.get_height() / 2
    bird_velocity = 0
    level = [0,0,0,0,0,0,0,0,1,2,3,1,2,3,1,2,3]
    flipped_pipe = pygame.transform.flip(pipe, 0, 1)
    frames = 0 # Here we count frames
    scene = pygame.Surface((240,180)) # This is where we compose the scene
    while True:
        scene.fill(0) # This fills the buffer with black pixels
        for event in pygame.event.get():
            if event.type == QUIT:
                return
            elif event.type == KEYDOWN and event.key == K_ESCAPE:
                return
            elif event.type == KEYDOWN and event.key == K_SPACE:
                bird_velocity -= 0.2
                print "Space was pressed!"
        bird_velocity += 0.001      # Gravity!
        bird_top += bird_velocity  # v = v0 + a*t
        frames += 1
       
        pipe_scroll = frames / 2
        background_scroll = frames / 10
       
        for index in range(0, 3):
            scene.blit(background, (index * background.get_width() - background_scroll % pipe.get_height(), 0))
        # This is basically the inverse of calculation of pipe_left below
        current_pipe = int((pipe_scroll + bird_left)/(pipe.get_width() * 3))
        for pipe_index, pipe_height in enumerate(level):
            if pipe_height == 0: continue # Skip pipes whose height is 0
            pipe_left = pipe_index * pipe.get_width() * 3 - pipe_scroll
            pipe_top  = 170 - pipe_height * 15
           
            scene.blit(pipe, (pipe_left, pipe_top))
            scene.blit(flipped_pipe, (pipe_left, pipe_top - 250))
            # Highlight the next pipe we could bump into
            if pipe_index == current_pipe:
                if bird_top > pipe_top:
                    #pygame.draw.rect(scene, (255,0,0), (px, py, 32, 160))
                    print "Game over: You bumped into pipe below!"
                    return
                if bird_top < pipe_top-250:
                    #pygame.draw.rect(scene, (255,0,0), (px, py-250, 32, 160))
                    print "Game over: You bumped into pipe above!"
                    return
        if current_pipe > len(level):
            print "You win! You've passed all the roadblocks!"
            return
        if bird_top + bird.get_height() > scene.get_height():
            print "Game over: Bird fell down!"
            return
        if bird_top < 0:
            print "Game over: You're aiming too high!"
            return
        scene.blit(bird, (bird_left, int(bird_top)), (18 * (int(frames/100) % 3), 0, 18, 13))
        pygame.transform.scale(scene, screen.get_size(), screen) # Here we scale it up and paste to screen
        pygame.display.flip()
def game_over(screen, score):
    # This does nothing yet!
    pass
def __main__():
    pygame.init()
    screen = pygame.display.set_mode((800,600))
    score = main_loop(screen)
    game_over(screen, score)
if __name__ == "__main__":
    __main__()
</source>
==A perhaps easier to follow version==
<source lang="python">
import os, sys
import pygame
from pygame.locals import *
pygame.init()
screen = pygame.display.set_mode((800,600))
pipe = pygame.image.load("img/pipe.png")
bird = pygame.image.load("img/bird.png")
background = pygame.image.load("img/bg.png")
pygame.display.set_caption('Floppy Bird')
pygame.mouse.set_visible(0)
print "Pipe dimensions:", pipe.get_size()   
print "Background dimensions:", background.get_size()
print "Bird dimensions:", bird.get_size()   
def game_loop():
    pipe_gap = 70
    bird_left = background.get_width() / 3
    bird_top = background.get_height() / 2
    bird_velocity = 0
    level = [0,0,0,0,0,0,0,0,1,2,3,1,2,3,1,2,3]
    flipped_pipe = pygame.transform.flip(pipe, 0, 1)
    frames = 0 # Here we count frames
    scene = pygame.Surface((240,180)) # This is where we compose the scene
    while True:
        scene.fill(0) # This fills the buffer with black pixels
        for event in pygame.event.get():
            if event.type == QUIT:
                return
            elif event.type == KEYDOWN and event.key == K_ESCAPE:
                return
            elif event.type == KEYDOWN and event.key == K_SPACE:
                bird_velocity -= 0.2
        bird_velocity += 0.001      # Gravity!
        bird_top += bird_velocity  # v = v0 + a*t
        frames += 1
       
        pipe_scroll = frames / 2
        background_scroll = frames / 10
       
        for index in range(0, 3):
            scene.blit(background, (index * background.get_width() - background_scroll % pipe.get_height(), 0))
        # This is basically the inverse of calculation of pipe_left below
        current_pipe = int((pipe_scroll + bird_left)/(pipe.get_width() * 3))
        for pipe_index, pipe_height in enumerate(level):
            if pipe_height == 0: continue # Skip pipes whose height is 0
            pipe_left = pipe_index * pipe.get_width() * 3 - pipe_scroll
            pipe_top  = 170 - pipe_height * 15
           
            scene.blit(pipe, (pipe_left, pipe_top))
            scene.blit(flipped_pipe, (pipe_left, pipe_top - pipe.get_height() - pipe_gap))
            # Highlight the next pipe we could bump into
            if pipe_index == current_pipe:
                if bird_top > pipe_top:
                    #pygame.draw.rect(scene, (255,0,0), (pipe_left, pipe_top, pipe.get_width(), pipe.get_height()))
                    print "Game over: You bumped into pipe below!"
                    return
                if bird_top < pipe_top - pipe_gap:
                    #pygame.draw.rect(scene, (255,0,0), (pipe_left, pipe_top  - pipe.get_height() - pipe_gap, pipe.get_width(), pipe.get_height()))
                    print "Game over: You bumped into pipe above!"
                    return
        if current_pipe > len(level):
            print "You win! You've passed all the roadblocks!"
            return
        if bird_top + bird.get_height() > scene.get_height():
            print "Game over: Bird fell down!"
            return
        if bird_top < 0:
            print "Game over: You're aiming too high!"
            return
        scene.blit(bird, (bird_left, int(bird_top)), (18 * (int(frames/100) % 3), 0, 18, 13))
        pygame.transform.scale(scene, screen.get_size(), screen) # Here we scale it up and paste to screen
        pygame.display.flip()
def game_over():
    font = pygame.font.SysFont("sans", 72)
    text = font.render("Game over!", True, (255, 0, 0))
    position = screen.get_width() / 2 - text.get_width() / 2, screen.get_height() / 2 - text.get_height() / 2
    screen.blit(text, position)
    while True:
        for event in pygame.event.get():
            if event.type == QUIT:
                return False
            elif event.type == KEYDOWN and event.key == K_ESCAPE:
                return False
            elif event.type == KEYDOWN and event.key == K_SPACE:
                return True
        pygame.display.flip()
again = True
while again:
    game_loop()
    again = game_over()
</source>

Latest revision as of 18:10, 24 March 2016

Pygame is a library for programming games with Python. It handles event loop and simplifies drawing the scene.

Installation

Ubuntu users can simply apt-get the package:

 apt-get install python-pygame

Windows users should download pygame .whl file corresponding to their Python runtime version. On Python 3.5 use following to install the module:

 pip3 install pygame-1.9.2a0-cp35-none-win_amd64.whl

Flappy Bird clones

Code example

An example pygame code:

import os, sys
import pygame
from pygame.locals import *

def __main__():
    pygame.init()
    pipe = pygame.image.load("img/pipe.png")
    bird = pygame.image.load("img/bird.png")
    background = pygame.image.load("img/bg.png")
    screen = pygame.display.set_mode((640,480))
    pygame.display.set_caption('Floppy Bird')
    pygame.mouse.set_visible(0)

    while True:
        for event in pygame.event.get():
            if event.type == QUIT:
                return
            elif event.type == KEYDOWN and event.key == K_ESCAPE:
                return
        screen.blit(background, (50, 50))
        screen.blit(bird, (100, 100))
        pygame.display.flip()

if __name__ == "__main__":
    __main__()

Download graphics:

wget http://enos.itcollege.ee/~lvosandi/floppy-gfx.zip
unzip floppy-gfx.zip


Execrises:

  • Implement panning of background
  • Draw the pipes on the screen, use pygame.transform.flip to flip the pipe image

Improved example

import os, sys
import pygame
from pygame.locals import *
 
def __main__():
    pygame.init()
    pipe = pygame.image.load("img/pipe.png")
    bird = pygame.image.load("img/bird.png")
    background = pygame.image.load("img/bg.png")
 
 
    screen = pygame.display.set_mode((800,600))
    pygame.display.set_caption('Floppy Bird')
    pygame.mouse.set_visible(0)
 
    print "Pipe dimensions:", pipe.get_size()    
    print "Background dimensions:", background.get_size()
 
    y = 0
    vy = 0
 
    level = [0, 1, 0, 0, 2, 0, 0, 3,0, 1, 0, 0, 2, 0, 0, 3,0, 1, 0, 0, 2, 0, 0, 3,]
 
    antipipe = pygame.transform.flip(pipe, 0, 1)
    frames = 0 # Here we count frames
    scene = pygame.Surface((240,180)) # This is where we compose the scene
    while True:
        scene.fill(0) # This fills the buffer with black pixels
        for event in pygame.event.get():
            if event.type == QUIT:
                return
            elif event.type == KEYDOWN and event.key == K_ESCAPE:
                return
            elif event.type == KEYDOWN and event.key == K_SPACE:
                vy -= 0.02
                print "Space was pressed!"

        vy += 0.0001
        y += vy
        for i in range(0, 3):
            scene.blit(background, (i*148 - (frames/10) % 148, 0), (0, 0, 200, 300))
        frames += 1
 
        for offset, pipe_height in enumerate(level):
            if pipe_height == 0: continue # Skip pipes whose height is 0
            scene.blit(pipe, (400-frames/2 + offset * 32, 180 - pipe_height * 32), (0, 0, 100,160))
            scene.blit(antipipe, (400-frames/2 + offset * 32, 180 - pipe_height * 32 - 250), (0, 0, 100,160))
 
        scene.blit(bird, (100, 100+int(y)), (18 * (int(frames/100) % 3), 0, 18, 13))
        pygame.transform.scale(scene, screen.get_size(), screen) # Here we scale it up and paste to screen
 
        pygame.display.flip()
 
if __name__ == "__main__":
    __main__()

Exercises:

  • Add collision detection
  • Add text overlays and scoring using pygame.font
  • Attempt to get fixed FPS

Almost there!

import os, sys
import pygame
from pygame.locals import *
 
 
def main_loop(screen):
    pipe = pygame.image.load("img/pipe.png")
    bird = pygame.image.load("img/bird.png")
    background = pygame.image.load("img/bg.png")
 
    pygame.display.set_caption('Floppy Bird')
    pygame.mouse.set_visible(0)
 
    print "Pipe dimensions:", pipe.get_size()    
    print "Background dimensions:", background.get_size()
    print "Bird dimensions:", bird.get_size()    
 
    bird_left = background.get_width() / 3
    bird_top = background.get_height() / 2
    bird_velocity = 0
    level = [0,0,0,0,0,0,0,0,1,2,3,1,2,3,1,2,3]
 
    flipped_pipe = pygame.transform.flip(pipe, 0, 1)
    frames = 0 # Here we count frames
    scene = pygame.Surface((240,180)) # This is where we compose the scene
    while True:
        scene.fill(0) # This fills the buffer with black pixels
        for event in pygame.event.get():
            if event.type == QUIT:
                return
            elif event.type == KEYDOWN and event.key == K_ESCAPE:
                return
            elif event.type == KEYDOWN and event.key == K_SPACE:
                bird_velocity -= 0.2
                print "Space was pressed!"
 
        bird_velocity += 0.001      # Gravity!
        bird_top += bird_velocity   # v = v0 + a*t
 
        frames += 1
        
        pipe_scroll = frames / 2
        background_scroll = frames / 10
        
        for index in range(0, 3):
            scene.blit(background, (index * background.get_width() - background_scroll % pipe.get_height(), 0))
 
        # This is basically the inverse of calculation of pipe_left below
        current_pipe = int((pipe_scroll + bird_left)/(pipe.get_width() * 3))
 
        for pipe_index, pipe_height in enumerate(level):
            if pipe_height == 0: continue # Skip pipes whose height is 0
            pipe_left = pipe_index * pipe.get_width() * 3 - pipe_scroll
            pipe_top  = 170 - pipe_height * 15
            
            scene.blit(pipe, (pipe_left, pipe_top))
            scene.blit(flipped_pipe, (pipe_left, pipe_top - 250))
 
            # Highlight the next pipe we could bump into
            if pipe_index == current_pipe:
 
                if bird_top > pipe_top:
                    #pygame.draw.rect(scene, (255,0,0), (px, py, 32, 160))
                    print "Game over: You bumped into pipe below!"
                    return
                if bird_top < pipe_top-250:
                    #pygame.draw.rect(scene, (255,0,0), (px, py-250, 32, 160))
                    print "Game over: You bumped into pipe above!"
                    return
 
        if current_pipe > len(level):
            print "You win! You've passed all the roadblocks!"
            return
 
        if bird_top + bird.get_height() > scene.get_height():
            print "Game over: Bird fell down!"
            return
        if bird_top < 0:
            print "Game over: You're aiming too high!"
            return
 
        scene.blit(bird, (bird_left, int(bird_top)), (18 * (int(frames/100) % 3), 0, 18, 13))
        pygame.transform.scale(scene, screen.get_size(), screen) # Here we scale it up and paste to screen
 
        pygame.display.flip()
 
 
def game_over(screen, score):
    # This does nothing yet!
    pass
 
def __main__():
    pygame.init()
    screen = pygame.display.set_mode((800,600))
    score = main_loop(screen)
    game_over(screen, score)
 
if __name__ == "__main__":
    __main__()


A perhaps easier to follow version

import os, sys
import pygame
from pygame.locals import *

pygame.init()
screen = pygame.display.set_mode((800,600))

pipe = pygame.image.load("img/pipe.png")
bird = pygame.image.load("img/bird.png")
background = pygame.image.load("img/bg.png")

pygame.display.set_caption('Floppy Bird')
pygame.mouse.set_visible(0)

print "Pipe dimensions:", pipe.get_size()    
print "Background dimensions:", background.get_size()
print "Bird dimensions:", bird.get_size()    
 
def game_loop():
    pipe_gap = 70
    bird_left = background.get_width() / 3
    bird_top = background.get_height() / 2
    bird_velocity = 0
    level = [0,0,0,0,0,0,0,0,1,2,3,1,2,3,1,2,3]
 
    flipped_pipe = pygame.transform.flip(pipe, 0, 1)
    frames = 0 # Here we count frames
    scene = pygame.Surface((240,180)) # This is where we compose the scene
    while True:
        scene.fill(0) # This fills the buffer with black pixels
        for event in pygame.event.get():
            if event.type == QUIT:
                return
            elif event.type == KEYDOWN and event.key == K_ESCAPE:
                return
            elif event.type == KEYDOWN and event.key == K_SPACE:
                bird_velocity -= 0.2
 
        bird_velocity += 0.001      # Gravity!
        bird_top += bird_velocity   # v = v0 + a*t
 
        frames += 1
        
        pipe_scroll = frames / 2
        background_scroll = frames / 10
        
        for index in range(0, 3):
            scene.blit(background, (index * background.get_width() - background_scroll % pipe.get_height(), 0))
 
        # This is basically the inverse of calculation of pipe_left below
        current_pipe = int((pipe_scroll + bird_left)/(pipe.get_width() * 3))
 
        for pipe_index, pipe_height in enumerate(level):
            if pipe_height == 0: continue # Skip pipes whose height is 0
            pipe_left = pipe_index * pipe.get_width() * 3 - pipe_scroll
            pipe_top  = 170 - pipe_height * 15
            
            scene.blit(pipe, (pipe_left, pipe_top))
            scene.blit(flipped_pipe, (pipe_left, pipe_top - pipe.get_height() - pipe_gap))
 
            # Highlight the next pipe we could bump into
            if pipe_index == current_pipe:
 
                if bird_top > pipe_top:
                    #pygame.draw.rect(scene, (255,0,0), (pipe_left, pipe_top, pipe.get_width(), pipe.get_height()))
                    print "Game over: You bumped into pipe below!"
                    return
                if bird_top < pipe_top - pipe_gap:
                    #pygame.draw.rect(scene, (255,0,0), (pipe_left, pipe_top  - pipe.get_height() - pipe_gap, pipe.get_width(), pipe.get_height()))
                    print "Game over: You bumped into pipe above!"
                    return
 
        if current_pipe > len(level):
            print "You win! You've passed all the roadblocks!"
            return
 
        if bird_top + bird.get_height() > scene.get_height():
            print "Game over: Bird fell down!"
            return
        if bird_top < 0:
            print "Game over: You're aiming too high!"
            return
 
        scene.blit(bird, (bird_left, int(bird_top)), (18 * (int(frames/100) % 3), 0, 18, 13))
        pygame.transform.scale(scene, screen.get_size(), screen) # Here we scale it up and paste to screen

        pygame.display.flip()
 
 
def game_over():
    font = pygame.font.SysFont("sans", 72)
    text = font.render("Game over!", True, (255, 0, 0))
    position = screen.get_width() / 2 - text.get_width() / 2, screen.get_height() / 2 - text.get_height() / 2
    screen.blit(text, position)
    while True:
        for event in pygame.event.get():
            if event.type == QUIT:
                return False
            elif event.type == KEYDOWN and event.key == K_ESCAPE:
                return False
            elif event.type == KEYDOWN and event.key == K_SPACE:
                return True
        pygame.display.flip()


again = True
while again:
    game_loop()
    again = game_over()