Qt小项目贪吃蛇实线,主要掌握定时器、信号与槽、按键事件、绘制事件、坐标运算、随机数生成等

Qt小项目贪吃蛇实线,主要掌握定时器、信号与槽、按键事件、绘制事件、坐标运算、随机数生成等

  • Qt 贪吃蛇演示
  • QWidget 绘制界面
  • 项目源文件 注释清晰
    • widget.h
    • widget.cpp
  • 拓展
    • QTimer
    • QKeyEvent
    • QRectF
    • QPointF
    • QPainter
    • QIcon

Qt 贪吃蛇演示

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

QWidget 绘制界面

在这里插入图片描述
在这里插入图片描述

项目源文件 注释清晰

widget.h

#ifndef WIDGET_H
#define WIDGET_H#include <QWidget>
#include <QtDebug>#include <QKeyEvent>
#include <QTimer>
#include <QRectF>
#include <QPointF>#include <QIcon>
#include <QPainter>QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACEtypedef enum direct {dir_UP,dir_DOWN,dir_LEFT,dir_RIGHT
}dir_t;class Widget : public QWidget
{Q_OBJECTpublic:Widget(QWidget *parent = nullptr);~Widget();private:void topAddRect();void deleteLast();void downAddRect();void leftAddRect();void rightAddRect();void addFood();bool  checkHit();protected:// 按键按压处理void keyPressEvent(QKeyEvent *event);void paintEvent(QPaintEvent *event);private Q_SLOTS:void my_timeout();private:Ui::Widget *ui;int moveFlag = dir_t::dir_UP;bool gameStart = false;QTimer *timer;const int startTime = 100;// 贪吃蛇QList<QRectF>snakeList;// 贪吃蛇小方块const int  nodeWidth = 20;const  int  nodeHeight = 20;// 食物QRectF food;};
#endif // WIDGET_H

widget.cpp

#include "widget.h"
#include "ui_widget.h"// 三个方块代码蛇,最上面的第一个和最后面的一个通过按键进行交替删除 即snake[0]永远表示头Widget::Widget(QWidget *parent): QWidget(parent), ui(new Ui::Widget)
{ui->setupUi(this);this->setWindowTitle("贪吃蛇疯子");this->setWindowIcon(QIcon(":/icons/title.jpg"));qDebug()<<"x = "<<this->x()<<" y = "<<this->y()<<" height = "<<this->height()<<" width = "<<this->width();timer = new QTimer();connect(timer, &QTimer::timeout, this , &Widget::my_timeout);// 初始化蛇身子 3个方块QRectF rect(this->width()/2,this->height()/2,nodeWidth,nodeHeight);snakeList.append(rect);topAddRect();topAddRect();// 添加食物addFood();
}Widget::~Widget()
{delete ui;
}// 增加一个方块 左上和右下确认一个方块坐标系
void Widget::topAddRect()
{QPointF leftTop;                // 左上角坐标QPointF rightBottom;       // 右上角坐标if (snakeList[0].y()  <=  0) {// 当蛇移动到窗口顶部:确认新坐标 y = 窗口高度 - 蛇方块高度leftTop = QPointF(snakeList[0].x(), this->height() - nodeHeight);rightBottom  = QPointF(snakeList[0].x() + nodeWidth , this->height());}else {// 向上移动: y坐标必然减少 减少的右下角坐标为之前的下一个方块的右上角坐标leftTop = QPointF(snakeList[0].x(), snakeList[0].y() - nodeHeight);rightBottom = snakeList[0].topRight();}// 插入矩形小方块1个,由于采用的是List链表,这是典型的前插,追加snakeList.insert(0,QRectF(leftTop,rightBottom));
}void Widget::downAddRect()
{QPointF leftTop;                // 左上角坐标QPointF rightBottom;       // 右上角坐标if (snakeList[0].y() > this->height() - nodeHeight) {// 当蛇移动到窗口底部:确认新坐标 y = 0leftTop = QPointF(snakeList[0].x(), 0);rightBottom  = QPointF(snakeList[0].x() + nodeWidth, 0 + nodeHeight);}else {// 向下移动: y坐标必然增加leftTop = QPointF(snakeList[0].x(), snakeList[0].y() + nodeHeight);rightBottom = snakeList[0].bottomRight() + QPointF(0,nodeHeight);}// 插入矩形小方块1个snakeList.insert(0,QRectF(leftTop,rightBottom));
}void Widget::leftAddRect()
{QPointF leftTop;                // 左上角坐标QPointF rightBottom;       // 右上角坐标if (snakeList[0].x() <= 0) {// 当蛇移动到窗口最左部:确认新坐标 x = 窗口宽度 - 小方块宽度leftTop = QPointF(this->width() - nodeWidth, snakeList[0].y());}else {// 向左移动:x坐标必然减少leftTop = QPointF(snakeList[0].x() - nodeWidth, snakeList[0].y());}// 右下角坐标 = 之前一个的左上角坐标x,y + 小方块的宽高rightBottom  = leftTop + QPointF(nodeWidth, nodeHeight);// 插入矩形小方块1个snakeList.insert(0,QRectF(leftTop,rightBottom));
}void Widget::rightAddRect()
{QPointF leftTop;                // 左上角坐标QPointF rightBottom;       // 右上角坐标if (snakeList[0].x() + nodeWidth > this->width()) {// 当蛇移动到窗口最右部:确认新坐标 x  = 0leftTop = QPointF(0, snakeList[0].y());}else {// 向右移动:x坐标必然增加leftTop = QPointF(snakeList[0].x() + nodeWidth, snakeList[0].y());}// 右下角坐标 = 之前一个的左上角坐标x,y + 小方块的宽高rightBottom  = leftTop + QPointF(nodeWidth, nodeHeight);// 插入矩形小方块1个snakeList.insert(0,QRectF(leftTop,rightBottom));
}// 删除蛇身最后一个
void Widget::deleteLast()
{snakeList.removeLast();
}// 食物是随机出现的
void Widget::addFood()
{int rectX = (qrand() % (this->width() / 20)) * 20;int rectY = (qrand() % (this->height() / 20)) * 20;// 控制小球出现的范围if (rectX >= 0 && rectY>=0) {food = QRectF(rectX, rectY, nodeWidth, nodeHeight);qDebug()<<"food = "<<food;}
}// 蛇头和蛇身碰撞检查,其实就是蛇头和蛇尾其中一个方块重合
bool Widget::checkHit()
{// 整个蛇的长度都遍历一遍for (int i =0 ; i < snakeList.length(); i++) {// 从蛇头后的第一个开始检查 只要有for (int j=  i+ 1; j < snakeList.length();j++) {// rect0和rectx相等,表示它们的坐标、宽度和高度都一致if (snakeList[0] == snakeList[j]) {return true;}}}return false;
}void Widget::keyPressEvent(QKeyEvent *event)
{switch (event->key()) {case Qt::Key::Key_Up:if (moveFlag != dir_DOWN) {moveFlag = dir_UP;}break;case Qt::Key::Key_Down:if (moveFlag != dir_UP) {moveFlag = dir_DOWN ;}break;case Qt::Key::Key_Left:if (moveFlag != dir_RIGHT) {moveFlag = dir_LEFT;}break;case Qt::Key::Key_Right:if (moveFlag != dir_LEFT) {moveFlag = dir_RIGHT;}break;case Qt::Key_Space:if (gameStart == false) {gameStart = true;timer->start(startTime); // 100ms}else {gameStart = false;timer->stop();}break;default:break;}
}void Widget::paintEvent(QPaintEvent *event)
{QPainter painter(this);QPen pen;QBrush brush;QPixmap pix;// 绘制图片背景pix.load(":/icons/bg_snake.png");painter.drawPixmap(0,0,this->width(),this->height(),pix);// 绘制蛇pen.setColor(Qt::color0);brush.setColor(Qt::darkGreen);          // 绿色brush.setStyle(Qt::SolidPattern);       // 实线图案painter.setPen(pen);painter.setBrush(brush);for (int i = 0; i < snakeList.length(); i++) {painter.drawRect(snakeList[i]);}// 分数ui->label_score->setText(QString::number(snakeList.length() -3));// 绘制食物painter.drawEllipse(food);// 蛇头碰到蛇身结束游戏if (checkHit()) {QFont font("方正舒体",30,QFont::ExtraLight,false);painter.setFont(font);painter.drawText((this->width() - 200)/2,this->height()/2,QString("游戏结束"));timer->stop();}QWidget::paintEvent(event);}// 定时器槽函数
void Widget::my_timeout()
{//int count = 1;  // 采用这种方式也可以加长蛇身,不过我还是喜欢我的烂方法// 判断蛇是否吃到食物 是否重合[交叉;相交;贯穿;横穿;横断]  蛇变长if (snakeList[0].intersects(food)) {qDebug()<<"吃到食物 snakeList[0] = "<<snakeList[0]<<" food = "<<food;//count ++ ;      // 例如 2switch (moveFlag) {case dir_UP:this->topAddRect();         // +1break;case dir_DOWN:this->downAddRect();     // +1break;case dir_LEFT:this->leftAddRect();        // +1break;case dir_RIGHT:this->rightAddRect();     // +1break;default:break;}addFood();    // 食物位置变化return;}// 加长蛇身  每次在最前面增加一个// while (count--) {switch (moveFlag) {case dir_UP:this->topAddRect();         // +1break;case dir_DOWN:this->downAddRect();     // +1break;case dir_LEFT:this->leftAddRect();        // +1break;case dir_RIGHT:this->rightAddRect();     // +1break;default:break;}//}// 为了动态显示每次最前面插入一个,最后面就减少一个deleteLast();       // 删除蛇尾  -1this->update(); // 刷新绘制函数
}

拓展

QTimer

QTimer是Qt中用于定时器操作的类,它提供了一些常用的函数来控制和管理定时器的运行。下面是一些常用的QTimer函数的简介:

  1. start(int msec): 启动定时器,以指定的毫秒数为间隔触发定时器的timeout()信号。
  2. stop(): 停止定时器,不再触发timeout()信号。
  3. setInterval(int msec): 设置定时器的间隔时间,以毫秒为单位。
  4. interval(): 获取当前定时器的间隔时间。
  5. isActive(): 判断定时器是否处于活动状态,即是否正在运行。
  6. setSingleShot(bool singleShot): 设置定时器的运行模式,如果设置为true,则定时器只触发一次;如果设置为false(默认值),则定时器会一直触发。
  7. singleShot(int msec, const QObject* receiver, const char* member): 创建一个单次触发的定时器,指定触发时间、接收信号的对象和相应的槽函数。
  8. remainingTime(): 获取定时器剩余的触发时间,以毫秒为单位。

这些函数提供了基本的定时器操作功能,可以配合定时器的信号timeout()以及连接(Qt的信号与槽机制)来实现所需的定时操作。

QKeyEvent

QKeyEvent是Qt中用于处理键盘事件的类,它提供了一些常用的函数来获取和处理键盘事件的相关信息。下面是一些常用的QKeyEvent函数的简介:

  1. key(): 获取触发键盘事件的按键的Qt键盘码,返回一个Qt::Key枚举值。
  2. text(): 获取触发键盘事件的按键对应的文字,返回一个QString。
  3. modifiers(): 获取触发键盘事件时的修饰键状态,返回一个Qt::KeyboardModifiers枚举值,可用于检查Shift、Ctrl、Alt等修饰键的状态。
  4. isAutoRepeat(): 判断触发键盘事件的按键是否是自动重复的按下事件。
  5. count(): 获取自动重复按键连续触发的次数。
  6. nativeVirtualKey(): 获取底层平台的虚拟键码,返回一个int值。
  7. nativeModifiers(): 获取底层平台的修饰键状态,返回一个int值。

这些函数可以帮助你获取与键盘事件相关的信息,例如获取按下的键是哪个键,是否同时按下了修饰键,以及自动重复事件的次数等。你可以使用这些函数来处理键盘事件并根据需要执行相应的操作。

QRectF

QRectF是Qt中用于表示浮点数精度的矩形区域的类,它提供了一些常用的函数来操作和管理矩形区域。下面是一些常用的QRectF函数的简介:

  1. QRectF(): 默认构造函数,创建一个无效的矩形区域。
  2. QRectF(qreal x, qreal y, qreal width, qreal height): 构造函数,创建一个以给定坐标、宽度和高度定义的矩形区域。
  3. setRect(qreal x, qreal y, qreal width, qreal height): 设置矩形区域的位置和尺寸。
  4. setCoords(qreal x1, qreal y1, qreal x2, qreal y2): 设置矩形区域的左上角和右下角的坐标。
  5. x(), y(), width(), height(): 获取矩形区域的左上角的x和y坐标,以及宽度和高度。
  6. left(), top(), right(), bottom(): 获取矩形区域的左、上、右、下边界的坐标。
  7. setX(qreal x), setY(qreal y), setWidth(qreal width), setHeight(qreal height): 设置矩形区域的左上角的x和y坐标,以及宽度和高度。
  8. setLeft(qreal left), setTop(qreal top), setRight(qreal right), setBottom(qreal bottom): 设置矩形区域的左、上、右、下边界的坐标。
  9. moveTo(qreal x, qreal y): 移动矩形区域的位置,将其左上角设置为给定坐标。
  10. translated(qreal dx, qreal dy): 平移矩形区域,沿x和y方向分别平移给定的距离。
  11. contains(const QPointF &point): 判断矩形区域是否包含给定的点。
  12. isEmpty(): 判断矩形区域是否为空,即宽度或高度是否为0。
  13. isNull(): 判断矩形区域是否为空,即宽度和高度是否为0。

QPointF

QPointF是Qt中用于表示浮点数精度的二维点的类,它提供了一些常用的函数来操作和管理点的坐标。下面是一些常用的QPointF函数的简介:

  1. QPointF(): 默认构造函数,创建一个具有零值坐标的点。
  2. QPointF(qreal x, qreal y): 构造函数,创建一个具有给定坐标的点。
  3. setX(qreal x), setY(qreal y): 设置点的x和y坐标。
  4. x(), y(): 获取点的x和y坐标。
  5. isNull(): 判断点是否为空,即坐标是否为零值。
  6. manhattanLength(): 计算点到坐标原点的曼哈顿距离(绝对值之和)。
  7. distanceToLine(const QLineF &line): 计算点到给定直线的距离。
  8. distanceToPoint(const QPointF &point): 计算点到给定点的距离。
  9. operator==(const QPointF &p1, const QPointF &p2): 判断两个点是否相等。
  10. operator!=(const QPointF &p1, const QPointF &p2): 判断两个点是否不相等。
  11. isNull(const QPointF &point): 判断给定点是否为空,即坐标是否为零值。

这些函数提供了对点的基本操作,包括设置坐标、获取坐标、判断点是否为空、计算与其他点或直线的距离等。你可以使用这些函数来创建、修改和计算点的坐标,以满足你的需求。

QPainter

QPainter是Qt提供的一个用于绘图的类,它封装了绘制图形、图像以及文本的功能。下面是一些常用的QPainter函数的简介:

  1. begin(QPaintDevice *device): 开始在给定的绘图设备上进行绘制,device可以是QWidget、QImage等。
  2. end(): 结束绘制操作。
  3. setPen(const QPen &pen): 设置绘制的画笔,用于定义线条的样式、颜色等属性。
  4. setBrush(const QBrush &brush): 设置绘制的刷子,用于填充封闭图形的颜色、渐变等属性。
  5. setRenderHint(RenderHint hint, bool on = true): 开启或关闭绘制的渲染提示,例如抗锯齿、文本反锯齿等。
  6. drawLine(const QLine &line): 绘制直线。
  7. drawRect(const QRect &rect): 绘制矩形。
  8. drawEllipse(const QRect &rect): 绘制椭圆。
  9. drawText(const QPointF &pos, const QString &text): 在给定点位置绘制文本。
  10. drawPixmap(const QRectF &targetRect, const QPixmap &pixmap): 在给定矩形区域绘制像素图。
  11. save(), restore(): 保存和恢复绘图状态,用于实现绘制状态的切换、叠加效果等。
  12. resetTransform(): 重置坐标变换矩阵。
  13. translate(qreal dx, qreal dy): 平移画笔的原点。
  14. scale(qreal sx, qreal sy): 缩放画笔,按比例调整绘图。
  15. rotate(qreal angle): 旋转画笔,围绕原点进行旋转。

这些函数提供了绘制基本图形、文本和图像的能力,通过设置画笔、刷子和渲染提示等属性,可以实现不同样式的绘制效果。同时,你还可以使用平移、缩放和旋转等变换函数来改变画笔的绘制位置和方向。使用这些函数,你可以实现各种绘图需求,创建出丰富多样的用户界面。

QIcon

QIcon是Qt提供的一个用于管理图标的类,它可以加载、显示和操作图标。下面是一些常用的QIcon函数的简介:

  1. QIcon(): 默认构造函数,创建一个空的图标。
  2. QIcon(const QString &filename): 构造函数,根据给定的文件名加载图标。
  3. QIcon(const QPixmap &pixmap): 构造函数,根据给定的像素图创建图标。
  4. QIcon(const QIcon &other): 拷贝构造函数,创建一个与给定图标相同的拷贝。
  5. addFile(const QString &filename, const QSize &size = QSize(), QIcon::Mode mode = QIcon::Normal, QIcon::State state = QIcon::Off): 添加一个图像文件到图标中,可以指定图像的大小、模式和状态。
  6. isNull(): 判断图标是否为空,即没有加载任何图像。
  7. pixmap(const QSize &size, QIcon::Mode mode = QIcon::Normal, QIcon::State state = QIcon::Off) const: 获取图标的像素图,可以指定图像的大小、模式和状态。
  8. paint(QPainter *painter, const QRect &rect, Qt::Alignment alignment = Qt::AlignCenter, QIcon::Mode mode = QIcon::Normal, QIcon::State state = QIcon::Off) const: 在给定的矩形区域内绘制图标,可以指定对齐方式、模式和状态。
  9. operator=(const QIcon &other): 赋值运算符,将给定图标的内容复制到当前图标。
  10. operator==, operator!=: 用于比较两个图标是否相等或不相等。

这些函数提供了加载、显示和操作图标的功能,你可以根据需要加载图像文件,设置图标的大小、模式和状态,绘制图标到指定的区域,并进行图标的比较和赋值操作。QIcon类在Qt中广泛用于界面开发,它使得图标的管理和使用变得简单和灵活。

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

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

相关文章

2023集成电路产业发展与产教融合高峰论坛会议顺利举行

8月5日&#xff0c;由中国半导体行业协会和市政府共同主办&#xff0c;天水师范学院、天水华天科技股份有限公司、杭州加速科技有限公司承办的2023集成电路产业发展与产教融合高峰论坛在天水举行。天水市委书记冯文戈&#xff0c;教育部学生服务与素质发展中心副主任方伟&#…

4 三组例子,用OpenCV玩转图像-AI-python

读取&#xff0c;缩放&#xff0c;旋转&#xff0c;写入图像 首先导入包&#xff0c;为了显示导入matplotlib/为了在matplotlib显示 导入CV2/查看版本 导入图片/查看图片类型 图片数组 数组大小 对于opencv通道顺序蓝色B、绿色G、红色R matplotlib通道顺序为 红色R、绿色G、蓝…

云原生K8S------Yaml文件详解

目录 一&#xff1a;K8S支持的文件格式 1&#xff0c;yaml和json的主要区别 2&#xff0c;YAML语言格式 二&#xff1a;yuml 1、查看 api 资源版本标签 2、写一个yaml文件demo 3、创建service服务对外提供访问并测试 4、详解k8s中的port 三&#xff1a;文件生成 1、kubec…

Stephen Wolfram:ChatGPT 的训练

The Training of ChatGPT ChatGPT 的训练 OK, so we’ve now given an outline of how ChatGPT works once it’s set up. But how did it get set up? How were all those 175 billion weights in its neural net determined? Basically they’re the result of very large…

以指标驱动,保险、零售、制造企业开启精益敏捷运营的新范式

近日&#xff0c;以“释放数智生产力”为主题的 Kyligence 用户大会在上海前滩香格里拉大酒店成功举行。大会包含上午的主论坛和下午的 4 场平行论坛&#xff0c;并举办了闭门会议、Open Day 等活动。来自金融、零售、制造、医药等行业的客户及合作伙伴带来了超过 23 场主题演讲…

python 封装sql 增删改查连接MySQL

select * from Teacher limit 10 连接字符串配置MysqlConfig.py class MysqlConfig:HOST 192.168.56.210PORT 3306USER rootPASSWORD 1qaz0987654321DBStudentDBCHARSET utf8封装增删改查MysqlConnection.py Author: tkhywang 2810248865qq.com Date: 2023-06-19 15:44:48 Las…

嵌入式虚拟仿真实验教学平台之登录注册功能使用

登录注册功能的使用 本文将介绍嵌入式虚拟仿真实验教学平台的账号如何注册以及登录账号。 注册账号 1、首先谷歌或Edge等主浏览器中输入https://app.puliedu.com/网址&#xff0c;然后会跳转到登录页&#xff0c;如下所示: 2、点击上图中框中的新注册账号&#xff0c;跳转…

代理模式:静态代理+JDK/CGLIB 动态代理

文章目录 1. 代理模式2. 静态代理3. 动态代理3.1. JDK 动态代理机制3.1.1. 介绍 3.1.2. JDK 动态代理类使用步骤3.1.3. 代码示例3.2. CGLIB 动态代理机制3.2.1. 介绍3.2.2. CGLIB 动态代理类使用步骤3.2.3. 代码示例 3.3. JDK 动态代理和 CGLIB 动态代理对比 4. 静态代理和动态…

图扑软件入选 2023 中国信通院“铸基计划”全景图

7 月 27 日&#xff0c;由中国信通院主办的“2023 数字生态发展大会”暨中国信通院“铸基计划”年中会议在北京召开。本次大会重磅发布了《高质量数字化转型产品及服务全景图&#xff08;2023 上半年度&#xff09;》。图扑软件凭借自研 HT for Web 数字孪生可视化产品成功入选…

伦敦金费用有哪几方面?

通常在网上开设伦敦金投资账户是没有成本的&#xff0c;而它交易的费用&#xff0c;主要是由点差和过夜利息&#xff08;仓息&#xff09;构成。如果伦敦金投资者只是做短线的日内交易&#xff0c;做一手完整的100盎司的标准合约&#xff0c;需要支付大约50美元点差费用&#x…

DAY02_Spring—第三方资源配置管理Spring容器Spring注解开发Spring整合Mybatis和Junit

目录 一 第三方资源配置管理1 管理DataSource连接池对象问题导入1.1 管理Druid连接池1.2 管理c3p0连接池 2 加载properties属性文件问题导入2.1 基本用法2.2 配置不加载系统属性2.3 加载properties文件写法 二 Spring容器1 Spring核心容器介绍问题导入1.1 创建容器1.2 获取bean…

C++ 多态性——虚函数

虚函数是动态绑定的基础。虚函数必须是非静态的成员函数。虚函数经过派生之后&#xff0c;在类族中就可以实现运行过程的多态。 根据类型兼容规则&#xff0c;可以使用派生类的对象代替基类的对象。如果基类类型的指针指向派生类对象&#xff0c;就可以通过这个指针来访问该对…