Python实现寻宝小游戏

Posted 楠change

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Python实现寻宝小游戏相关的知识,希望对你有一定的参考价值。

#!/usr/bin/env python3
# -*- coding: utf-8 -*-
"""
Treasure Hunter

In this game, you will play as a treasure hunter to find a mysterious treasure. 
In order to find the treasure, you need to defeat a great number of monsters with 
time limitation.

Created by Group ShowMaker

"""

import os
import pygame
import random
import numpy

#Initialise Screen
pygame.init()
pygame.mixer.init()
pygame.display.init()
pygame.font.init()
pygame.display.set_caption(‘Treasure Hunter‘)
FPS = 13
width = 960
height = 540
screen = pygame.display.set_mode((width, height))
clock = pygame.time.Clock()
dt = clock.tick(FPS) / 1000 #the time interval of two images in the animation

# assets folder
img_dir = ‘assets/pictures‘
music_dir = ‘assets/music‘

# sound effects
shooting_sound = pygame.mixer.Sound(os.path.join(music_dir, ‘shoot.wav‘))
boss_bgm = os.path.join(music_dir, ‘boss_bgm.wav‘)
regular_bgm = os.path.join(music_dir, ‘regular_bgm.wav‘)
die_sound = pygame.mixer.Sound(os.path.join(music_dir, ‘enemy_die.wav‘))
boss_entry = pygame.mixer.Sound(os.path.join(music_dir, ‘boss_entry.wav‘))
boss_die = pygame.mixer.Sound(os.path.join(music_dir, ‘boss_die.wav‘))
level_up = pygame.mixer.Sound(os.path.join(music_dir, ‘level_up.wav‘))
open_chest = pygame.mixer.Sound(os.path.join(music_dir, ‘open_chest.wav‘))
time_limit=pygame.mixer.Sound(os.path.join(music_dir, ‘time_limit.wav‘))

#game background images
background = pygame.image.load(os.path.join(img_dir, ‘bg1.png‘))
background2 = pygame.image.load(os.path.join(img_dir, ‘bg2.jpg‘))
background3 = pygame.image.load(os.path.join(img_dir, ‘bg1.png‘))
background4 = pygame.image.load(os.path.join(img_dir, ‘bg3.jpg‘))

#menu related images
menu_image1 = pygame.image.load(os.path.join(img_dir, ‘main1.jpg‘))
menu_image2 = pygame.image.load(os.path.join(img_dir, ‘main2.jpg‘))
button_quit = pygame.image.load(os.path.join(img_dir, ‘button_quit2.png‘))
button_start = pygame.image.load(os.path.join(img_dir, ‘button_start2.png‘))
button_normal = pygame.image.load(os.path.join(img_dir, ‘normal.png‘))
button_difficult = pygame.image.load(os.path.join(img_dir, ‘difficult.png‘))
die_image = pygame.image.load(os.path.join(img_dir, ‘died.jpg‘))
final1 = pygame.image.load(os.path.join(img_dir, ‘congrats.jpg‘))
bginfo = pygame.image.load(os.path.join(img_dir, ‘bginfo.jpg‘))

#hero images
heroimages = [pygame.image.load(os.path.join(img_dir, ‘h1.png‘)),
              pygame.image.load(os.path.join(img_dir, ‘h2.png‘)),
              pygame.image.load(os.path.join(img_dir, ‘h3.png‘)),
              pygame.image.load(os.path.join(img_dir, ‘h4.png‘)),
              pygame.image.load(os.path.join(img_dir, ‘h5.png‘))]
#enemy images
e1images = [pygame.image.load(os.path.join(img_dir, ‘boss01_1.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss01_2.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss01_3.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss01_4.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss01_5.png‘))]
e2images = [pygame.image.load(os.path.join(img_dir, ‘boss05_1.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss05_2.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss05_3.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss05_4.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss05_5.png‘))]
e3images = [pygame.image.load(os.path.join(img_dir, ‘boss02_1.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss02_2.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss02_3.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss02_4.png‘)),
            pygame.image.load(os.path.join(img_dir, ‘boss02_5.png‘))]
bossimages = [pygame.image.load(os.path.join(img_dir, ‘boss08_1.png‘)),
              pygame.image.load(os.path.join(img_dir, ‘boss08_2.png‘)),
              pygame.image.load(os.path.join(img_dir, ‘boss08_3.png‘)),
              pygame.image.load(os.path.join(img_dir, ‘boss08_4.png‘)),
              pygame.image.load(os.path.join(img_dir, ‘boss08_5.png‘))]

#bullet images
bull1 = pygame.image.load(os.path.join(img_dir, ‘bullet1.png‘))
bull2 = pygame.image.load(os.path.join(img_dir, ‘bullet2.png‘))

#props
warning_img = pygame.image.load(os.path.join(img_dir, ‘warning.png‘))
chest = pygame.image.load(os.path.join(img_dir, ‘chest.png‘))
book = pygame.image.load(os.path.join(img_dir, ‘book.png‘))
weapon = pygame.image.load(os.path.join(img_dir, ‘weapon.png‘))
gatei1 = pygame.image.load(os.path.join(img_dir, ‘gate_1.png‘))
heart = pygame.image.load(os.path.join(img_dir, ‘heart.png‘))

def main_menu():
    """
    Generate three screens:
        first screen: Start or quit the game
        Second screen: Instruction and Difficulty selection
        Third screen: Background Story (Press Enter to enter the game)
    """
    global mode, screen
    screen.blit(menu_image1,(0, 0))
    clock.tick(FPS)
    pygame.mixer.music.load(regular_bgm)
    pygame.mixer.music.play(-1)
    menu_layer = 1
    while True:
        #Mouse detection method
        buttons = pygame.mouse.get_pressed()
        x1, y1 = pygame.mouse.get_pos()
        #Start screen
        if menu_layer == 1:
            screen.blit(button_start, (370, 300))
            screen.blit(button_quit, (370, 400))
            if 370 <= x1 <= 610 and 300 <= y1 <= 385: #Hit button start
                if buttons[0]:
                    menu_layer = 2
            elif 370 <= x1 <= 610 and 400 <= y1 <= 490: #Hit button quit
                if buttons[0]:
                    os._exit(0)
            pygame.display.update()
        #Select Difficulty Level
        elif menu_layer == 2:
            screen.blit(menu_image2, (0, 0))
            screen.blit(button_normal, (550, 70))
            screen.blit(button_difficult, (550, 200))
            if 620 <= x1 <= 830 and 110 <= y1 <= 210: #click button normal
                if buttons[0]:
                    mode = ‘normal‘
                    menu_layer = 3
            elif 620 <= x1 <= 830 and 230 <= y1 <= 330: #click button difficult
                if buttons[0]:
                    mode = ‘difficult‘        
                    menu_layer = 3
            pygame.display.update()
        #Backgroud story
        elif menu_layer == 3:
            screen.blit(bginfo, (0, 0))
            if pygame.key.get_pressed()[pygame.K_RETURN]:
                break
            pygame.display.update()
        #Quit method
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                os._exit(0)
    pygame.mixer.music.stop()
    
def draw_hero_health(hp):
    """
    Draw several hearts on the top left corner of the screen to represent the hp of hero
    """
    for i in range(int(hp) + 1):
        if i > 0:
            screen.blit(heart, (35*i, 10))

def draw_boss_health(health):
    """
    Draw a health bar on the top right corner of the screen
    """
    if health > 0: #the coordinates of the health bar
        WIDTH = 100
        HEIGHT = 20
        space = 20
        pygame.draw.rect(screen, pygame.Color("green"), pygame.Rect(width-WIDTH-space, space, WIDTH + 4, HEIGHT))
        pygame.draw.rect(screen, pygame.Color("red"), pygame.Rect(width-WIDTH-space + 2, space+2, WIDTH/boss_health*health, HEIGHT-4))

def gen_enemy(num):
    """
    Generate given number of mob and add to sprite group all_sprites
    """
    for i in range(num): 
        mob_element = Mob()
        all_sprites.add(mob_element)
        mobs.add(mob_element)

def ticktock(limit,ticks,font):   
    """
    show the time before hero die, display at the top of the screen
    """
    text = font.render("Time Remaining    " + str((limit-ticks)//60000) + ":" +
							   str((limit-ticks)//1000%60).zfill(2), True, (0,255,0))
    Rect = text.get_rect()
    Rect = (300, 10)     #position of the time display
    screen.blit(text,Rect)  
    
class animated_sprite(pygame.sprite.Sprite):
    """
    parent class of all animated objects, take a list of images and automaticlly generate a list of
    mirrored images.
    use current_time and animation_time to achieve looping images inside the list
    """
    def __init__(self, images): # Call the parent class (Sprite) constructor
        super().__init__()
        self.images = images
        self.animation_time = 0.01
        self.current_time = 0 #keep track of time
        self.index = 0 #used to keep track of which image is in use
        self.image = images[self.index] #current image of the animation
        self.images_right = self.images #images face right in default
        self.images_left = [] #turn the image to face left and store in a list
        for image in images:
            self.images_left.append(pygame.transform.flip(image, True, False))
        
class Hero(animated_sprite):
    """
    The object player will control, hero is an animated object with ability to 
    move and shoot
    Attributes:
        update: use WASD to move the character on the screen, two lists of left and right images 
            will be automatically switched
        shoot: use key arrows to fire a bullet, maximum speen is one bullet per 0.8 second
    """
    def __init__(self, position, images, speed = 18, shoot_delay = 800):
        super().__init__(images)
        size = (39, 37)  # The size of the images
        self.health = hero_health
        self.rect = pygame.Rect(position, size)
        self.velocity = pygame.math.Vector2(0, 0)
        self.shoot_delay = shoot_delay
        self.last_shot = pygame.time.get_ticks()
        self.bull_image = bull1
        self.speed = speed
    def update(self, dt):
        #Move
        if pygame.key.get_pressed()[pygame.K_a] and self.rect.center[0] > boundaryx[0] - 120:
            self.velocity.x = -self.speed
        elif pygame.key.get_pressed()[pygame.K_d] and self.rect.center[0] < boundaryx[1]:
            self.velocity.x = self.speed
        else:
            self.velocity.x = 0   
        if pygame.key.get_pressed()[pygame.K_s] and self.rect.center[1] <boundaryy[1]:
            self.velocity.y = self.speed
        elif pygame.key.get_pressed()[pygame.K_w] and self.rect.center[1] > boundaryy[0]:
            self.velocity.y = -self.speed
        else:
            self.velocity.y = 0
        #Shoot
        if pygame.key.get_pressed()[pygame.K_UP]:
            self.shoot(1)		
        elif pygame.key.get_pressed()[pygame.K_DOWN]:
            self.shoot(2)		
        elif pygame.key.get_pressed()[pygame.K_LEFT]:
            self.shoot(3)			
        elif pygame.key.get_pressed()[pygame.K_RIGHT]:
            self.shoot(4)
        #Update frame 
        if self.velocity.x > 0:  # Use the right images if sprite is moving right.
            self.images = self.images_right
        elif self.velocity.x < 0: 
            self.images = self.images_left
        #Looping the images in the list
        self.current_time += dt
        if self.current_time >= self.animation_time and self.velocity != (0,0):
            self.current_time = 0
            self.index = (self.index + 1) % len(self.images)
            self.image = self.images[self.index]
        else:
            self.image = self.images[3]
        self.rect.move_ip(*self.velocity)

    def shoot(self,dir):
        now = pygame.time.get_ticks()
        if now - self.last_shot > self.shoot_delay: #set the shoot frequate of bullet 
            self.last_shot = now
            shooting_sound.play()
            bullet = Bullet(self.rect.center[0],self.rect.center[1],self.bull_image,dir)  #create new bullet
            all_sprites.add(bullet)
            bullets.add(bullet)

class Mob(animated_sprite):
    """
    the class of regular enemy, move towards the hero if close enough, otherwise will wander around

    attributes:
        update: animation method (automatically switch left and right images) and move method.
    """
    def __init__(self, size = (40,40)):
        self.images = random.choice([e1images, e2images, e3images])
        super().__init__(self.images)
        self.rect = pygame.Rect((0,0),size)
        self.image = self.images[self.index]
        self.rect.centerx = random.randrange(boundaryx[0],boundaryx[1])
        self.rect.centery = random.randrange(boundaryy[0],boundaryy[1])
        self.speedy = 8   
        self.speedx = 8
        self.speed = 8

    def update(self, dt):
        distanceX = hero.rect.center[0] - self.rect.centerx
        distanceY = hero.rect.center[1] - self.rect.centery
        distance = numpy.sqrt(pow(distanceX,2) + pow(distanceY,2))
        # if enemies are too far away from the hero, they will just wander around
        if distance > 400:
            self.rect.centerx += self.speedx
            self.rect.centery += self.speedy
            if self.speedx > 0:  # Use the right images if sprite is moving right.
                self.images = self.images_right
            elif self.speedx < 0:
                self.images = self.images_left
        # if enemies is close enough to the hero, they will go towards him
        else:
            self.rect.centerx += int(self.speed * distanceX /(distance+0.001))
            self.rect.centery += int(self.speed * distanceY /(distance+0.001))
        
            if distanceX > 0:
                self.images = self.images_right
            else:
                self.images = self.images_left
        # if touch the boundary
        if self.rect.centerx < boundaryx[0] or self.rect.centerx > boundaryx[1]:
            self.speedx = -self.speedx
        if self.rect.centery < boundaryy[0] or self.rect.centery > boundaryy[1]:
            self.speedy = -self.speedy
        #Update frame        
        self.current_time += dt
        if self.current_time >= self.animation_time:
            self.current_time = 0
            self.index = (self.index + 1) % len(self.images)
            self.image = self.images[self.index]

class Boss(animated_sprite):
    """
    The boss will move towards hero, and have a much bigger size than mobs
    Attributes:
        update: animation and move method
    """
    def __init__(self, images, health, speed, position = (600, 270)):
        super().__init__(images)
        size = (50,100)
        self.rect = pygame.Rect(position,size)
        self.health = health
        self.speed = speed

    def update(self, dt):
        distanceX = hero.rect.center[0] - self.rect.centerx
        distanceY = hero.rect.center[1] - self.rect.centery
        distance = numpy.sqrt(pow(distanceX,2) + pow(distanceY,2))
        self.rect.centerx += int(self.speed * distanceX /(distance+0.001))
        self.rect.centery += int(self.speed * distanceY /(distance+0.001))
        #Update frame
        if distanceX > 0:
            self.images = self.images_right
        else:
            self.images = self.images_left
        self.current_time += dt
        if self.current_time >= self.animation_time:
            self.current_time = 0
            self.index = (self.index + 1) % len(self.images)
            self.image = self.images[self.index]

class Bullet(pygame.sprite.Sprite):
    """
    after the hero fire, bullet will fly toward the direction set by the arrow key
    Attributes:
        update: move method
    """
    def __init__(self, x, y, pic, dir):
        pygame.sprite.Sprite.__init__(self)
        self.image = pic #Initialise the image of bullet
        self.speed = 40  #the speed of bullet
        self.rect = self.image.get_rect() 
        #Initialise the direction and original position of bullet
        if dir == 1:   #up
            self.vel = (0, -1) 
            self.rect.centerx = x
            self.rect.bottom = y  
        elif dir == 2: #down
            self.vel = (0, 1) 
            self.rect.centerx = x
            self.rect.top = y  
        elif dir == 3: #left
            self.vel = (-1, 0)
            self.rect.centery = y
            self.rect.right = x  
        elif dir == 4: #right
            self.vel = (1,0) 
            self.rect.centery = y
            self.rect.left = x
            
    def update(self,dt):   #Update the position of bullet
        self.rect.move_ip(int(self.speed * self.vel[0]), int(self.speed * self.vel[1]))

class gate(pygame.sprite.Sprite):
    """
    sprite used to represent teleport gate (sprite adopted for collision detection)
    """
    def __init__(self, pic, x, y):
        super().__init__()
        self.image = pic
        self.rect = self.image.get_rect()
        self.x = x
        self.y = y
    def update(self, dt):
        self.rect.centerx = self.x
        self.rect.bottom = self.y

class treasure(pygame.sprite.Sprite):
    """
    sprite used to represent weapon upgrade and chest that will generate props
    """
    def __init__(self,pic,x,y):
        super().__init__()
        self.image = pic
        self.rect = self.image.get_rect()
        self.rect.centerx = x #position of the props
        self.rect.centery = y 
    
tickfont = pygame.font.Font(None, 40)
boundaryx = (170,920)
boundaryy = (170,510)
mode = ‘normal‘
diesound_play = False


while True:
    level = 1
    main_menu()
    if mode == ‘normal‘:
        hero_health = 5
        boss_health = 5
        #the number of enemy in each level
        l1_enemy = random.randint(4,6)
        l2_enemy = random.randint(5,8)
        l3_enemy = random.randint(1,3)
    elif mode == ‘difficult‘: 
        hero_health = 3
        boss_health = 10
        l1_enemy = random.randint(5,8)
        l2_enemy = random.randint(8,12)
        l3_enemy = random.randint(3,6)
    #load sprites
    gate1 = gate(gatei1, 454, 180)
    gate2 = gate(gatei1, 674, 180)
    gate3 = gate(gatei1, 450, 180)
    fake_gate = gate(gatei1, 10, 300)
    hero = Hero((100, 300), heroimages)
    all_sprites = pygame.sprite.Group(hero)
    bullets = pygame.sprite.Group()
    mobs = pygame.sprite.Group()
    boss1 = Boss(bossimages, boss_health, 10)
    chest1= treasure(chest,600,260)
    chest2 = treasure(weapon,random.randrange(boundaryx[0],boundaryx[1]),random.randrange(boundaryy[0],boundaryy[1]))
    # game start
    background_i = background
    gen_enemy(l1_enemy)
    pygame.mixer.music.play(-1)
    # time limit conditions
    timelimit = (l1_enemy + l2_enemy + l3_enemy + boss1.health + 10) * 2000  #set time limit(millisecond) in the battle levels
    clock_limit = pygame.time.Clock() 
    ticks = 0
    timewarning = 5  # set the play times of time warning music
    
    while True:
        #Quit method
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                os._exit(0)
        #Game Loop
        clock.tick(FPS)
        all_sprites.update(dt)
        screen.blit(background_i, (0, 0))
        draw_hero_health(str(hero.health))
        all_sprites.draw(screen)
        #Display time remaining
        if level < 4 and hero.health != 0:   #count down 
            clock_limit.tick()
            ticks += clock_limit.get_time() 
            ticktock(timelimit, ticks, tickfont) #update the time left
            if timewarning > 0 and (timelimit-ticks)/(timewarning*1000) <= 1:
                time_limit.play()
                timewarning -= 1
            if ticks >= timelimit: #hero die if exceed the time limit
                hero.health = 0   
        #if a bullet hits an enemy
        if len(pygame.sprite.groupcollide(mobs, bullets, True, True)) != 0:
            die_sound.play()
        #if hero get hit by an enemy
        if len(pygame.sprite.spritecollide(hero, mobs, True)) != 0:
            die_sound.play()
            hero.health -= 1
        #if hero died, play sound
        if hero.health <= 0 and not diesound_play:
            boss_entry.play()
            boss_entry.play()
            diesound_play = True
        #restart the game
        if hero.health <= 0 and diesound_play:
            all_sprites.empty()
            screen.blit(die_image, (0, 0))
            level = 1
            pygame.mixer.music.stop()
            if pygame.key.get_pressed()[pygame.K_RETURN]:
                diesound_play = False
                break
        #First level
        if len(mobs.sprites()) < 1 and level == 1:#clear the first level
            all_sprites.add(gate1)
            if pygame.sprite.collide_rect(hero, gate1):#teleport to next level
                bullets.empty()
                background_i = background2
                all_sprites.add(fake_gate)
                hero.rect.center = (10, 280)
                gate1.rect.centerx = 18000
                gate1.kill()
                gen_enemy(l2_enemy)
                level = 2
        #Level 2, opportunity to upgrade weapon after clear the stage
        if len(mobs.sprites()) < 1 and level == 2:
            fake_gate.kill()
            all_sprites.add(gate2)
            screen.blit(warning_img, (285, 230))
            pygame.mixer.music.stop()
            all_sprites.add(chest2)
            if pygame.sprite.collide_rect(hero, chest2):
                screen.blit(warning_img, (285, 230))
                hero.bull_image = bull2
                level_up.play()
                chest2.kill()
                level=2.2
            elif pygame.sprite.collide_rect(hero, gate2):
                level = 2.2
        #Teleport to Boss scene, set background pic and music
        if level == 2.2:
            screen.blit(warning_img, (285, 230))
            if pygame.sprite.collide_rect(hero, gate2):
                chest2.kill()
                gate2.kill()
                bullets.empty()
                background_i = background3
                hero.rect.center = (10, 280)
                all_sprites.add(boss1)
                all_sprites.add(fake_gate)
                gen_enemy(l3_enemy)
                boss_entry.play()
                pygame.mixer.music.load(boss_bgm)
                pygame.mixer.music.play(-1)
                level = 3
        #Level 3, boss scene
        if level == 3:
            draw_boss_health(boss1.health)  
            if pygame.sprite.collide_rect(hero, boss1):
                hero.health = 0
            if len(pygame.sprite.spritecollide(boss1, bullets, True)) != 0:     
                boss1.health -= 1
                boss_entry.play()
            if boss1.health <= 0:
                boss1.kill()
                all_sprites.remove(mobs, bullets)
                mobs.empty()
                fake_gate.kill()
                boss_die.play()
                boss_die.play()
                boss_die.play()
                level = 3.1
        #Teleport to next level
        if level == 3.1:
            all_sprites.add(gate3) 
            if pygame.sprite.collide_rect(hero, gate3):
                gate3.kill()
                background_i = background4
                hero.rect.center = (40,280)
                level = 4
        #Level 4, Treasure Cave
        if level == 4:
             all_sprites.add(chest1) 
             if pygame.sprite.collide_rect(hero,chest1):
                 open_chest.play()
                 chest1.kill()
                 level = 5
                 pygame.time.delay(500)
        #Level 5, final scene of the game
        if level == 5:
            screen.blit(final1,(0,0))
            if pygame.key.get_pressed()[pygame.K_RETURN]:
                pygame.mixer.music.stop()
                break
      
        pygame.display.update()

  

以上是关于Python实现寻宝小游戏的主要内容,如果未能解决你的问题,请参考以下文章

GitHub寻宝机器学习实战python3代码分享

[BZOJ3991][SDOI2015]寻宝游戏

[SDOI2015]寻宝游戏

BZOJ3991 寻宝游戏

寻宝游戏(bzoj 3991)

BZOJ3991[SDOI2015]寻宝游戏 树链的并+set