小昆虫接金币闯关

"""
小昆虫接金币闯关,这是用Python Arcade街机模块制作的一个多关卡小游戏。游戏虽然简单,但是它演示一个多关卡房间游戏的基本框架。
本游戏首先需要加载tmx格式的地图,它是由Tiled软件制作的。共有3个关卡,小昆虫接完所有道具后自动进入下一关卡。
下一个版本实现昆虫的造型能左右分开,再下一个版本安排一扇门,当接完所有道具后门就显示出来。小昆虫碰到了门才能进入下一关。
再下一个版本实现可移动的敌人,小昆虫碰到敌人游戏就会失败。再下一个版本加一个封面和结尾,剧终。

"""

import arcade
import time

SPRITE_SCALING = 1                    # 定义缩放比例
SCREEN_WIDTH = 960                    # 定义所渲染的屏幕宽度 
SCREEN_HEIGHT = 768                   # 定义所渲染的屏幕高度 
SCREEN_TITLE = "Python平铺地图多关卡示例_ "
SPRITE_PIXEL_SIZE = 64               # 地图方块尺寸 

# 定义物理常数 
MOVEMENT_SPEED = 5
JUMP_SPEED = 23
GRAVITY = 1.1

class MyGame(arcade.Window):
    """ 继承自窗口的游戏类. """

    def __init__(self):
        """
        初始化方法
        """
        super().__init__(SCREEN_WIDTH, SCREEN_HEIGHT, SCREEN_TITLE)
        
        # 定义角色列表
        self.wall_list = None
        self.player_list = None
        self.coin_list = None

        # 定义玩家相关变量
        self.score = 0
        self.player_sprite = None

        self.physics_engine = None         
        self.game_over = False
        self.last_time = None       # 用来算fps的上次时间变量
        self.frame_count = 0        # 帧计数器
        self.fps_message = None     # fps字符串

        self.level = 1              # 开始关卡号
        self.max_level = 3          # 关卡数量

    def setup(self):
        """ 设置与初始化变量的值. """

        # 角色列表初始化
        self.player_list = arcade.SpriteList()
        self.coin_list = arcade.SpriteList()

        # 玩家操作的角色实例化,这里是一只昆虫
        self.player_sprite = arcade.Sprite("images/lixingqiu.png", SPRITE_SCALING*0.5)

        # 给昆虫设定坐标
        self.player_sprite.center_x = 128
        self.player_sprite.center_y = 64
        self.player_list.append(self.player_sprite)

        self.load_level(self.level) # 加载第一关

        self.game_over = False

    def load_level(self, level):
        # 读取第level关的地图
        my_map = arcade.read_tiled_map(f"lxq_{level}.tmx", SPRITE_SCALING)
        
        # 读取不可移动的平台数据阵列'ground'是一图层的名称
        map_array = my_map.layers_int_data['ground']        
          
        # 从墙生成地图列表
        self.wall_list = arcade.generate_sprites(my_map, 'ground', SPRITE_SCALING)
        self.wall_list.move(SPRITE_PIXEL_SIZE ,0)
        
        # 加载平台型的物理引擎
        self.physics_engine = arcade.PhysicsEnginePlatformer(self.player_sprite,
                                                             self.wall_list,
                                                             gravity_constant=GRAVITY)        
        # 读取金币阵列,added by lixingqiu,picked是地图中的一个层,这个层里是可拾取的道具。
        coins_array = my_map.layers_int_data['picked']
        # 生成金币列表
        self.coin_list = arcade.generate_sprites(my_map, 'picked', SPRITE_SCALING)
        self.coin_list.move(SPRITE_PIXEL_SIZE ,0)        

        # 其它情况,如有背景颜色属性,则用地图的背景颜色
        if my_map.backgroundcolor:
            arcade.set_background_color(my_map.backgroundcolor)      

    def on_draw(self):
        """
        渲染屏幕
        """
        self.frame_count += 1      # 帧计数器

        # 开始渲染屏幕
        arcade.start_render()

        # 画所有的角色
        self.player_list.draw()
        self.wall_list.draw()
        self.coin_list.draw()
        # 计算FPS,帧率
        if self.last_time and self.frame_count % 60 == 0:      # 约每秒画一次
            fps = 1.0 / (time.time() - self.last_time) * 60
            self.fps_message = f"FPS: {fps:5.0f}"

        if self.fps_message:
            arcade.draw_text(self.fps_message, 10,40, arcade.color.BLACK, 14)

        if self.frame_count % 60 == 0:
            self.last_time = time.time()
 
        if self.game_over:
            arcade.draw_text("游戏结束,游戏制作:李兴球,powered by lixingqiu", 100,  200, arcade.color.BLACK, 20,font_name='simkai')

    def on_key_press(self, key, modifiers):
        """
        当按键时调用此方法
        """
        if key == arcade.key.UP:
            if self.physics_engine.can_jump():          # 这里是检测有没有在地面
                self.player_sprite.change_y = JUMP_SPEED
        elif key == arcade.key.LEFT:
            self.player_sprite.change_x = -MOVEMENT_SPEED
        elif key == arcade.key.RIGHT:
            self.player_sprite.change_x = MOVEMENT_SPEED

    def on_key_release(self, key, modifiers):
        """
        当松开键时调用此方法
        """
        if key == arcade.key.LEFT or key == arcade.key.RIGHT:
            self.player_sprite.change_x = 0

    def update(self, delta_time):
        """ 移动与游戏的逻辑,这个方法不断地执行,如果本关金币数加载完了,那么此关结束 """

        if len(self.coin_list)==0:           # 如果金币都接完成,准备进入下一关
            if self.level < self.max_level:
                self.level += 1
                self.load_level(self.level)
                self.player_sprite.center_x = 128  # 角色重新摆位
                self.player_sprite.center_y = 64
            else:
                self.game_over = True

        # 角色更新
        if not self.game_over:
            self.physics_engine.update()
        # 角色与金币的碰撞检测,返回列表,列表不为空则把里面的金币删除并加分
        coins_hit = arcade.check_for_collision_with_list(self.player_sprite, self.coin_list)
        for coin in coins_hit:
            coin.kill()
            self.score += 1

def main():
    window = MyGame()
    window.setup()
    arcade.run()


if __name__ == "__main__":
    main()