I have the beginnings of a game developed in pygame with python3, following tutorials on www.teachyourselfpython.com There is a main.py, player.py and walls.py (with player and walls class respectively). The player class contains code for collision detection and movement. Unfortunately, the player does avoid the tile, but on collision, moves to a position on the right hand side of the screen, instead of expected movement (just stopping). Is anyone able to help with the logic of this and correcting the erroneous/undesirable movement on collision. Below, are the three files: main.py, player.py and walls.py. MAIN.PY
#main.py import pygame import random from player import Player from collectable import Collectable from walls import Wall pygame.init() BLACK=(0,0,0) WHITE=(255,255,255) RED=(255,0,0) GREEN =(0,255,0) BLUE=(0,0,255) GOLD=(255,215,0) WIDTH=500 HEIGHT=500 size= (WIDTH,HEIGHT) screen=pygame.display.set_mode(size) pygame.display.set_caption("The Life Game") done = False clock=pygame.time.Clock() wall_list=pygame.sprite.Group() all_sprites = pygame.sprite.Group() enemy_list = pygame.sprite.Group() player=Player() player.walls=wall_list all_sprites.add(player) for i in range(random.randrange(100,200)): whiteStar = Collectable(WHITE, 3, 3, "White Star", "Rect") whiteStar.rect.x = random.randrange(size[0]) whiteStar.rect.y = random.randrange(size[1]) all_sprites.add(whiteStar) for i in range(50): enemy = Collectable(RED,6, 6,"Enemy","Ellipse") enemy.rect.x = random.randrange(300) enemy.rect.y = random.randrange(300) enemy_list.add(enemy) all_sprites.add(enemy) coin1 = Collectable(GOLD,50,50,"Coin","Ellipse") coin1.rect.x=440 coin1.rect.y=0 all_sprites.add(coin1) coin2 = Collectable(GOLD,50,50,"Coin","Ellipse") coin2.rect.x=0 coin2.rect.y=440 all_sprites.add(coin2) enemy = Collectable(RED,100,100,"Enemy","Ellipse") enemy.rect.x=70 enemy.rect.y=230 all_sprites.add(enemy) #Make the walls (x_pos,y_pos, width, height,colour) wall=Wall(0,0,10,600,GREEN) wall_list.add(wall) all_sprites.add(wall_list) wall = Wall(50, 300, 400, 10,RED) wall_list.add(wall) all_sprites.add(wall_list) wall = Wall(10, 200, 100, 10,BLUE) wall_list.add(wall) all_sprites.add(wall_list) score=0 health=100 #- - - - - - - - - - - - - -Main Program Loop - - - - - - - - - - - - - - - - def main(): done=False score=0 health=100 while not done: #- - - - - - Main event loop (this is where code for handling keyboard and mouse clicks will go) #Loop until the user clicks the 'x' button (to close program) for event in pygame.event.get(): #User does something if event.type == pygame.QUIT: #If the user clicked close done = True #set the done flag to 'true' to exit the loop keys = pygame.key.get_pressed() #checking pressed keys if keys[pygame.K_LEFT]: player.moveLeft(5) if keys[pygame.K_RIGHT]: player.moveRight(5) if keys[pygame.K_UP]: player.moveUp(5) if keys[pygame.K_DOWN]: player.moveDown(5) #>>----------DRAW SECTION ----------------------------------- #Clear the screen to BLACK. Any drawing commands should be put BELOW this or they will be reased with this command screen.fill(BLACK) #Select the font to be used (size, bold, italics, etc) font_score = pygame.font.SysFont('Calibri',20,True,False) font_health = pygame.font.SysFont('Calibri',20,True,False) #Printing a variable (score or health) to the screen involves converting the score (if integer) to a string first.score_label = font_score.render("Score: " + str(score),True,BLACK) health_label = font_health.render("Health: "+str(health),True,WHITE) score_label = font_score.render("Score: " + str(score),True, WHITE) #Now we can use this line of code to put the image of the text on the screen at a given position screen.blit(score_label,[100,480]) screen.blit(health_label,[190,480]) #>>---------UPDATE SECTION / Put the logic of your game here (i.e. how objects move, when to fire them, etc) all_sprites.update() if coin1.collision_with(player): score=score+1 coin1.kill() coin1.rect.x=-20 coin1.rect.y=-330 if coin2.collision_with(player): score=score+1 coin2.kill() coin2.rect.x=-20 coin2.rect.y=-330 if enemy.collision_with(player): health=health-25 enemy.kill() enemy.rect.x=-20 enemy.rect.y=-330 enemy.update() #-------------PRINTING VARIABLES LIKE SCORE TO SCREEN #Any drawing/graphics code should go here all_sprites.draw(screen) #Update the screen to show whatever you have drawn pygame.display.flip() #Set the frames per second (e.g. 30, 60 etc) clock.tick(120) main()
PLAYER.PY
import pygame import random from walls import Wall class Player(pygame.sprite.Sprite): #-------------------Define Variables here speed=0 #------------------Initialise Constructor def __init__(self): pygame.sprite.Sprite.__init__(self) self.image=pygame.image.load("player.png") self.rect = self.image.get_rect() #SET THE INITIAL SPEED TO ZERO self.change_x = 0 self.change_y = 0 #--------------Fetch the rectangle object that has the dimensions of the image self.rect =self.image.get_rect() #---------------Define movement def moveRight(self,pixels): self.rect.x+=pixels def moveLeft(self,pixels): self.rect.x-=pixels def moveUp(self,pixels): self.rect.y-=pixels def moveDown(self,pixels): self.rect.y+=pixels # Make our top-left corner the passed-in location. def settopleft(): self.rect = self.image.get_rect() self.rect.y = y self.rect.x = x # Set speed vector self.change_x = 0 self.change_y = 0 self.walls = None def changespeed(self, x, y): """ Change the speed of the player. """ self.change_x += x self.change_y += y def update(self): # Did this update cause us to hit a wall? block_hit_list = pygame.sprite.spritecollide(self, self.walls, False) for block in block_hit_list: # If we are moving right, set our right side to the left side of # the item we hit if self.change_x > 0: self.rect.right = block.rect.left else: # Otherwise if we are moving left, do the opposite. self.rect.left = block.rect.right # Move up/down self.rect.y += self.change_y # Check and see if we hit anything block_hit_list = pygame.sprite.spritecollide(self, self.walls, False) for block in block_hit_list: # Reset our position based on the top/bottom of the object. if self.change_y > 0: self.rect.top = block.rect.top else: self.rect.top = block.rect.bottom
WALLS.PY
import pygame class Wall(pygame.sprite.Sprite): #Wall a player can run into def __init__(self, x, y, width, height,colour): #Constructor fo rthe wall that the player can run into #call the parent's constructor super().__init__() #Make a green wall, of the size specified in paramenters self.image=pygame.Surface([width,height]) self.image.fill(colour) #Make the "passed-in" location ,the top left corner self.rect=self.image.get_rect() self.rect.y=y self.rect.x=x
Attached, area also images for player and bg:
Answer
Okay, let’s talk about the movement problem first. The code in the Player
s update method should move the player along the x-axis first, check if he collides with a wall and if he collides, set his position to the block edge. Afterwards you do the same with the y-axis. It has to be done in this way, because we wouldn’t know the direction of the player otherwise and couldn’t reset his position to the correct edge. You seem to have changed the code so that the self.change_x
and change_y
attributes aren’t used anymore and move the player with pygame.key.get_pressed
instead. I’d delete the pygame.key.get_pressed
block and do the movement changes in the event loop. A few things in the update
method had to be fixed as well, e.g.:
if self.change_y > 0: self.rect.bottom = block.rect.top
For the collision detection, you can use pygame.sprite.spritecollide
and pass the player and the sprite group that you want to check. Then iterate over the returned list and do something for every collided sprite.
Here’s your updated code:
import sys import random import pygame as pg pg.init() class Wall(pg.sprite.Sprite): """Wall a player can run into.""" def __init__(self, x, y, width, height, colour): super().__init__() self.image = pg.Surface([width, height]) self.image.fill(colour) # Make the "passed-in" location the top left corner. self.rect = self.image.get_rect(topleft=(x, y)) class Collectable(pg.sprite.Sprite): """A collectable item.""" def __init__(self, colour, x, y, image, rect): super().__init__() self.image = pg.Surface((5, 5)) self.image.fill(colour) self.rect = self.image.get_rect(topleft=(x, y)) class Player(pg.sprite.Sprite): def __init__(self): pg.sprite.Sprite.__init__(self) self.image = pg.Surface((30, 30)) self.image.fill((50, 150, 250)) self.rect = self.image.get_rect() #SET THE INITIAL SPEED TO ZERO self.change_x = 0 self.change_y = 0 self.health = 100 def update(self): # Move left/right. self.rect.x += self.change_x # Did this update cause us to hit a wall? block_hit_list = pg.sprite.spritecollide(self, self.walls, False) for block in block_hit_list: # If we are moving right, set our right side to the left side of # the item we hit if self.change_x > 0: self.rect.right = block.rect.left else: # Otherwise if we are moving left, do the opposite. self.rect.left = block.rect.right # Move up/down. self.rect.y += self.change_y # Check and see if we hit anything. block_hit_list = pg.sprite.spritecollide(self, self.walls, False) for block in block_hit_list: # Reset our position based on the top/bottom of the object. if self.change_y > 0: self.rect.bottom = block.rect.top else: self.rect.top = block.rect.bottom BLACK = (0,0,0) WHITE = (255,255,255) GREEN = (0,255,0) RED = (255,0,0) BLUE = (0,0,255) GOLD = (255,215,0) size = (500, 500) screen = pg.display.set_mode(size) pg.display.set_caption("The Life Game") wall_list = pg.sprite.Group() all_sprites = pg.sprite.Group() enemy_list = pg.sprite.Group() coins = pg.sprite.Group() player = Player() player.walls = wall_list all_sprites.add(player) for i in range(random.randrange(100,200)): x = random.randrange(size[0]) y = random.randrange(size[1]) whiteStar = Collectable(WHITE, x, y, "White Star", "Rect") all_sprites.add(whiteStar) for i in range(50): x = random.randrange(size[0]) y = random.randrange(size[1]) enemy = Collectable(RED, x, y, "Enemy","Ellipse") enemy_list.add(enemy) all_sprites.add(enemy) coin1 = Collectable(GOLD,240,200,"Coin","Ellipse") coin2 = Collectable(GOLD,100,340,"Coin","Ellipse") all_sprites.add(coin1, coin2) coins.add(coin1, coin2) # Make the walls. walls = [Wall(0,0,10,600,GREEN), Wall(50, 300, 400, 10,RED), Wall(10, 200, 100, 10,BLUE)] wall_list.add(walls) all_sprites.add(walls) def main(): clock = pg.time.Clock() done = False score = 0 font_score = pg.font.SysFont('Calibri',20,True,False) font_health = pg.font.SysFont('Calibri',20,True,False) while not done: for event in pg.event.get(): if event.type == pg.QUIT: done = True # Do the movement in the event loop by setting # the player's change_x and y attributes. elif event.type == pg.KEYDOWN: if event.key == pg.K_LEFT: player.change_x = -3 elif event.key == pg.K_RIGHT: player.change_x = 3 elif event.key == pg.K_UP: player.change_y = -3 elif event.key == pg.K_DOWN: player.change_y = 3 elif event.type == pg.KEYUP: if event.key == pg.K_LEFT and player.change_x < 0: player.change_x = 0 elif event.key == pg.K_RIGHT and player.change_x > 0: player.change_x = 0 elif event.key == pg.K_UP and player.change_y < 0: player.change_y = 0 elif event.key == pg.K_DOWN and player.change_y > 0: player.change_y = 0 # UPDATE SECTION / Put the logic of your game here (i.e. how # objects move, when to fire them, etc). all_sprites.update() # spritecollide returns a list of the collided sprites in the # passed group. Iterate over this list to do something per # collided sprite. Set dokill argument to True to kill the sprite. collided_enemies = pg.sprite.spritecollide(player, enemy_list, True) for enemy in collided_enemies: player.health -= 25 collided_coins = pg.sprite.spritecollide(player, coins, True) for coin in collided_coins: score += 1 # DRAW SECTION screen.fill(BLACK) all_sprites.draw(screen) health_label = font_health.render("Health: "+str(player.health),True,WHITE) score_label = font_score.render("Score: " + str(score),True, WHITE) screen.blit(score_label,[100,480]) screen.blit(health_label,[190,480]) pg.display.flip() clock.tick(60) if __name__ == '__main__': main() pg.quit() sys.exit()