俄罗斯方块

news2024/11/17 19:47:48

俄罗斯方块简单实现

使用 pygame 模块实现俄罗斯方块的简单实现,这里没有使用pygame 自带的碰撞检测,而是自定义的方法实现边界碰撞和方块间碰撞检测。

代码实现

import random
import pygame
import time
# 初始化游戏
pygame.init()

# 设置游戏窗口大小
WINDOW_WIDTH = 400
WINDOW_HEIGHT = 600
WINDOW_SIZE = (WINDOW_WIDTH, WINDOW_HEIGHT)

# 设置游戏窗口标题
pygame.display.set_caption("俄罗斯方块")

# 设置游戏窗口
screen = pygame.display.set_mode(WINDOW_SIZE)

# 定义方块大小和颜色
BLOCK_SIZE = 20
BLOCK_COLOR = (255, 255, 255)


# 定义方块类
class Block:

    def __init__(self, x, y):
        self.x = x
        self.y = y

    def draw(self, color):
        pygame.draw.rect(screen, color, (self.x, self.y, BLOCK_SIZE, BLOCK_SIZE))


# 定义方块组类
class BlockGroup:
    block_groups = list()
    shape_set = {
        "1": [(0, 1), (0, 2), (0, 3)],
        "一": [(1, 0), (2, 0), (3, 0)],
        "T": [(1, 0), (1, 1), (2, 0)],
        "Z": [(1, 0), (1, 1), (2, 1)],
        "田": [(1, 0), (0, 1), (1, 1)],
        "L": [(0, 1), (0, 2), (1, 2)],
    }
    bottom_boundary_points = set()

    def __init__(self):
        self.blocks = list()
        self.color = (random.randint(0, 255), random.randint(0, 255), random.randint(0, 255))

    def init(self):
        group = self.generate_group()
        self.block_groups.append(group)

    @staticmethod
    def generate_group():
        group = BlockGroup()
        x0 = random.randint(0, WINDOW_WIDTH-2 * BLOCK_SIZE)//BLOCK_SIZE * BLOCK_SIZE
        y0 = -3 * BLOCK_SIZE

        group.blocks.append(Block(x0, y0))
        shape = random.choice(list(group.shape_set.keys()))
        for _x, _y in group.shape_set[shape]:
            # block = Block(random.randint(0, WINDOW_WIDTH - BLOCK_SIZE), i * BLOCK_SIZE)
            x = x0 + _x * BLOCK_SIZE
            y = y0 + _y * BLOCK_SIZE
            group.blocks.append(Block(x, y))
            group.shape = shape

        return group

    def draw(self):
        for group in self.block_groups:
            for block in group.blocks:
                block.draw(group.color)

    def move_down(self):
        if self.collision_detection("bottom"):
            group = self.generate_group()
            BlockGroup.block_groups.append(group)
            return
        for block in self.block_groups[-1].blocks:
            block.y += BLOCK_SIZE

    def move_left(self):
        if self.collision_detection("left"):
            return
        for block in self.block_groups[-1].blocks:
            block.x -= BLOCK_SIZE

    def move_right(self):
        if self.collision_detection("right"):
            return
        for block in self.block_groups[-1].blocks:
            block.x += BLOCK_SIZE

    def collision_detection(self, move_direction):
        # 移动方向上的偏移量
        offset = {
            "left": {"x0": -1*BLOCK_SIZE, "y0": 0},
            "right": {"x0": 1*BLOCK_SIZE, "y0": 0},
            "bottom": {"x0": 0, "y0": 1*BLOCK_SIZE}
        }
        block_collision = False
        boundary_collision = False
        # 获取当前活动组每个方块的坐标像素值
        for block in self.block_groups[-1].blocks:
            after_offset_pos = (block.x + offset[move_direction]["x0"], block.y + offset[move_direction]["y0"])
            print(after_offset_pos, self.bottom_boundary_points)
            if after_offset_pos in self.bottom_boundary_points:
                print(f"预测到方块碰撞点", after_offset_pos)
                block_collision = True

            if after_offset_pos[0] < 0 or after_offset_pos[0] >= WINDOW_WIDTH or after_offset_pos[1] >= WINDOW_HEIGHT:
                print(f"预测到第边界碰撞点", after_offset_pos)
                boundary_collision = True

            # 方块下移时,发生方块碰撞
            if (block_collision or after_offset_pos[1] >= WINDOW_HEIGHT) and move_direction == "bottom":
                shape = self.block_groups[-1].shape
                print(self.blocks)
                start_block = self.block_groups[-1].blocks[0]

                self.bottom_boundary_points.update({(start_block.x, start_block.y)})
                for _x, _y in self.block_groups[-1].shape_set[shape]:
                    self.bottom_boundary_points.update({(start_block.x + _x * BLOCK_SIZE, start_block.y + _y * BLOCK_SIZE)})

                print(len(self.bottom_boundary_points), self.bottom_boundary_points)
                # time.sleep(3)
                # print(f"{self}检测到碰撞, 坐标{(block.x, block.y)}")

        return block_collision or boundary_collision


# 定义游戏主循环
def main():
    clock = pygame.time.Clock()
    # 创建方块组
    block_groups = BlockGroup()
    block_groups.init()

    # 游戏循环
    while True:
        clock.tick(3)

        # 处理事件
        for event in pygame.event.get():
            if event.type == pygame.QUIT:
                pygame.quit()
                exit()
            if event.type == pygame.KEYDOWN:
                if event.key == pygame.K_LEFT:
                    block_groups.move_left()
                elif event.key == pygame.K_RIGHT:
                    block_groups.move_right()
        # 绘制背景
        screen.fill((0, 0, 0))
        # 移动方块组
        block_groups.move_down()
        # 绘制方块组
        block_groups.draw()
        # 更新屏幕
        pygame.display.update()


# 启动游戏
if __name__ == '__main__':
    main()

效果展示
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/837303.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

libcurl开源的、跨平台的网络传输库,用于在程序中实现数据传输功能的编译

文章目录 前言1、libcurl关键特点和功能2、没有使用openssl以及libssh2编译libcurl的文件和使用openssl和libssh2编译3、libcurl网络库的下载4、libcurl网络库的编译4.1、直接使用cmake编译&#xff0c;不使用 OpenSSL 和 libssh2库编译的出来的libcurl库4.2、使用 OpenSSL 和 …

windows之关闭占用端口的进程

1. 查询端口占用进程&#xff0c;这里以8001为例 netstat -ano | findstr 8001 2.结束进程 taskkill -pid 37072 -f

解释器模式-自定义语言的实现

有时&#xff0c;我们希望输入一串字符串&#xff0c;然后计算机能够按照预先定义的文法规则来对这个字符串进行解释&#xff0c;从而实现相应的功能。 例如&#xff0c;我们想实现简单的加减法接收器&#xff0c;只需输入一个表达式&#xff0c;它就能计算出表达式结果。比如…

SpringBoot统一功能处理(拦截器)

1.用户登录权限校验 1.1自定义拦截器 写一个类去实现HandlerInterceptor接口表示当前类是一个拦截器,再重写HandlerInterceptor接口中的方法,preHandle为在方法执行前拦截,postHandle为方法执行中拦截,afterCompletion为方法执行中拦截.需要在什么时候拦截就重写什么方法 Co…

微信公众号自动登录方案

基于微信公众号登录 借助微信公众号来试实现社区登录。登录的时候展示的是一个二维码&#xff0c;但实际上的操作是借助这个展示的过程&#xff0c;和前端构建一个半长连接&#xff0c;当用户向公众号发送验证码之后&#xff0c;微信公众平台会将用户发送的消息转发给服务器&a…

Unity进阶--通过PhotonServer实现联网登录注册功能(服务器端)--PhotonServer(二)

文章目录 Unity进阶--通过PhotonServer实现联网登录注册功能(服务器端)--PhotonServer(二)服务器端大体结构图BLL层&#xff08;控制层&#xff09;DAL层&#xff08;数据控制层&#xff09;模型层DLC 服务器配置类 发送消息类 以及消息类 Unity进阶–通过PhotonServer实现联网…

Gartner发布《2023年全球RPA魔力象限》:90%RPA厂商,将提供生成式AI自动化

8月3日&#xff0c;全球著名咨询调查机构Gartner发布了《2023年全球RPA魔力象限》&#xff0c;通过产品能力、技术创新、市场影响力等维度&#xff0c;对全球16家卓越RPA厂商进行了深度评估。 弘玑Cyclone&#xff08;Cyclone Robotics&#xff09;、来也&#xff08;Laiye&am…

【蓝图】p47下车减速功能

p47下车减速功能 p47下车减速功能加速功能下车减速功能 p47下车减速功能 加速功能 上图是ue自带的加速功能&#xff0c;检测到按w时输入轴会传1给设置油门输入&#xff0c;就会加速 所以&#xff0c;减速也可以通过蓝图反方向制作 下车减速功能 打开Sedan蓝图类的上下车图表…

day51-Mybatis-Plus/代码生成器

1.Mybatis-Plus 定义&#xff1a;是一个Mybatis的增强工具&#xff0c;只在Mybatis基础上增强不做改变&#xff0c;简化开发&#xff0c;提升效率 2.MP实战 2.1 创建springboot工程&#xff0c;勾选web&#xff0c;引入依赖 <dependency> <groupId>mysql<…

人工智能可解释性分析导论(初稿)

目录 思维导图 1.黑箱所带来的问题 2.从应用面论述为什么要进行可解释性分析 2.1可解释性分析指什么 2.2可解释性分析结合人工智能应用实例 2.3 可解释性分析的脑回路&#xff08;以可视化为例如何&#xff09; 3.如何研究可解释性分析 3.1使用好解释的模型 3.2传统机器学…

antDv table组件滚动截图方法的实现

在开发中经常遇到table内容过多产生滚动的场景&#xff0c;正常情况下不产生滚动进行截图就很好实现&#xff0c;一旦产生滚动就会变得有点棘手。 下面分两种场景阐述解决的方法过程 场景一&#xff1a;右侧不固定列的情况 场景二&#xff1a;右侧固定列的情况 场景一 打开…

理解树的结构

树的重要性 二分查找算法、几种核心的排序算法以及图算法都与树有非常密切的关系。有句话锁&#xff0c;“没学会树&#xff0c;算法相当于白学”&#xff0c;可见&#xff0c;树在算法中的地位。 树的考察方面 层次遍历以及拓展问题 前后序遍历与拓展问题 中序遍历与搜索树问…

数据结构入门指南:带头双向循环链表

目录 文章目录 前言 1.结构与优势 2.链表实现 2.1 定义链表 2.2 创建头节点 2.3 尾插 2.4 输出链表 2.5 尾删 2.6 头插 2.7头删 2.8 节点个数 2.9 查找 2.10 位置插入 2.11 位置删除 2.12 销毁链表 3. 源码 总结 前言 链表一共有8种结构&#xff0c;但最常用的就是无头单…

Docker网络模型使用详解(2)Docker网络模式

安装Docker时会自动创建3个网络&#xff0c;可以使用docker network ls命令列出这些网络。 [rootlocalhost ~]# docker network ls NETWORK ID NAME DRIVER SCOPE ebcfad6f4255 bridge bridge local b881c67f8813 compose_lnmp_lnmp…

Vue2升级Vue3报错:Right-hand side of ‘instanceof‘ is not an object

属性prop设置多类型报错&#xff1a; Vue2 写法&#xff1a;支持用竖线隔开。Vue2 Prop expandLevel: {type: Number | String,default: 1, }, Vue3 写法&#xff1a;改为数组&#xff0c;不支持竖线隔开。Vue3 Prop expandLevel: {type: [Number, String],default: 1, }

二次元美少女【InsCode Stable Diffusion 美图活动一期】

目录 Stable Diffusion 模型在线使用地址 一、背景介绍 二、模板介绍&#xff1a; 三、操作步骤 1.在线运行地址 2.进入在线运行网址&#xff0c;并点击运行及使用 3.购买GPU并创建项目 4.打开工作台并选择算力资源 5.点击下图中所示框框 6.进入Stable Diffusion WebU…

VR内容研发公司 | VR流感病毒实验虚拟现实课件

由广州华锐互动开发的《VR流感病毒实验虚拟现实课件》是一种新型的教学模式&#xff0c;可以为学生提供更加真实和直观的流感病毒分离鉴定实验操作体验&#xff0c;从而提高学生的实验技能和工作效率。 《VR流感病毒实验虚拟现实课件》涉及了生物安全二级实验室(BSL-2)和流感病…

.jpeg转.jpg,cv2.resize()

from PIL import Image import os # 定义原文件夹路径和目标文件夹路径 source_folder "path/to/source/folder" target_folder "path/to/target/folder" # 遍历原文件夹中的所有图片文件 for filename in os.listdir(source_folder): if fil…

【iOS安全】安装Filza || 安装Flexdecrypt

&#xff08;成功&#xff09;使用Cydia安装Filza 直接在Cydia里搜索filza&#xff0c;安装“Filza File Manager” 使用Filza安装flexdecrypt 参考&#xff1a; https://github.com/JohnCoates/flexdecrypt 下载flexdecrypt.deb到手机&#xff1a; https://github.com/JohnC…

猎聘:2023届高校毕业生就业数据报告(附下载

关于报告的所有内容&#xff0c;公众【营销人星球】获取下载查看 核心观点 较 2022 届应届生职位同比增长较明显的TOP5 一级行业为能源/化工/环保、医疗健康、汽车、机械/制造、电子/通信/半导体&#xff0c;其中能源/化工/环保同比增长为 42.30%&#xff0c;增速最高.在全世…