Skip to content

Commit 14fe3d5

Browse files
committed
add pacman game tutorial
1 parent a7a830c commit 14fe3d5

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

60 files changed

+488
-0
lines changed

README.md

+1

gui-programming/pacman-game/README.md

+1
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
from os import walk
2+
import pygame
3+
4+
def import_sprite(path):
5+
surface_list = []
6+
for _, __, img_file in walk(path):
7+
for image in img_file:
8+
full_path = f"{path}/{image}"
9+
img_surface = pygame.image.load(full_path).convert_alpha()
10+
surface_list.append(img_surface)
11+
return surface_list
903 Bytes
289 Bytes
259 Bytes
284 Bytes
280 Bytes
259 Bytes
284 Bytes
259 Bytes
322 Bytes
259 Bytes

gui-programming/pacman-game/berry.py

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
import pygame
2+
3+
from settings import CHAR_SIZE, PLAYER_SPEED
4+
5+
class Berry(pygame.sprite.Sprite):
6+
def __init__(self, row, col, size, is_power_up = False):
7+
super().__init__()
8+
self.power_up = is_power_up
9+
self.size = size
10+
self.color = pygame.Color("violetred")
11+
self.thickness = size
12+
self.abs_x = (row * CHAR_SIZE) + (CHAR_SIZE // 2)
13+
self.abs_y = (col * CHAR_SIZE) + (CHAR_SIZE // 2)
14+
15+
# temporary rect for colliderect-checking
16+
self.rect = pygame.Rect(self.abs_x,self.abs_y, self.size * 2, self.size * 2)
17+
18+
def update(self, screen):
19+
self.rect = pygame.draw.circle(screen, self.color, (self.abs_x, self.abs_y), self.size, self.thickness)
20+

gui-programming/pacman-game/cell.py

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
import pygame
2+
3+
class Cell(pygame.sprite.Sprite):
4+
def __init__(self, row, col, length, width):
5+
super().__init__()
6+
self.width = length
7+
self.height = width
8+
self.id = (row, col)
9+
self.abs_x = row * self.width
10+
self.abs_y = col * self.height
11+
12+
self.rect = pygame.Rect(self.abs_x,self.abs_y,self.width,self.height)
13+
14+
self.occupying_piece = None
15+
16+
def update(self, screen):
17+
pygame.draw.rect(screen, pygame.Color("blue2"), self.rect)
+40
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,40 @@
1+
import pygame
2+
3+
from settings import WIDTH, HEIGHT, CHAR_SIZE
4+
5+
pygame.font.init()
6+
7+
class Display:
8+
def __init__(self, screen):
9+
self.screen = screen
10+
self.font = pygame.font.SysFont("ubuntumono", CHAR_SIZE)
11+
self.game_over_font = pygame.font.SysFont("dejavusansmono", 48)
12+
self.text_color = pygame.Color("crimson")
13+
14+
def show_life(self, life):
15+
img_path = "assets/life/life.png"
16+
life_image = pygame.image.load(img_path)
17+
life_image = pygame.transform.scale(life_image, (CHAR_SIZE, CHAR_SIZE))
18+
life_x = CHAR_SIZE // 2
19+
20+
if life != 0:
21+
for life in range(life):
22+
self.screen.blit(life_image, (life_x, HEIGHT + (CHAR_SIZE // 2)))
23+
life_x += CHAR_SIZE
24+
25+
def show_level(self, level):
26+
level_x = WIDTH // 3
27+
level = self.font.render(f'Level {level}', True, self.text_color)
28+
self.screen.blit(level, (level_x, (HEIGHT + (CHAR_SIZE // 2))))
29+
30+
def show_score(self, score):
31+
score_x = WIDTH // 3
32+
score = self.font.render(f'{score}', True, self.text_color)
33+
self.screen.blit(score, (score_x * 2, (HEIGHT + (CHAR_SIZE // 2))))
34+
35+
# add game over message
36+
def game_over(self):
37+
message = self.game_over_font.render(f'GAME OVER!!', True, pygame.Color("chartreuse"))
38+
instruction = self.font.render(f'Press "R" to Restart', True, pygame.Color("aqua"))
39+
self.screen.blit(message, ((WIDTH // 4), (HEIGHT // 3)))
40+
self.screen.blit(instruction, ((WIDTH // 4), (HEIGHT // 2)))

gui-programming/pacman-game/ghost.py

+70
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,70 @@
1+
import pygame
2+
import random
3+
import time
4+
5+
from settings import WIDTH, CHAR_SIZE, GHOST_SPEED
6+
7+
class Ghost(pygame.sprite.Sprite):
8+
def __init__(self, row, col, color):
9+
super().__init__()
10+
self.abs_x = (row * CHAR_SIZE)
11+
self.abs_y = (col * CHAR_SIZE)
12+
13+
self.rect = pygame.Rect(self.abs_x, self.abs_y, CHAR_SIZE, CHAR_SIZE)
14+
self.move_speed = GHOST_SPEED
15+
self.color = pygame.Color(color)
16+
self.move_directions = [(-1,0), (0,-1), (1,0), (0,1)]
17+
18+
self.moving_dir = "up"
19+
self.img_path = f'assets/ghosts/{color}/'
20+
self.img_name = f'{self.moving_dir}.png'
21+
self.image = pygame.image.load(self.img_path + self.img_name)
22+
self.image = pygame.transform.scale(self.image, (CHAR_SIZE, CHAR_SIZE))
23+
self.rect = self.image.get_rect(topleft = (self.abs_x, self.abs_y))
24+
self.mask = pygame.mask.from_surface(self.image)
25+
26+
self.directions = {'left': (-self.move_speed, 0), 'right': (self.move_speed, 0), 'up': (0, -self.move_speed), 'down': (0, self.move_speed)}
27+
self.keys = ['left', 'right', 'up', 'down']
28+
self.direction = (0, 0)
29+
30+
def move_to_start_pos(self):
31+
self.rect.x = self.abs_x
32+
self.rect.y = self.abs_y
33+
34+
def is_collide(self, x, y, walls_collide_list):
35+
tmp_rect = self.rect.move(x, y)
36+
if tmp_rect.collidelist(walls_collide_list) == -1:
37+
return False
38+
return True
39+
40+
def _animate(self):
41+
self.img_name = f'{self.moving_dir}.png'
42+
self.image = pygame.image.load(self.img_path + self.img_name)
43+
self.image = pygame.transform.scale(self.image, (CHAR_SIZE, CHAR_SIZE))
44+
self.rect = self.image.get_rect(topleft=(self.rect.x, self.rect.y))
45+
46+
def update(self, walls_collide_list):
47+
# ghost movement
48+
available_moves = []
49+
for key in self.keys:
50+
if not self.is_collide(*self.directions[key], walls_collide_list):
51+
available_moves.append(key)
52+
53+
randomizing = False if len(available_moves) <= 2 and self.direction != (0,0) else True
54+
# 60% chance of randomizing ghost move
55+
if randomizing and random.randrange( 0,100 ) <= 60:
56+
self.moving_dir = random.choice(available_moves)
57+
self.direction = self.directions[self.moving_dir]
58+
59+
if not self.is_collide(*self.direction, walls_collide_list):
60+
self.rect.move_ip(self.direction)
61+
else:
62+
self.direction = (0,0)
63+
64+
# teleporting to the other side of the map
65+
if self.rect.right <= 0:
66+
self.rect.x = WIDTH
67+
elif self.rect.left >= WIDTH:
68+
self.rect.x = 0
69+
70+
self._animate()

gui-programming/pacman-game/main.py

+32
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
import pygame, sys
2+
from settings import WIDTH, HEIGHT, NAV_HEIGHT
3+
from world import World
4+
5+
pygame.init()
6+
7+
screen = pygame.display.set_mode((WIDTH, HEIGHT + NAV_HEIGHT))
8+
pygame.display.set_caption("PacMan")
9+
10+
class Main:
11+
def __init__(self, screen):
12+
self.screen = screen
13+
self.FPS = pygame.time.Clock()
14+
15+
def main(self):
16+
world = World(self.screen)
17+
while True:
18+
self.screen.fill("black")
19+
20+
for event in pygame.event.get():
21+
if event.type == pygame.QUIT:
22+
pygame.quit()
23+
sys.exit()
24+
25+
world.update()
26+
pygame.display.update()
27+
self.FPS.tick(30)
28+
29+
30+
if __name__ == "__main__":
31+
play = Main(screen)
32+
play.main()

gui-programming/pacman-game/pac.py

+93
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,93 @@
1+
import pygame
2+
3+
from settings import CHAR_SIZE, PLAYER_SPEED
4+
from animation import import_sprite
5+
6+
class Pac(pygame.sprite.Sprite):
7+
def __init__(self, row, col):
8+
super().__init__()
9+
10+
self.abs_x = (row * CHAR_SIZE)
11+
self.abs_y = (col * CHAR_SIZE)
12+
13+
# pac animation
14+
self._import_character_assets()
15+
self.frame_index = 0
16+
self.animation_speed = 0.5
17+
self.image = self.animations["idle"][self.frame_index]
18+
self.rect = self.image.get_rect(topleft = (self.abs_x, self.abs_y))
19+
self.mask = pygame.mask.from_surface(self.image)
20+
21+
self.pac_speed = PLAYER_SPEED
22+
self.immune_time = 0
23+
self.immune = False
24+
25+
self.directions = {'left': (-PLAYER_SPEED, 0), 'right': (PLAYER_SPEED, 0), 'up': (0, -PLAYER_SPEED), 'down': (0, PLAYER_SPEED)}
26+
self.keys = {'left': pygame.K_LEFT, 'right': pygame.K_RIGHT, 'up': pygame.K_UP, 'down': pygame.K_DOWN}
27+
self.direction = (0, 0)
28+
29+
# pac status
30+
self.status = "idle"
31+
self.life = 3
32+
self.pac_score = 0
33+
34+
35+
# gets all the image needed for animating specific player action
36+
def _import_character_assets(self):
37+
character_path = "assets/pac/"
38+
self.animations = {
39+
"up": [],
40+
"down": [],
41+
"left": [],
42+
"right": [],
43+
"idle": [],
44+
"power_up": []
45+
}
46+
for animation in self.animations.keys():
47+
full_path = character_path + animation
48+
self.animations[animation] = import_sprite(full_path)
49+
50+
51+
def _is_collide(self, x, y):
52+
tmp_rect = self.rect.move(x, y)
53+
if tmp_rect.collidelist(self.walls_collide_list) == -1:
54+
return False
55+
return True
56+
57+
58+
def move_to_start_pos(self):
59+
self.rect.x = self.abs_x
60+
self.rect.y = self.abs_y
61+
62+
63+
# update with sprite/sheets
64+
def animate(self, pressed_key, walls_collide_list):
65+
animation = self.animations[self.status]
66+
67+
# loop over frame index
68+
self.frame_index += self.animation_speed
69+
if self.frame_index >= len(animation):
70+
self.frame_index = 0
71+
image = animation[int(self.frame_index)]
72+
self.image = pygame.transform.scale(image, (CHAR_SIZE, CHAR_SIZE))
73+
74+
self.walls_collide_list = walls_collide_list
75+
for key, key_value in self.keys.items():
76+
if pressed_key[key_value] and not self._is_collide(*self.directions[key]):
77+
self.direction = self.directions[key]
78+
self.status = key if not self.immune else "power_up"
79+
break
80+
81+
if not self._is_collide(*self.direction):
82+
self.rect.move_ip(self.direction)
83+
self.status = self.status if not self.immune else "power_up"
84+
if self._is_collide(*self.direction):
85+
self.status = "idle" if not self.immune else "power_up"
86+
87+
88+
def update(self):
89+
# Timer based from FPS count
90+
self.immune = True if self.immune_time > 0 else False
91+
self.immune_time -= 1 if self.immune_time > 0 else 0
92+
93+
self.rect = self.image.get_rect(topleft=(self.rect.x, self.rect.y))
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
pygame
+33
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
MAP = [
2+
['1','1','1','1','1','1','1','1','1','1','1','1','1','1','1','1','1','1','1'],
3+
['1',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ','1'],
4+
['1','B','1','1',' ','1','1','1',' ','1',' ','1','1','1',' ','1','1','B','1'],
5+
['1',' ',' ',' ',' ','1',' ',' ',' ','1',' ',' ',' ','1',' ',' ',' ',' ','1'],
6+
['1','1',' ','1',' ','1',' ','1',' ','1',' ','1',' ','1',' ','1',' ','1','1'],
7+
['1',' ',' ','1',' ',' ',' ','1',' ',' ',' ','1',' ',' ',' ','1',' ',' ','1'],
8+
['1',' ','1','1','1','1',' ','1','1','1','1','1',' ','1','1','1','1',' ','1'],
9+
['1',' ',' ',' ',' ',' ',' ',' ',' ','r',' ',' ',' ',' ',' ',' ',' ',' ','1'],
10+
['1','1',' ','1','1','1',' ','1','1','-','1','1',' ','1','1','1',' ','1','1'],
11+
[' ',' ',' ',' ',' ','1',' ','1','s','p','o','1',' ','1',' ',' ',' ',' ',' '],
12+
['1','1',' ','1',' ','1',' ','1','1','1','1','1',' ','1',' ','1',' ','1','1'],
13+
['1',' ',' ','1',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ','1',' ',' ','1'],
14+
['1',' ','1','1','1','1',' ','1','1','1','1','1',' ','1','1','1','1',' ','1'],
15+
['1',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ','1'],
16+
['1','1','1',' ','1','1','1',' ','1','1','1',' ','1','1','1',' ','1','1','1'],
17+
['1',' ',' ',' ','1',' ',' ',' ',' ','P',' ',' ',' ',' ','1',' ',' ',' ','1'],
18+
['1','B','1',' ','1',' ','1',' ','1','1','1',' ','1',' ','1',' ','1','B','1'],
19+
['1',' ','1',' ',' ',' ','1',' ',' ',' ',' ',' ','1',' ',' ',' ','1',' ','1'],
20+
['1',' ','1','1','1',' ','1','1','1',' ','1','1','1',' ','1','1','1',' ','1'],
21+
['1',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ',' ','1'],
22+
['1','1','1','1','1','1','1','1','1','1','1','1','1','1','1','1','1','1','1']
23+
]
24+
25+
BOARD_RATIO = (len(MAP[0]), len(MAP))
26+
CHAR_SIZE = 32
27+
28+
WIDTH, HEIGHT = (BOARD_RATIO[0] * CHAR_SIZE, BOARD_RATIO[1] * CHAR_SIZE)
29+
NAV_HEIGHT = 64
30+
31+
PLAYER_SPEED = CHAR_SIZE // 4
32+
33+
GHOST_SPEED = 4

0 commit comments

Comments
 (0)