2024/9/20 使用QT实现扫雷游戏

有三种难度初级6x6 中级10x10  高级16x16

 

完成游戏

游戏失败后,无法再次完成游戏,只能重新开始一局

对Qpushbutton进行重写

mybutton.h

#ifndef MYBUTTON_H
#define MYBUTTON_H

#include <QObject>
#include <QWidget>
#include <QPushButton>
#include <QMouseEvent>
class mybutton : public QPushButton
{
    Q_OBJECT
public:
    mybutton(QWidget *parent = nullptr);
    void set_value(int hang,int lie);
    //重写鼠标点击事件
protected:
    virtual void    mouseReleaseEvent(QMouseEvent * event);

    //定义两个信号
signals:
    void LeftClicked(int hang,int lie);
    void RightClicked(int hang,int lie);

private:
    int lie;
    int hang;
};

#endif // MYBUTTON_H
 

mybutton.c

#include "mybutton.h"

 mybutton::mybutton(QWidget *parent):
    QPushButton(parent)
{

}

 void mybutton::mouseReleaseEvent(QMouseEvent * event)
 {
     //判断是鼠标左键点击还是右键点击
     if(event->button() ==  Qt::LeftButton)
     {//左键点击了
         emit LeftClicked(hang,lie);
     }
     if(event->button() ==  Qt::RightButton)
     {//右键点击了
         emit RightClicked(hang,lie);
     }
 }
void mybutton::set_value(int hang,int lie)
{
    this->hang=hang;
    this->lie=lie;
}
 

widget.h

#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QTimer>
#include "mybutton.h"
QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE

#define Lei 10 //设置雷占格子的多少

class Widget : public QWidget
{
    Q_OBJECT

public:
    Widget(QWidget *parent = nullptr);
    ~Widget();
    void time_out(void);
    int GetThisMine(int hang,int lie,int i,int j);
    void Initbutton(int i,int j);
    void clear_button(int i,int j);
    //定义两个槽函数 用于处理按键左右键点击的信号
public slots:
   void LeftClickedSlots(int hang,int lie);
   void RightClickedSlots(int hang,int lie);
   void on_pushButton_clicked();
   void on_pushButton_2_clicked();
   void on_pushButton_4_clicked();
   void on_pushButton_3_clicked();

private slots:
   void on_pushButton_5_clicked();

private:
    Ui::Widget *ui;
    QTimer *tm1;
    int i,j;
    mybutton *Buttons;
    bool *mine;
    int flag=0;
    int s=0;
    int win=0;
};
#endif // WIDGET_H
 

widget.cpp

#include "widget.h"
#include "ui_widget.h"
#include "QDebug"
#include <QThread>
#include <QMessageBox>
Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
    tm1 = new QTimer(this);
    //2,绑定定时器时间到槽
    connect(tm1, &QTimer::timeout, this,&Widget::time_out);
    //3.启动定时器
    tm1->start(1000); //定时器时间1000ms
    ui->lcdNumber->display(100);
    this->setWindowTitle("扫雷");
    //设置布局框大小
    ui->gridLayoutWidget->resize(this->size());
    ui->gridLayoutWidget->move(0,0);
}

Widget::~Widget()
{
    delete ui;
    delete []mine;
    delete []Buttons;
}

void Widget::time_out()
{
    //qDebug()<< "定时器时间到";
    int num = ui->lcdNumber->value();
    num -= 1;
    if(num==0)
        tm1->stop(); //停止定时器
    ui->lcdNumber->display(num);
}
void Widget::Initbutton(int i,int j)
{   this->i=i;
    this->j=j;
    Buttons =new mybutton[i*j];
    for(int hang = 0; hang <i; hang++)
        for(int lie =0; lie < j; lie++)
        {
            Buttons[hang*j+lie].set_value(hang,lie);
            Buttons[hang*j+lie].setSizePolicy(QSizePolicy::Preferred,QSizePolicy::Preferred);
            //将按钮放到布局中
            ui->gridLayout->addWidget(&Buttons[hang*j+lie], hang+1, lie);
            ui->gridLayout->setRowStretch(hang,0);
            ui->gridLayout->setColumnStretch(lie,0);
            //绑定信号和槽
            connect(&Buttons[hang*j+lie], &mybutton::LeftClicked, this, &Widget::LeftClickedSlots);
            connect(&Buttons[hang*j+lie], &mybutton::RightClicked, this, &Widget::RightClickedSlots);
        }
    //摇雷 ((i*j)/4)个
    srand(time(0));
    mine = new bool[i*j];
    for(int k=0;k<i*j;k++) mine[k]=false;
    for(int k=0;k<((i*j)/Lei);k++)
    {
        while(1){
            int h = rand() % i;
            int l = rand() % j;
            if(!mine[h*j+l])
            {//此处没有雷
                mine[h*j+l] = true;
                break;
            }
        }
    }
//    for(int h=0;h<i*j;h++)
//        qDebug()<<mine[h];
    }
void Widget::clear_button(int i,int j)
{
    for(int hang = 0; hang <i; hang++)
        for(int lie =0; lie < j; lie++)
            ui->gridLayout->removeWidget(&Buttons[hang*j+lie]);
    delete []Buttons;
    s=0;
    win=0;
}

void Widget::on_pushButton_clicked()//低级难度
{
    switch(flag)
    {
    case 0:Initbutton(6,6);break;
    case 1:clear_button(6,6);Initbutton(6,6);break;
    case 2:clear_button(10,10);Initbutton(6,6);break;
    case 3:clear_button(16,16);Initbutton(6,6);break;
    default: qDebug()<<"游戏未开始";
    }
    flag=1;
}


void Widget::on_pushButton_2_clicked()//中级难度
{
    switch(flag)
    {
    case 0:Initbutton(10,10);break;
    case 1:clear_button(6,6);Initbutton(10,10);break;
    case 2:clear_button(10,10);Initbutton(10,10);break;
    case 3:clear_button(16,16);Initbutton(10,10);break;
    default: qDebug()<<"游戏未开始";
    }
    flag=2;
}

void Widget::on_pushButton_4_clicked()//高级难度
{
    switch(flag)
    {
    case 0:Initbutton(16,16);break;
    case 1:clear_button(6,6);Initbutton(16,16);break;
    case 2:clear_button(10,10);Initbutton(16,16);break;
    case 3:clear_button(16,16);Initbutton(16,16);break;
    default: qDebug()<<"游戏未开始";
    }
    flag=3;
}

void Widget::on_pushButton_3_clicked()//重玩
{
    switch(flag)
    {
    case 1:clear_button(6,6);Initbutton(6,6);break;
    case 2:clear_button(10,10);Initbutton(10,10);break;
    case 3:clear_button(16,16);Initbutton(16,16);break;
    default: qDebug()<<"游戏未开始";
    }
}
void Widget::LeftClickedSlots(int hang,int lie)//左击
{
    int n=i,m=j;
    qDebug() << "左键点击:" << hang << lie;
    Buttons[hang*m+lie].setCheckable(true);
    Buttons[hang*m+lie].setChecked(true); //显示为按键按下
    Buttons[hang*m+lie].setDisabled(true); // 按键不能交互
    ui->gridLayoutWidget->update();
    qDebug() << "按键锁定:";
    //显示周围有多少雷
    if(GetThisMine(hang,lie,n,m)==-1){
        Buttons[hang*m+lie].setText(QString::number(-1));
        QMessageBox::information(nullptr, "游戏失败", "恭喜你踩了一个雷");
    }
    //int cnt = GetThisMine(hang,lie,n,m);
    //Buttons[hang*m+lie].setText(QString::number(cnt));

}

void Widget::RightClickedSlots(int hang,int lie)//右击
{
    int n=i,m=j;
    qDebug() << "右键点击:" << hang << lie;
    //标记是雷
    QIcon icon(":/1.png");//使用资源包,相对路径
    if(Buttons[hang*m+lie].icon().isNull())
    {
    Buttons[hang*m+lie].setIcon(icon);
    qDebug()<<n;
    Buttons[hang*m+lie].setIconSize(QSize(Buttons[hang*m+lie].size().width()-20,
                                    Buttons[hang*m+lie].size().height() -20));
    win++;
    if(win==i*j) QMessageBox::information(nullptr, "游戏胜利", "恭喜你完成了游戏");
    return;
    }
    if(!Buttons[hang*m+lie].icon().isNull())
        Buttons[hang*m+lie].setIcon(QIcon());
    win--;

}

//计算hang,lie位置有多少雷
int Widget::GetThisMine(int hang,int lie,int n,int m)
{
    if(mine[hang*m+lie]==true) return -1;
    int cnt = 0;
    for(int i=-1; i<= 1; i++)
        for(int j=-1; j<=1; j++)
        {
            if(  !(i==0 && j==0) )
            {//排除自己
                if( hang+i < 0 || hang+i >= n || lie+j >= m || lie+j < 0 )
                    continue; //排除越界
                if(mine[(hang+i)*m+lie+j]) cnt ++;
            }
        }
    QThread::msleep(50);
    if(s==n*m) return cnt;
    qDebug() << "当前有值:"<<cnt<< s++<<endl;
    Buttons[hang*m+lie].setText(QString::number(cnt));
    Buttons[hang*m+lie].setCheckable(true);
    Buttons[hang*m+lie].setChecked(true); //显示为按键按下
    Buttons[hang*m+lie].setDisabled(true); // 按键不能交互
    ui->gridLayoutWidget->update();
    win++;
    if(win==i*j) QMessageBox::information(nullptr, "游戏胜利", "恭喜你完成了游戏");
    if(cnt==0)
    {
        //四个方向 左 上 右 下
        if(lie-1>=0&&Buttons[hang*m+lie-1].text().isEmpty()){
            qDebug() << "左连:";
            GetThisMine(hang,lie-1,n,m);
        }
        if(hang-1>=0&&Buttons[(hang-1)*m+lie].text().isEmpty()){
            qDebug() << "上连:";
            GetThisMine(hang-1,lie,n,m);
        }
        if(lie+1<m&&Buttons[hang*m+lie+1].text().isEmpty()){
            qDebug() << "右连:";
            GetThisMine(hang,lie+1,n,m);
        }
        if(hang+1<n&&Buttons[(hang+1)*m+lie].text().isEmpty()){
            qDebug() << "下连:";
            GetThisMine(hang+1,lie,n,m);
        }
        //四个角落 左上 右上 左下 右下
        if(hang-1>=0&&lie-1>=0&&Buttons[(hang-1)*m+lie-1].text().isEmpty()){
            qDebug() << "左上连:";
            GetThisMine(hang-1,lie-1,n,m);
        }
        if(hang-1>=0&&lie+1<m&&Buttons[(hang-1)*m+lie+1].text().isEmpty()){
            qDebug() << "右上连:";
            GetThisMine(hang-1,lie+1,n,m);
        }
        if(hang+1<n&&lie-1>=0&&Buttons[(hang+1)*m+lie-1].text().isEmpty()){
            qDebug() << "左下连:";
            GetThisMine(hang+1,lie-1,n,m);
        }
        if(hang+1<n&&lie+1<m&&Buttons[(hang+1)*m+lie+1].text().isEmpty()){
            qDebug() << "右下连:";
            GetThisMine(hang+1,lie+1,n,m);
        }
    }
    return cnt;
}

void Widget::on_pushButton_5_clicked()
{
    //设置布局框大小
    ui->gridLayoutWidget->resize(this->size());
    ui->gridLayoutWidget->move(0,0);
}
 

main.c

#include "widget.h"

#include <QApplication>

int main(int argc, char *argv[])
{
    QApplication a(argc, argv);
    Widget w;
    w.show();
    return a.exec();
}
 

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2024年8月HarmonyOS鸿蒙应用开发者高级认证全新题库

有题库在手&#xff0c;一小时轻松拿下鸿蒙高级。你们需要也可以无偿分享哦&#xff01; 项目需要为不同的设备形态(如手机 、 智能手表)提供定制化构建 。请说明如何在 DevEcostudio 中 设置不同的构建配置&#xff0c; 以生成针对不同设备的 hap 包&#xff1a; 在模块级别 b…

JavaWeb的Filter详解

过滤器Filter 什么是Filter&#xff1f; 依据字面上的中文意思为过滤器。Filter的作用 当用户的请求到达指定的URL之前&#xff0c;可以借助Filter来改变这些请求的内容&#xff1b;同样地&#xff0c;当响应结果到达客户端之前&#xff0c;可以使用Filter修改输出的内容。什么…

【数据结构入门】排序算法之三路划分与非比较排序

文章目录 前言 一、三路划分优化 1.1. 基本思想 1.2. 实现步骤 1.3. 优点 1.4 代码实现 二、非比较排序 2.1 计数排序 2.1.1基本思想 2.1.2具体步骤 2.1.3算法特性 2.1.4算法实现 2.2 基数排序 2.2.1基本思想 2.2.2具体步骤 2.2.3 基数排序的方法 2.2.4算法特…

MongoDB在Linux系统中的安装与配置指南

在这篇文章中&#xff0c;我们将介绍如何在CentOS 7服务器上安装MongoDB&#xff0c;并通过DataX将数据从MongoDB迁移到MySQL数据库。这将包括MongoDB的安装、配置、数据准备以及使用DataX进行数据迁移的详细步骤。 MongoDB简介 MongoDB是一个高性能、开源、无模式的文档型数据…

Leetcode面试经典150题-97.交错字符串

给定三个字符串 s1、s2、s3&#xff0c;请你帮忙验证 s3 是否是由 s1 和 s2 交错 组成的。 两个字符串 s 和 t 交错 的定义与过程如下&#xff0c;其中每个字符串都会被分割成若干 非空 子字符串 &#xff1a; s s1 s2 ... snt t1 t2 ... tm|n - m| < 1交错 是…

Springboot3 + MyBatis-Plus + MySql + Uniapp 实现商品规格选择sku(附带自设计数据库,最新保姆级教程)

Springboot3 MyBatis-Plus MySql Uniapp 实现商品规格选择sku&#xff08;附带自设计数据库&#xff0c;最新保姆级教程&#xff09; 1、效果展示2、数据库设计2.1 商品表2.2 商品价格和规格中间表2.3 商品规格表 3、后端代码3.1 model3.2 vo3.3 mapper、server、serverImp3…

使用express或koa或nginx部署history路由模式的单页面应用

使用hash模式会有#&#xff0c;影响美观&#xff0c;所以使用history模式会是个更好的选择。 前端项目打包上线部署&#xff0c;可以使用下面的方式部署history模式的项目&#xff0c;下面以 jyH5 为例 expressjs部署 express脚手架搭建的app.js中添加如下代码&#xff1a; …

CDA Level 1 业务数据分析

目录 理解业务数据分析方法、掌握业务数据分析流程、能够使用及设计创建业务指标、能够结合业务模型及业务分析方法正确理解业务问题&#xff0c;找到问题原因&#xff0c;并能够提出解决问题建议&#xff0c;这个章节的应用会考的比较多&#xff08;终于正经起来了呢&#xf…

设计模式 组合模式(Composite Pattern)

组合模式简绍 组合模式&#xff08;Composite Pattern&#xff09;是一种结构型设计模式&#xff0c;它允许你将对象组合成树形结构来表示“部分-整体”的层次结构。组合模式使得客户端可以用一致的方式处理单个对象和组合对象。这样&#xff0c;可以在不知道对象具体类型的条…

7--SpringBoot-后端开发、原理详解(面试高频提问点)

目录 SpringBoot原理 起步依赖 自动配置 配置优先级 Bean设置 获取Bean 第三方Bean SpringBoot原理 内容偏向于底层的原理分析 基于Spring框架进行项目的开发有两个不足的地方&#xff1a; 在pom.xml中依赖配置比较繁琐&#xff0c;在项目开发时&#xff0c;需要自己去找…

手写Spring

简单实现Spring基于注解配置 ComponentScan Target(ElementType.TYPE) Retention(RetentionPolicy.RUNTIME) public interface ComponentScan {String value() default ""; } 相当于component-scan HspSpringConfig ComponentScan(value "spring.write.com…

C#自定义曲线绘图面板

一、实现功能 1、显示面板绘制。 2、拖动面板&#xff0c;X轴、Y轴都可以拖动。 3、显示面板缩放&#xff0c;放大或者缩小。 4、鼠标在面板中对应的XY轴数值。 5、自动生成的数据数组&#xff0c;曲线显示。 6、鼠标是否在曲线上检测。 二、界面 拖动面板 鼠标在曲线上…

【随手笔记】使用J-LINK读写芯片内存数据

第一种使用JLINK.exe 1. 打开j-link.exe 2.输入【usb】 3. 连接芯片 输入【connect】输入芯片型号【STM32L071RB】输入连接方式 【S】 使用SWD连接方式输入连接速率 【4000】连接成功 4. 输入【&#xff1f;】查看指令提示 5. 读写指令 Mem Mem [<Zone>…

DataFrame生成excel后为什么多了一行数字

问题描述 python查询数据生成excel文件&#xff0c;生成的excel多了第一行数字索引&#xff0c;1,2,3,4,5...... 代码&#xff1a; df pd.DataFrame(data)df.to_excel(filename, sheet_name用户信息表, indexFalse) 解决&#xff1a; 原理也很简单&#xff0c;就是设置个参…

【python设计模式7】行为型模式2

目录 策略模式 模板方法模式 策略模式 定义一个个算法&#xff0c;把它们封装起来&#xff0c;并且使它们可以相互替换。本模式使得算法可独立于使用它的客户而变化。角色有&#xff1a;抽象策略、具体策略和上下文。 from abc import abstractmethod, ABCMeta from datetim…

前端vue-ref与document.querySelector的对比

ref只在本组件中查找&#xff0c;而document.querySelector是在整个页面查找

Golang | Leetcode Golang题解之第414题第三大的数

题目&#xff1a; 题解&#xff1a; func thirdMax(nums []int) int {var a, b, c *intfor _, num : range nums {num : numif a nil || num > *a {a, b, c &num, a, b} else if *a > num && (b nil || num > *b) {b, c &num, b} else if b ! ni…

SQL Server性能优化之读写分离

理论部分: 数据库读写分离&#xff1a; 主库&#xff1a;负责数据库操作增删改 20% 多个从库&#xff1a;负责数据库查询操作 80% 读写分离的四种模式 1.快照发布&#xff1a;发布服务器按照预定的时间间隔向订阅服务器发送已发布的数据快照 2.事务发布[比较主流常见]&#xf…

Ceph 基本架构(一)

Ceph架构图 Ceph整体组成 Ceph 是一个开源的分布式存储系统&#xff0c;设计用于提供优秀的性能、可靠性和可扩展性。Ceph 的架构主要由几个核心组件构成&#xff0c;每个组件都有特定的功能&#xff0c;共同协作以实现高可用性和数据的一致性。 以下是 Ceph 的整体架构及其…

2024 “华为杯” 中国研究生数学建模竞赛(C题)深度剖析|数据驱动下磁性元件的磁芯损耗建模|数学建模完整代码+建模过程全解全析

当大家面临着复杂的数学建模问题时&#xff0c;你是否曾经感到茫然无措&#xff1f;作为2022年美国大学生数学建模比赛的O奖得主&#xff0c;我为大家提供了一套优秀的解题思路&#xff0c;让你轻松应对各种难题&#xff01; CS团队倾注了大量时间和心血&#xff0c;深入挖掘解…