""" ----------------------------------------- Project: Escape Drake Standard: 91896 (2.7) School: Tauranga Boys' Collage Auther: Hunter Moss Date: May 2024 Python: 3.11.2 ----------------------------------------- """ import pygame import os import pickle from os import path from pygame import mixer pygame.mixer.pre_init(48000, -16, 2, 512) mixer.init pygame.init() # Screen Size screen_width = 1000 screen_height = 1000 screen = pygame.display.set_mode((screen_width, screen_height)) pygame.display.set_caption('Escape Drake') # Load images background_img = pygame.image.load('Images/Background.png') mute_img = pygame.image.load('Images/mute_icn.png') unmute_img = pygame.image.load('Images/unmute_icn.png') mute_img = pygame.transform.scale(mute_img, (50, 50)) unmute_img = pygame.transform.scale(unmute_img, (50, 50)) # Buttons mute_button = mute_img.get_rect() mute_button = mute_img.get_rect(topleft=(screen_width - 50, 0)) # Colours Red = (255, 0, 0) Black = (0, 0, 0) # Constants tile_size = 50 fps = 60 # Game variables level = 1 font_size = 60 #Fonts monsterfnt = pygame.font.Font('Fonts/Melted Monster.ttf', 60) monsterfnt2 = pygame.font.Font('Fonts/Melted Monster.ttf', 120) monsterfnt3 = pygame.font.Font('Fonts/Melted Monster.ttf', 80) monsterfnt4 = pygame.font.Font('Fonts/Melted Monster.ttf', 40) #load music bg_music = pygame.mixer.music.load('Music/Not Like Us.mp3') pygame.mixer.music.play(-1, 0.0, 0) pygame.mixer.music.set_volume(0.5) #Halve the volume of the music # Load player images function def load_images(folder, prefix, count): return [pygame.transform.scale(pygame.image.load(os.path.join(folder, f"{prefix} ({i}).png")), (tile_size, tile_size)) for i in range(1, count + 1)] class Player(pygame.sprite.Sprite): def __init__(self): super().__init__() self.images = { 'idle': load_images('Images/Player', 'idle', 16), 'run': load_images('Images/Player', 'run', 20), 'jump': load_images('Images/Player', 'jump', 30), 'dead': load_images('Images/Player', 'dead', 30) } self.image = self.images['idle'][0] self.rect = self.image.get_rect(topleft=(50, screen_height - 50)) self.vel = pygame.math.Vector2(0, 0) self.speed = 5 self.gravity = 0.5 self.jump_power = -10 self.on_ground = False self.jumping = False self.facing_right = True self.frame = 0 self.idle_animation_speed = 0.2 self.animation_speed = 0.6 self.alive = True self.keys_collected = 0 # Add key counter self.moving = False # Track if player has started moving self.facing_before_death = True # Add this to track facing direction before death def reset(self): self.rect.topleft = (10, screen_height - 50) self.alive = True self.keys_collected = 0 # Reset key counter self.moving = False # Reset moving state self.frame = 0 # Reset animation frame def update(self, tiles, door, drakes, trogs, keys, boss_drake): if self.alive: keys_pressed = pygame.key.get_pressed() # Horizontal movement if keys_pressed[pygame.K_LEFT] or keys_pressed[pygame.K_RIGHT]: self.moving = True # Player has started moving if keys_pressed[pygame.K_LEFT]: self.vel.x = -self.speed self.facing_right = False elif keys_pressed[pygame.K_RIGHT]: self.vel.x = self.speed self.facing_right = True else: self.vel.x = 0 # Jumping if (keys_pressed[pygame.K_UP]) and self.on_ground: self.vel.y = self.jump_power self.on_ground = False self.jumping = True # Apply gravity self.vel.y += self.gravity # Horizontal collision self.rect.x += self.vel.x self.check_collision(tiles, 'x') # Vertical collision self.rect.y += self.vel.y self.check_collision(tiles, 'y') # Screen collision self.rect.clamp_ip(pygame.Rect(0, 0, screen_width, screen_height)) # Stops the player from walking off the size of the screen # Check door collision if door and self.rect.colliderect(door.rect): door.open = True if level < 3: increase_level() else: end_game() # Check collision with DRAKES for drake in drakes: if self.rect.colliderect(drake.rect): self.alive = False self.facing_before_death = self.facing_right # Store facing direction before death for trog in trogs: if self.rect.colliderect(trog.rect): self.alive = False self.facing_before_death = self.facing_right # Check collision with keys for key in keys: if self.rect.colliderect(key.rect): keys.remove(key) self.keys_collected += 1 # Check collision with Boss Drake if boss_drake and self.rect.colliderect(boss_drake.rect): self.alive = False self.facing_before_death = self.facing_right # Store facing direction before death # Animation handling if self.jumping: self.image = self.images['jump'][0] elif self.vel.x != 0: self.frame = (self.frame + self.animation_speed) % len(self.images['run']) self.image = self.images['run'][int(self.frame)] else: self.frame = (self.frame + self.idle_animation_speed) % len(self.images['idle']) self.image = self.images['idle'][int(self.frame)] # Flip sprite if not self.facing_right: self.image = pygame.transform.flip(self.image, True, False) # Reset jumping state if on ground if self.rect.bottom >= screen_height: self.on_ground = True self.jumping = False self.vel.y = 0 else: # Play death animation self.frame = (self.frame + self.animation_speed) % len(self.images['dead']) self.image = self.images['dead'][int(self.frame)] # Flip death animation based on the stored direction if not self.facing_before_death: self.image = pygame.transform.flip(self.image, True, False) # Show retry screen after death animation finishes if self.frame >= len(self.images['dead']) - 1: show_retry_screen() if boss_drake: boss_drake.reset() # Reset boss drake position when player dies def check_collision(self, tiles, direction): for tile in tiles: if self.rect.colliderect(tile.rect): if direction == 'x': if self.vel.x > 0: # Moving right self.rect.right = tile.rect.left elif self.vel.x < 0: # Moving left self.rect.left = tile.rect.right elif direction == 'y': if self.vel.y > 0: # Moving down self.rect.bottom = tile.rect.top self.vel.y = 0 self.on_ground = True self.jumping = False elif self.vel.y < 0: # Moving up self.rect.top = tile.rect.bottom self.vel.y = 0 # Check if collided with a platform if isinstance(tile, Platform) and direction == 'y': if self.vel.y > 0: # Moving down self.rect.bottom = tile.rect.top self.vel.y = 0 self.on_ground = True self.jumping = False elif self.vel.y < 0: # Moving up self.rect.top = tile.rect.bottom self.vel.y = 0 # Check collision with platform groups platform_groups = pygame.sprite.Group(world.platform_vert_group, world.platform_horz_group) platform_collisions = pygame.sprite.spritecollide(self, platform_groups, False) for platform in platform_collisions: if direction == 'y': if self.vel.y > 0: # Moving down self.rect.bottom = platform.rect.top self.vel.y = 0 self.on_ground = True self.jumping = False elif self.vel.y < 0: # Moving up self.rect.top = platform.rect.bottom self.vel.y = 0 # Block Classes class Block(pygame.sprite.Sprite): def __init__(self, x, y, image, scale_to_tile=True): super().__init__() if scale_to_tile: self.image = pygame.transform.scale(image, (tile_size, tile_size)) else: self.image = image self.rect = self.image.get_rect(topleft=(x, y)) class Brick0(Block): def __init__(self, x, y): image = pygame.image.load('Images/Level/brick_dark0.png') super().__init__(x, y, image) class Brick1(Block): def __init__(self, x, y): image = pygame.image.load('Images/Level/brick_dark1.png') super().__init__(x, y, image) class Brick2(Block): def __init__(self, x, y): image = pygame.image.load('Images/Level/brick_dark2.png') super().__init__(x, y, image) class Brick3(Block): def __init__(self, x, y): image = pygame.image.load('Images/Level/brick_dark3.png') super().__init__(x, y, image) class Cobbweb(Block): def __init__(self, x, y): image = pygame.image.load('Images/Level/Cobbwebs.png') super().__init__(x, y, image) class Grill(Block): def __init__(self, x, y): image = pygame.image.load('Images/Level/Grill.png') super().__init__(x, y, image) class Skulls(Block): def __init__(self, x, y): image = pygame.image.load('Images/Level/skulls.png') super().__init__(x, y, image) class Platform(Block): def __init__(self, x, y): image = pygame.image.load('Images/Level/platform.png') original_width, original_height = image.get_size() resized_image = pygame.transform.scale(image, (tile_size, original_height + 5)) adjusted_y = y + tile_size - original_height - 5 # Position at the bottom of the tile super().__init__(x, adjusted_y, resized_image, scale_to_tile=False) class Platform_horz(Block): def __init__(self, x, y): image = pygame.image.load('Images/Level/platform_horz.png') original_width, original_height = image.get_size() resized_image = pygame.transform.scale(image, (tile_size, original_height + 5)) adjusted_y = y + tile_size - original_height - 5 # Position at the bottom of the tile super().__init__(x, adjusted_y, resized_image, scale_to_tile=False) self.move_speed = 2 self.range = 150 # Adjust this value to change the platform's movement range self.start_x = x self.direction = 1 def update(self): self.rect.x += self.move_speed * self.direction if self.direction == 1: if self.rect.x - self.start_x >= self.range: self.direction = -1 else: if self.start_x - self.rect.x >= self.range: self.direction = 1 class Platform_vert(Block): def __init__(self, x, y): image = pygame.image.load('Images/Level/platform_vert.png') original_width, original_height = image.get_size() resized_image = pygame.transform.scale(image, (tile_size, original_height + 5)) adjusted_y = y + tile_size - original_height - 5 # Position at the bottom of the tile super().__init__(x, adjusted_y, resized_image, scale_to_tile=False) self.move_speed = 3 self.range = 150 # Adjust this value to change the platform's movement range self.start_y = y # Remember the starting y position self.direction = 1 def update(self): self.rect.y += self.move_speed * self.direction if self.direction == 1: if self.rect.y - self.start_y >= self.range: self.direction = -1 else: if self.start_y - self.rect.y >= self.range: self.direction = 1 class Trog(Block): def __init__(self, x, y): image = pygame.image.load('Images/dngn_altar_trog.png') image = pygame.transform.scale(image, (tile_size, tile_size)) super().__init__(x, y, image) class Key(Block): def __init__(self, x, y): image = pygame.image.load('Images/Key.png') resized_image = pygame.transform.scale(image, (50, 20)) # Resize key image to 50x20 pixels adjusted_y = y + (tile_size - resized_image.get_height()) // 2 super().__init__(x, adjusted_y, resized_image, scale_to_tile=False) class Drake(Block): def __init__(self, x, y): image = pygame.image.load('Images/DRAKE.png') image = pygame.transform.scale(image, (tile_size, tile_size)) super().__init__(x, y, image) class BossDrake(pygame.sprite.Sprite): def __init__(self, x, y): super().__init__() image = pygame.image.load('Images/DRAKE.png') self.image = pygame.transform.scale(image, (tile_size * 2, tile_size * 2)) # Make Boss Drake twice the size of a normal one self.rect = self.image.get_rect(topleft=(x, y)) self.starting_position = (0 + (tile_size * 2), 0) self.speed = 0.6 # Speed at which the boss drake moves def reset(self): self.rect.topleft = self.starting_position def update(self, player): #Allows drake to follow the player in the boss level if self.rect.x < player.rect.x: self.rect.x += self.speed elif self.rect.x > player.rect.x: self.rect.x -= self.speed if self.rect.y < player.rect.y: self.rect.y += self.speed elif self.rect.y > player.rect.y: self.rect.y -= self.speed class Door(Block): def __init__(self, x, y): self.closed_image = pygame.image.load('Images/dngn_closed_door.png') self.open_image = pygame.image.load('Images/dngn_open_door.png') # Resize images to match the tile size self.closed_image = pygame.transform.scale(self.closed_image, (tile_size, tile_size)) self.open_image = pygame.transform.scale(self.open_image, (tile_size, tile_size)) super().__init__(x, y, self.closed_image) self.open = False def update(self): if self.open: self.image = self.open_image else: self.image = self.closed_image # World class class World: def __init__(self, data): self.tile_list = [] self.floor_group = pygame.sprite.Group() # Group for floor tiles self.drake_group = pygame.sprite.Group() # Group for drakes self.trog_group = pygame.sprite.Group() self.key_group = pygame.sprite.Group() # Group for keys self.platform_vert_group = pygame.sprite.Group() # Group for vertical platforms self.platform_horz_group = pygame.sprite.Group() # Group for horizontal platforms self.door_group = pygame.sprite.GroupSingle() # Group for the door self.load_world(data) def load_world(self, data): block_types = { 'brick0': Brick0, 'brick1': Brick1, 'brick2': Brick2, 'brick3': Brick3, 'cobbweb': Cobbweb, 'grill': Grill, 'skulls': Skulls, 'platform': Platform, 'platform_vert': Platform_vert, 'platform_horz': Platform_horz, 'trog': Trog, 'key': Key, 'drake': Drake, 'door': Door } for block in data: x, y, block_type = block if block_type in block_types: block_class = block_types[block_type] block_instance = block_class(x, y) if block_type == 'door': self.door_group.add(block_instance) elif block_type == 'drake': self.drake_group.add(block_instance) elif block_type == 'trog': self.trog_group.add(block_instance) elif block_type == 'key': self.key_group.add(block_instance) elif block_type == 'platform_vert': self.platform_vert_group.add(block_instance) elif block_type == 'platform_horz': self.platform_horz_group.add(block_instance) else: self.tile_list.append(block_instance) self.floor_group.add(block_instance) def draw(self): for tile in self.tile_list: screen.blit(tile.image, tile.rect) self.door_group.draw(screen) self.drake_group.draw(screen) self.trog_group.draw(screen) self.key_group.draw(screen) self.platform_vert_group.draw(screen) self.platform_horz_group.draw(screen) def get_tiles(self): return self.floor_group def get_keys(self): return self.key_group def increase_level(): global world global player global level level += 1 print(f"Loading level {level}") if path.exists(f'level_data/level{level}_data'): with open(f'level_data/level{level}_data', 'rb') as pickle_in: world_data = pickle.load(pickle_in) player.reset() # Reset player state world = World(world_data) # Re-initialize the world else: end_game() def reset_game(): global level, world, player level = 1 player.reset() reset_level() def reset_level(): global world global player global level print(f"Resetting level {level}") if path.exists(f'level_data/level{level}_data'): with open(f'level_data/level{level}_data', 'rb') as pickle_in: world_data = pickle.load(pickle_in) world = World(world_data) else: world = World([]) # Default to an empty world if level data is not found player.reset() # Reset player state def end_game(): global run, retry_screen text = monsterfnt3.render('You Escaped Drake', True, (255, 255, 255)) text_rect = text.get_rect(center=(screen_width // 2, screen_height // 2)) retry_button_rect = pygame.Rect(0, 0, 900, 80) retry_button_rect.center = (screen_width // 2, screen_height // 2 + 200) retry_screen = True while retry_screen: for event in pygame.event.get(): if event.type == pygame.QUIT: pygame.quit() quit() if event.type == pygame.KEYDOWN: if event.key == pygame.K_SPACE: # Press 'Space' to retry reset_game() # Reset the game state start_menu() # Return to the main menu retry_screen = False screen.blit(text, text_rect) pygame.draw.rect(screen, (255, 0, 0), retry_button_rect) retry_text = monsterfnt4.render('Press Space to Escape Again With a Twist', True, (255, 255, 255)) retry_text_rect = retry_text.get_rect(center=retry_button_rect.center) screen.blit(retry_text, retry_text_rect) pygame.display.flip() def show_retry_screen(): global run, retry_screen text = monsterfnt.render('You failed to Escape Drake!', True, (255, 255, 255)) text_rect = text.get_rect(center=(screen_width // 2, screen_height // 2)) retry_button_rect = pygame.Rect(0, 0, 650, 80) retry_button_rect.center = (screen_width // 2, screen_height // 2 + 100) retry_screen = True while retry_screen: for event in pygame.event.get(): if event.type == pygame.QUIT: pygame.quit() quit() if event.type == pygame.KEYDOWN: if event.key == pygame.K_SPACE: # Press 'Space' to retry reset_level() retry_screen = False screen.blit(text, text_rect) pygame.draw.rect(screen, (255, 0, 0), retry_button_rect) retry_text = monsterfnt.render('Press Space to Retry', True, (0, 0, 0)) retry_text_rect = retry_text.get_rect(center=retry_button_rect.center) screen.blit(retry_text, retry_text_rect) pygame.display.flip() # Start Menu function def start_menu(): global run menu = True while menu: title = monsterfnt2.render('Escape Drake', True, (255, 0, 0)) screen.fill((0, 0, 0)) screen.blit(title, (screen_width // 2 - title.get_width() // 2, screen_height // 4)) start_button = pygame.Rect(screen_width // 2 - 250, screen_height // 2 - 50, 500, 100) pygame.draw.rect(screen, (0, 255, 0), start_button) start_text = monsterfnt.render('Start Running!', True, (0, 0, 0)) screen.blit(start_text, (screen_width // 2 - start_text.get_width() // 2, screen_height // 2 - start_text.get_height() // 2)) quit_button = pygame.Rect(screen_width // 2 - 250, screen_height // 2 + 125, 500, 100) pygame.draw.rect(screen, (255, 0, 0), quit_button) quit_text = monsterfnt.render('Nevermind :(', True, (0, 0, 0)) screen.blit(quit_text, (screen_width // 2 - quit_text.get_width() // 2, screen_height // 2 + 100 + quit_text.get_height() // 2)) pygame.display.flip() for event in pygame.event.get(): if event.type == pygame.QUIT: pygame.quit() exit() if event.type == pygame.MOUSEBUTTONDOWN: if start_button.collidepoint(event.pos): reset_game() # Reset the game state when starting a new game menu = False if quit_button.collidepoint(event.pos): pygame.quit() exit() def main(): global world global player global level global run run = True clock = pygame.time.Clock() player = Player() player_group = pygame.sprite.Group() player_group.add(player) is_muted = False boss_drake = False # Initialize boss_drake variable # Draw mute button mute_image = mute_img if is_muted else unmute_img screen.blit(mute_image, mute_button.topleft) # Load the first level start_menu() reset_level() while run: # Event handling for event in pygame.event.get(): if event.type == pygame.QUIT: run = False if event.type == pygame.MOUSEBUTTONDOWN: if mute_button.collidepoint(event.pos): is_muted = not is_muted if is_muted: pygame.mixer.music.pause() else: pygame.mixer.music.unpause() # Check if in the final level and player has started moving if level == 3 and player.moving and boss_drake is False: boss_drake = BossDrake(0 + (tile_size * 2), 0) # Update player and check collisions player.update(world.get_tiles(), world.door_group.sprite, world.drake_group, world.trog_group, world.get_keys(), boss_drake) if world.door_group.sprite: world.door_group.update() # Update moving platforms for platform in world.platform_vert_group: platform.update() for platform in world.platform_horz_group: platform.update() # Drawing screen.blit(background_img, (0, 0)) # Draw background world.draw() player_group.draw(screen) world.door_group.draw(screen) world.key_group.draw(screen) world.drake_group.draw(screen) world.trog_group.draw(screen) world.platform_vert_group.draw(screen) world.platform_horz_group.draw(screen) # Draw mute button mute_image = mute_img if is_muted else unmute_img screen.blit(mute_image, mute_button.topleft) if boss_drake: boss_drake.update(player) screen.blit(boss_drake.image, boss_drake.rect.topleft) # Update display pygame.display.flip() clock.tick(fps) pygame.quit() if __name__ == "__main__": main()