【Python游戏开发】扫雷游戏demo

news2024/11/18 19:46:13

准备步骤

项目开发使用【Mu 编辑器】

1.使用Mu编辑器创建新项目

在这里插入图片描述

2.点击【保存】,将项目保存到指定路径中

在这里插入图片描述

3.点击【图片】,会打开项目图片存储位置,将所需图片复制粘贴至该文件夹中

在这里插入图片描述

游戏编写

1.设置游戏场景

ROWS = 15				# 设置行数
COLS = 15				# 设置列数
SIZE = 25				# 方块尺寸
WIDTH = SIZE * COLS		# 游戏场景宽度
HEIGHT = SIZE * ROWS	# 游戏场景高度

2.生成方块阵列

blocks = []
# 初始化方块列表
for i in range(ROWS):
    for j in range(COLS):
        block = Actor("minesweep_block")	# 方块初始显示背面状态
        block.left = j * SIZE				# 方块对应横坐标位置
        block.top = i * SIZE				# 方块对应纵坐标位置
        block.isbomb = False			# 地雷标识,False表示其不是地雷
        blocks.append(block)
#  绘制图像
def draw():
    for block in blocks:
        block.draw()

点击【开始】运行游戏,可见窗口显示下方方块矩阵

在这里插入图片描述

3.矩阵埋雷

import random

BOMBS = 20            						# 地雷数量
# 使用random.shuffle()方法打乱地雷列表,改变每块方块对应的索引值
random.shuffle(blocks)      		
# 埋设地雷,因为矩阵方块对应的索引值被打乱了,所以直接设置前“BOMBS”个方块为地雷即可
for i in range(BOMBS):
    blocks[i].isbomb = True					# 设置其为地雷
    blocks[i].image = "minesweep_bomb"		# 顺便将图片更换为地雷,方便测试

点击【开始】运行游戏,可见原来的方块矩阵中出现了地雷,而且地雷的位置是随机的

在这里插入图片描述

4.实现右键插旗操作

# Pgzero提供的on_mouse_down函数,用于处理鼠标的单机操作    
def on_mouse_down(pos, button):
    for block in blocks:
        # 判断点击矩阵中的哪个方块
        if block.collidepoint(pos):
            # 判断点击操作是否为鼠标右键
            if button == mouse.RIGHT:
                # 处理右键插旗操作
                set_flag(block)
                
# 右键插旗操作           
def set_flag(block):
    # 判断是否已标记插旗,False:未标记,则标记为插旗状态,True:已标记,则取消标记
    if not block.isflag:
        block.image = "minesweep_flag"		# 更改为旗子图标
        block.isflag = True
    else:
        block.image = "minesweep_block"		# 更改为普通图标
        block.isflag = False

注:需先在初始化地雷列表代码中定义 block.isflag = False,否则会报AttributeError: 'Actor' object has no attribute 'isflag'的错误

# 初始化地雷列表
for i in range(ROWS):
    for j in range(COLS):
        block = Actor("minesweep_block")	
        block.left = j * SIZE			
        block.top = i * SIZE				
        blocks[i].isbomb = False
        block.isflag = False			 # 定义插旗标识
        blocks.append(block)

点击【开始】运行游戏,右键点击方块,会使方块变为插旗状态,再次点击,会便会普通状态

在这里插入图片描述

5.实现左键翻开操作

def on_mouse_down(pos, button):
    for block in blocks:
        if block.collidepoint(pos):
            if button == mouse.RIGHT:
                set_flag(block)
            # 判断点击操作是否为鼠标左键,且点击方块被插旗
            elif button == mouse.LEFT and not block.isflag:
                # 处理左键翻开操作
                open_block(block)
                
# 左键翻开操作
def open_block(bk):
    bk.isopen = True					# 标识为已翻开
    bombnum = get_bomb_number(bk)		# 获取其周围地雷数量
    bk.image = "minesweep_number" + str(bombnum)
    # 如果周围地雷数量不为0,则返回
    if bombnum != 0:
        return
    # 如果周围地雷数量为0,则翻开周围未翻开的方块
    for block in get_neighbours(bk):
        if not block.isopen:
            # 递归翻开方块九宫格区域未翻开的方块
            open_block(block)

# 获取翻开方块周围9宫格区域存在的未翻开的方块
def get_neighbours(bk):
    nblocks = []
    for block in blocks:
        # 如果方块已翻开,则不插入列表中
        if block.isopen:
            continue
        # 如果方块在当前点击方块的九宫格区域内,则插入列表中
        if block.x == bk.x - SIZE and block.y == bk.y \
            or block.x == bk.x + SIZE and block.y == bk.y \
            or block.x == bk.x and block.y == bk.y - SIZE \
            or block.x == bk.x and block.y == bk.y + SIZE \
            or block.x == bk.x - SIZE and block.y == bk.y - SIZE \
            or block.x == bk.x + SIZE and block.y == bk.y - SIZE \
            or block.x == bk.x - SIZE and block.y == bk.y + SIZE \
            or block.x == bk.x + SIZE and block.y == bk.y + SIZE :
            nblocks.append(block)
    return nblocks

# 计算翻开方块周围9宫格区域的地雷数
def get_bomb_number(bk):
    num = 0
    for block in get_neighbours(bk):
        if block.isbomb:
            num += 1
    return num
   `block.x == bk.x - SIZE and block.y == bk.y`					# 该方块在当前点击方块左方,即九宫格的4位置
   `block.x == bk.x + SIZE and block.y == bk.y`					# 该方块在当前点击方块右方,即九宫格的6位置
   `block.x == bk.x and block.y == bk.y - SIZE`					# 该方块在当前点击方块下方,即九宫格的2位置
   `block.x == bk.x and block.y == bk.y + SIZE`					# 该方块在当前点击方块上方,即九宫格的8位置
   `block.x == bk.x - SIZE and block.y == bk.y - SIZE`	  # 该方块在当前点击方块左下方,即九宫格的1位置
   `block.x == bk.x + SIZE and block.y == bk.y - SIZE`	  # 该方块在当前点击方块右下方,即九宫格的3位置
   `block.x == bk.x - SIZE and block.y == bk.y + SIZE`	  # 该方块在当前点击方块左上方,即九宫格的7位置
   `block.x == bk.x + SIZE and block.y == bk.y + SIZE `	  # 该方块在当前点击方块右上方,即九宫格的9位置

注:需先在初始化地雷列表代码中定义 block.isbomb= False,否则会报AttributeError: 'Actor' object has no attribute 'isbomb'的错误

# 初始化地雷列表
for i in range(ROWS):
    for j in range(COLS):
        block = Actor("minesweep_block")	
        block.left = j * SIZE			
        block.top = i * SIZE				
        blocks[i].isbomb = False
        block.isflag = False	
        block.isopen = False		     # 定义翻开标识
        blocks.append(block)

到这一步后,运行游戏,点击非地雷方块,若九宫格内都无地雷,会全部展开,否则,会告知其周围有几个地雷

在这里插入图片描述

6.判断点击地雷

failed = False		# 游戏失败标识
def on_mouse_down(pos, button):
    for block in blocks:
        if block.collidepoint(pos):
            if button == mouse.RIGHT:
                set_flag(block)
            elif button == mouse.LEFT and not block.isflag:
                # 判断点击的方块是否为地雷块
                if block.isbomb:
                    blow_up()
                else:
                    open_block(block)
                    
# 翻开地雷,判断游戏失败
def blow_up():
    global failed
    # 将failed改为True,代表游戏失败
    failed = True
        
def draw():
    for block in blocks:
        block.draw()
    # 游戏失败判定
    if failed:
        screen.draw.text("Failed",center=(WIDTH // 2, HEIGHT // 2),fontsize = 100, color = "red")

运行游戏,直接点击地雷会出现失败提示

在这里插入图片描述

7.游戏胜利判定

finished = False			# 游戏胜利标识
# 使用update函数实时检测游戏是否已达成结束条件
def update():
    global finished
    # 游戏已胜利或失败,则退出,游戏继续
    if finished or failed:
        return
    # 遍历所有方块
    for block in blocks:
        # 如果存在不是地雷、且未翻开的方块,则退出,游戏继续
        if not block.isbomb and not block.isopen:
            return
    # 所有非地雷的方块都已被翻开,则游戏结束
    finished = True
def draw():
    for block in blocks:
        block.draw()
    if failed:
        screen.draw.text("Failed",center=(WIDTH // 2, HEIGHT // 2),fontsize = 100, color = "red")
    # 游戏胜利提示
    if finished:
        screen.draw.text("finished",center=(WIDTH // 2, HEIGHT // 2),fontsize = 100, color = "red")

运行游戏,翻开所有非地雷的方块后

在这里插入图片描述

完整游戏代码

import random

ROWS = 15				                    # 设置行数
COLS = 15				                    # 设置列数
SIZE = 25				                    # 方块尺寸
WIDTH = SIZE * COLS		                    # 游戏场景宽度
HEIGHT = SIZE * ROWS	                    # 游戏场景高度
failed = False		                        # 游戏失败标识
finished = False			                # 游戏胜利标识
blocks = []                                 # 方块列表
BOMBS = 20            		                # 地雷数量

# 初始化方块列表
for i in range(ROWS):
    for j in range(COLS):
        block = Actor("minesweep_block")	# 方块初始显示背面状态
        block.left = j * SIZE				# 方块对应横坐标位置
        block.top = i * SIZE				# 方块对应纵坐标位置
        block.isbomb = False			    # 地雷标识,False表示其不是地雷
        block.isflag = False			    # 定义插旗标识
        block.isopen = False		        # 定义翻开标识
        blocks.append(block)
        
random.shuffle(blocks)                      # 打乱方块列表
# 埋设地雷
for i in range(BOMBS):
    blocks[i].isbomb = True					# 设置其为地雷
    # blocks[i].image = "minesweep_bomb"	# 顺便将图片更换为地雷,方便测试

# 处理鼠标事件
def on_mouse_down(pos, button):
    # 游戏已结束,不处理点击事件
    if finished or failed:
        return
    for block in blocks:
        # 判断点击矩阵中的哪个方块
        if block.collidepoint(pos) and not block.isopen:
            # 鼠标右键点击
            if button == mouse.RIGHT:
                # 处理右键插旗操作
                set_flag(block)
            # 鼠标左键点击
            elif button == mouse.LEFT and not block.isflag:
                # 点击地雷块
                if block.isbomb:
                    blow_up()
                # 点击普通方块
                else:
                    open_block(block)
                
# 右键插旗操作           
def set_flag(block):
    # 判断是否已标记插旗,False:未标记,则标记为插旗状态,True:已标记,则取消标记
    if not block.isflag:
        block.image = "minesweep_flag"		# 更改为旗子图标
        block.isflag = True
    else:
        block.image = "minesweep_block"		# 更改为普通图标
        block.isflag = False
    
# 左键翻开操作
def open_block(bk):
    bk.isopen = True						# 标识为已翻开
    bombnum = get_bomb_number(bk)			# 获取其周围地雷数量
    bk.image = "minesweep_number" + str(bombnum)
    # 如果周围地雷数量不为0,则返回
    if bombnum != 0:
        return
    # 如果周围地雷数量为0,则翻开周围未翻开的方块
    for block in get_neighbours(bk):
        if not block.isopen:
            # 递归翻开方块九宫格区域未翻开的方块
            open_block(block)

# 获取翻开方块周围9宫格区域存在的未翻开的方块
def get_neighbours(bk):
    nblocks = []
    for block in blocks:
        # 如果方块已翻开,则不插入列表中
        if block.isopen:
            continue
        # 如果方块在当前点击方块的九宫格区域内,则插入列表中
        if block.x == bk.x - SIZE and block.y == bk.y \
            or block.x == bk.x + SIZE and block.y == bk.y \
            or block.x == bk.x and block.y == bk.y - SIZE \
            or block.x == bk.x and block.y == bk.y + SIZE \
            or block.x == bk.x - SIZE and block.y == bk.y - SIZE \
            or block.x == bk.x + SIZE and block.y == bk.y - SIZE \
            or block.x == bk.x - SIZE and block.y == bk.y + SIZE \
            or block.x == bk.x + SIZE and block.y == bk.y + SIZE :		
            nblocks.append(block)
    return nblocks

# 计算翻开方块周围9宫格区域的地雷数
def get_bomb_number(bk):
    num = 0
    # 遍历九宫格区域方块,计算地雷数量
    for block in get_neighbours(bk):
        if block.isbomb:
            num += 1
    return num
  
# 翻开地雷,判断游戏失败
def blow_up():
    global failed
    # 将failed改为True,代表游戏失败
    failed = True
 
# 使用update函数实时检测游戏是否已达成结束条件
def update():
    global finished
    # 游戏已胜利或失败,则退出,游戏继续
    if finished or failed:
        return
    # 遍历所有方块
    for block in blocks:
        # 如果存在不是地雷、且未翻开的方块,则退出,游戏继续
        if not block.isbomb and not block.isopen:
            return
    # 所有非地雷的方块都已被翻开,游戏胜利
    finished = True
    
#  绘制图像
def draw():
    for block in blocks:
        block.draw()
    # 游戏失败判定
    if failed:
        screen.draw.text("Failed",center=(WIDTH // 2, HEIGHT // 2),fontsize = 100, color = "red")
    # 游戏胜利提示
    if finished:
        screen.draw.text("finished",center=(WIDTH // 2, HEIGHT // 2),fontsize = 100, color = "red")

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

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

相关文章

【面试题】软件测试实习(含答案)

软件测试实习常见面试题,主要是功能测试相关的基础问题 目录 一、软件测试基础 1、介绍一下你最近的项目,以及工作职责 2、软件项目的测试流程? 3、黑盒测试与白盒测试的区别? 4、黑盒测试常见的设计方法?怎么理解等价类方法和边界值方法 1&…

GO语言环境搭建和字符说明

Go的三种安装方式 Go有多种安装方式,你可以选择自己喜欢的。这里我们介绍三种最常见的安装方式: 1.Go源码安装:这是一种标准的软件安装方式。对于经常使用Unix类系统的用户,尤其对于开发者来说,从源码安装是最方便而…

传奇架设教程:传奇登录器公告窗口如何设置?link.htm网页文件制作教程

今天给大家分享传奇登录器公告窗口的设置教程: 我们玩服时可以看到很多服的登录器公告看上去都很高大上,并且窗口比例也非常的合适,但是自己架设起来并没有看到窗口公告,这是为什么呢? 这是因为你的列表没有设置公告…

计算机视觉中的2D变换:用Python让图像‘动’起来

嘿,小伙伴们!今天我们要聊的是计算机视觉中的2D变换——那些能让图像‘动’起来的技术。无论你是初学者还是资深开发者,这篇文章都将带你深入了解2D变换,并通过Python代码实战演示。准备好了吗?让我们开始吧&#xff0…

Humans or LLMs as the Judge? A Study on Judgement Bias

文章目录 题目摘要引言相关作品论法官的偏见实验方案结果与讨论欺骗LLM法官结论 题目 人类还是LLMs作为裁判?判断偏差研究 论文地址:https://arxiv.org/pdf/2402.10669 摘要 采用人类和大型语言模型(LLM)作为评估LLM性能的评判者(也称为人类和LLM-as-a…

多车合作自动驾驶框架CoDrivingLLM:基于大语言模型驱动的决策框架

Abstract 目前,全球范围内已开始对联网自动驾驶汽车(CAVs)进行道路测试,但它们在复杂场景中的安全性和效率表现仍不令人满意。合作驾驶利用CAVs的连接能力,通过协同作用超越个体表现,使其成为在复杂场景中…

MySQL篇(主从复制/读写分离/分库分表)(持续更新迭代)

目录 讲解一:主从复制 一、何为主从复制? 二、主从复制的优点 三、主从复制原理 四、主从复制环境搭建 1. 准备 2. 主库配置 2.1. 修改配置文件 /etc/my.cnf 2.2. 重启MySQL服务器 2.3. 授予主从复制权限 2.4. 查看二进制日志坐标 3. 从库配置…

ubuntu 安装harbor

#安装包 wget https://github.com/goharbor/harbor/releases/download/v2.10.3/harbor-offline-installer-v2.10.3.tgz wget https://github.com/goharbor/harbor/releases/download/v2.10.3/harbor-offline-installer-v2.10.3.tgz.asc#导入签名公钥 gpg --keyserver hkps://ke…

EDM平台大比拼 用户体验与营销效果双重测评

本文评测了ZohoCampaigns、Mailchimp、Sendinblue、AWeber四款EDM平台,分别适合中小企业、多平台集成、多功能集成、初学者等需求。建议企业根据自身规模、技术水平和功能需求选择最适合的平台。 一、Zoho Campaigns 功能概述 Zoho Campaigns是Zoho旗下的一款专注…

每日OJ题_牛客_拼三角_枚举/DFS_C++_Java

目录 牛客_拼三角_枚举/DFS 题目解析 C代码1 C代码2 Java代码 牛客_拼三角_枚举/DFS 拼三角_枚举/DFS 题目解析 简单枚举&#xff0c;不过有很多种枚举方法&#xff0c;这里直接用简单粗暴的枚举方式。 C代码1 #include <iostream> #include <algorithm> …

探索RAG、AI Agents和Agentic RAG的架构、应用程序和主要区别

人工智能催生了能够执行各种任务的强大模型。该领域最具影响力的两项进步是检索增强生成 (RAG) 和Agents&#xff0c;它们在改进AI驱动的应用程序中发挥着独特的作用。然而&#xff0c;新兴的Agentic RAG概念提出了一种利用两种系统优势的混合模型。Agentic RAG是什么&#xff…

UML中类和类之间关系和各种图

文章目录 1.UML概述2.UML建模工具3.常见UML图_类图4.类图之类和类之间的关系5.用例图1.用例图是什么2.用例图所包含的元素3.用例图所包含的关系 6.状态图1.状态图是什么&#xff1f;2.状态图所包含的元素 7.活动图1.活动图的元素 8.时序图1.时序图是什么&#xff1f;2.时序图的…

混合密码系统——用对称密钥提高速度,用公钥密码保护会话密钥

混合密码系统&#xff08;Hybrid Cryptosystem&#xff09;是一种结合了多种密码学技术和算法的加密方案&#xff0c;旨在充分利用不同密码算法的优势&#xff0c;以提供更强大的安全性、更高的效率或更好的功能特性。以下是对混合密码系统的详细解释&#xff1a; 组成要素 对…

操作系统与进程

1.操作系统 操作系统是计算机中的一个重要软件&#xff0c;它是一个专门进行管理的软件。操作系统可以通过驱动程序来间接管理外部硬件&#xff0c;也可以为计算机中的程序提供一个稳定的运行环境&#xff0c;从而来方便管理各种程序的运行&#xff0c;让程序之间的运行互不影…

用于探索和测试API的开源IDE工具-Bruno

1、前言 在进行软件开发与测试过程中&#xff0c;无论是开发人员还是测试人员&#xff0c;都会或多或少地进行接口调试与接口测试。尤其针对那种测试流程规范性很高的项目&#xff0c;测试人员进行接口测试是不可或缺的一部分。而这其中&#xff0c;大多数都会使用 Postman 作…

.Net 基于IIS部署blazor webassembly或WebApi

1.安装IIS(若安装&#xff0c;请忽略) 选择:控制面板–>程序–>程序和功能 选择:启动或关闭Windows功能&#xff0c;勾选相关项&#xff0c;再点击确定即可。 2.安装Hosting Bundle 以.net6为例&#xff0c;点击连接https://dotnet.microsoft.com/en-us/download/dot…

免费的录屏软件有哪些?可以试试这4款。

录屏软件已经被用于很多的领域和场景当中&#xff0c;能够帮助我们进行在线教学&#xff0c;线上培训&#xff0c;游戏直播与分享&#xff0c;视频记录等等。并且很多的录屏软件都有免费的功能&#xff0c;它们让大家的录屏变得更加的方便。如果大家需要录屏工具的话&#xff0…

828华为云征文|基于华为云Flexus X实例部署Uptime-Kuma服务器监控面板

目录 前言 一、Flexus云服务器X介绍 1.1 Flexus云服务器X实例简介 1.2 Flexus云服务器X实例特点 1.3 Flexus云服务器X实例场景需求 二、Flexus云服务器X购买 2.1 Flexus X实例购买 2.2 重置密码 2.3 登录服务器 三、Flexus X安装uptime-kuma面板 3.1 uptime-kuma介绍 3.2 uptim…

现在增长空间原来在这里

我们常常说&#xff0c;现在的经济形势不太好&#xff0c;赚钱太难了&#xff0c;卷得太厉害了。只能出海去卷&#xff0c;或者是到老年人的群体里去卷银发经济。 但是&#xff0c;现在就有一个巨大的市场&#xff0c;等待着大家去挖掘。这就是下沉市场。 以前我们也提过下沉…

[Admin] Things Need to Know

List View Bulk Actions Highlight: To take bulk actions on all of the available records in a list, you click the bulk action button without selecting any records.