【Qt】深入探索Qt事件处理:从基础到高级自定义:QEvent

news2024/9/17 3:50:30

文章目录

  • 前言:
  • 1. 事件的介绍
  • 2. 事件的处理
    • 2.1. 示例1: 重写鼠标进入和鼠标离开事件
    • 2.2. 示例2:当鼠标点击时,获取对应的坐标值;
    • 2.3. 鼠标释放事件
    • 2.4. 鼠标双击事件
    • 2.5. 鼠标移动事件
    • 2.6. 鼠标滚轮的滚动事件
  • 3. 按键事件
  • 4. 定时器事件
  • 6. 窗口移动事件
  • 总结

虽然 Qt 是跨平台的 C++ 开发框架,Qt的很多能力其实是操作系统提供的,只不过 Qt 封装了系统的 API。
程序是运行在操作系统上的,需要系统给我们支撑

  1. 事件
  2. 文件操作
  3. 多线程编程
  4. 网络编程
  5. 多媒体(音频,视频)

前言:

在现代软件开发中,用户界面的交互性是衡量软件质量的重要标准之一。Qt,作为一个功能强大的跨平台C++开发框架,提供了丰富的机制来处理用户界面的各种交互事件。本文旨在深入探讨Qt中的事件处理机制,包括鼠标、键盘、定时器以及窗口事件等,并通过具体的示例代码,展示如何在Qt应用程序中重写和处理这些事件。通过本文的学习,开发者将能够更加灵活和深入地定制用户界面的行为,提升应用程序的交互性和用户体验。

1. 事件的介绍

  • 信号槽:
    用户进行的各种操作,就可能会产生出信号。可以给某个信号指定槽函数,当信号触发时,就能够自动执行到对应的槽函数。
  • 事件非常类似
    用户的各种操作,也会产生事件。程序员同样可以给事件关联上处理函数(处理的逻辑),当事件触发的时候,就能够执行对应的代码。

事件本身是操作系统提供的机制,Qt 也同样把操作系统事件机制经行了封装,拿到了 Qt 中,但是由于事件对应的代码编写起来的代码编写起来不是很方便,Qt 对事件机制又进行了进一步的封装,就得到了信号槽。
所以,信号槽就是对于事件的进一步封装,事件是信号槽的底层机制。

实际 Qt 开发过程中,绝大部分和用户之间进行的操作都是通过“信号槽”来完成的。有些特殊情况下,信号槽不一定能搞定(某个用户的动作行为,Qt
没有提供对应的信号…)此时就需要通过重写事件处理函数的形式,来手动处理事件的响应逻辑。

开发事件机制给咱们,咱们就可以根据实际需要进行更深度的定制化的 DIY 操作了。
用户进行了很多操作,就会产生很多的事件(当然也会产生很多的信号)
在这里插入图片描述
QEvent: 事件概念
子类:代表各种具体的事件。不同场景下,要关注的点是不一样的,这些事件的子类中就会包含一些对应不同的属性。

2. 事件的处理

事件的处理一般常用的方法为:重写相关的 Event 函数
在 Qt 中,几乎所有的 Event 函数都是虚函数,所以可以重新实现。如:在实现鼠标的进入和离开事件时,直接重新实现 enterEvent()leaveEvent() 即可。

让一段代码和某个事件关联起来,当事件触发的时候,就能指定到这段代码,之前信号槽这里通过 connect 来完成上述关联的,对于事件来说还不太一样。

对于事件来收,让当前类重写某个事件处理函数(这里用到的是“多态”的机制创建子类,继承自 Qt 已有的类,在子类中重写父类的事件处理函数)
后续的事件触发过程中,就会通过多态这样的机制,执行到咱们自己写的子类的函数中。

2.1. 示例1: 重写鼠标进入和鼠标离开事件

处理一下鼠标进入和鼠标离开:enterEvent()leaveEvent()
它们都是虚函数,虚函数才能被子类重写。
在这里插入图片描述
在这里插入图片描述
边框显示出来,方便观察当前鼠标是否进去离开:
在这里插入图片描述
这里需要创建 QLabel 的子类,重写 enterEventleaveEvent
在这里插入图片描述
在这里插入图片描述

// label.h
#include <QWidget>
#include <QLabel>

class Label : public QLabel
{
    Q_OBJECT
public:
    Label(QWidget* parent); // QWidget* parent 让 Label有父控件
};
// label.cpp
#include "label.h"

Label::Label(QWidget* parent) : QLabel(parent)
{

}

为创建的类,重写 enterEventleaveEvent这两个函数

void enterEvent(QEvent* event);
void leaveEvent(QEvent* event);

要想重写父类的函数,就需要保证你这边写的函数名字和函数的参数列表都完全一致(形参名无所谓),谨防单词拼写错误。 正常来说Qt Creator
应该要能够提示出来,但实际上没有,所以这里只能通过手动的方式从Qt文档把它复制过来了,以免拼写错误。

上述代码,虽然重写了这两函数,但是还是有点问题
在这里插入图片描述
当前界面上的这个 Label 其实是 QLable, 不是咱们自己写的 Label,必须确保界面上的这个 label 是一个咱们自己定义的 Label 类的实例,才会执行到。
在这里插入图片描述
在这里插入图片描述
一定要确保,类名和头文件名与自定义的是匹配的,一定不能有拼写错误!
在这里插入图片描述
通过“提升为”这样的方式,就可以把 Qt Designer 中拖上去的控件的类型转换成自定义的控件类型。
在这里插入图片描述

2.2. 示例2:当鼠标点击时,获取对应的坐标值;

mousePressEvent 这个函数,按下左键,右键,滚轮都能触发。
有的鼠标还带有前进后退侧键,也是可以触发。
但是还有的鼠标,有更多的按键,更多按键就不一定了

void Label::mousePressEvent(QMouseEvent *event)
{
    // 当前 event 对象就包含了鼠标点击位置的坐标
    qDebug() << event;
    if (event->button() == Qt::LeftButton) {
        qDebug() << "按下左键";
    } else if (event->button() == Qt::RightButton) {
        qDebug() << "按下右键";
    }
    qDebug() << event->x() << " , "<< event->y();
    // globalX 和 globalY 是以屏幕左上角为原点,获取的坐标
    qDebug() << event->globalX() << " , " << event->globalY();
}

在这里插入图片描述

2.3. 鼠标释放事件

鼠标释放事件是通过虚函数 mouseReleaseEvent() 来捕获的。

void Label::mouseReleaseEvent(QMouseEvent *event)
{
    if (event->button() == Qt::LeftButton) {
        qDebug() << "释放左键";
    } else if (event->button() == Qt::RightButton) {
        qDebug() << "释放右键";
    }
}

clicked 这样的信号,就相当于是一次鼠标按下事件和一次鼠标释放事件

2.4. 鼠标双击事件

在这里插入图片描述

当第二按下的时候,才能够识别到是“双击”
比如有的程序,可能是单击有一些逻辑,双击有另一些逻辑。
如果我们没注意,可能双击操作就能触发单击逻辑,可能就有bug

2.5. 鼠标移动事件

刚才重写鼠标的操作,都是在自定义的 Label 中完成的。此时鼠标只有在 Label 范围内进行动作的时候,才能捕捉到。
也可以把这些操作直接放到 Widget(QWidget 子类) 来完成。这样的话,鼠标在整个窗口中进行各种动作都能获取了。

// widget.h
#ifndef WIDGET_H
#define WIDGET_H

#include <QWidget>
#include <QMouseEvent>

QT_BEGIN_NAMESPACE
namespace Ui { class Widget; }
QT_END_NAMESPACE

class Widget : public QWidget
{
    Q_OBJECT

public:
    Widget(QWidget *parent = nullptr);
    ~Widget();

    void mouseMoveEvent(QMouseEvent* event);

private:
    Ui::Widget *ui;
};

#endif // WIDGET_H
// widget.cpp
#include "widget.h"
#include "ui_widget.h"
#include <QDebug>

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);

    // 把这个选项设置为 true,才能够追踪鼠标移动的位置
    this->setMouseTracking(true);
}

Widget::~Widget()
{
    delete ui;
}

void Widget::mouseMoveEvent(QMouseEvent *event)
{
    qDebug() << event->x() << "," << event->y();
}

鼠标移动不同于鼠标按下,随便移动一下鼠标,就会产生大量的鼠标移动事件。当你进行捕获事件的时候,尤其是在这里进行一些复杂逻辑的时候,程序负担就很重,很容易产生卡顿之类的情况。
Qt 为了保证系统的流畅性,默认情况下不会对鼠标移动进行追踪,鼠标移动的时候不会调用mouseMoveEvent。除非显示告诉 Qt 就要追踪鼠标位置。

2.6. 鼠标滚轮的滚动事件

在 Qt 中,鼠标滚轮事件是通过 QWheelEvent 类来实现的。滚轮滑动的距离可以通过 delta() 函数获取。

#include "widget.h"
#include "ui_widget.h"
#include <QDebug>
#include <QWheelEvent>

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
    _total = 0;
}

Widget::~Widget()
{
    delete ui;
}

void Widget::wheelEvent(QWheelEvent *event)
{
    _total += event->delta();
    qDebug() << _total;
}

3. 按键事件

Qt 中的按键事件是通过 QKeyEvent 类来实现的。当键盘上的按键被按下或者被释放时,键盘事件便会触发。
之前 QShortCut, 设置快捷键用到过键盘。这是信号槽机制封装过的,获取键盘按键的方式。
站在更底层的角度,也可以通过事件获取到当前用户键盘按下 的情况。

keyPressEvent(QKeyEvent*); 

看你捕获键盘是在哪个控件范围内

void Widget::keyPressEvent(QKeyEvent *event)
{
//    qDebug() << event;
//    qDebug() << event->key();
    if (event->key() == Qt::Key_A) {
        qDebug() << "按下了 A 键";
    }
}

如果是组合键呢?

void Widget::keyPressEvent(QKeyEvent *event)
{
//    qDebug() << event;
//    qDebug() << event->key();
    if (event->key() == Qt::Key_A && event->modifiers() == Qt::ControlModifier) {
        qDebug() << "按下了 ctrl + A 键";
    }
}

4. 定时器事件

QTimer 实现了定时器的功能,在QTimer 背后是 QTimerEvent 定时器事件进行支撑的。
QObject 提供了一个 timerEvent 这个函数。
startTimer 启动定时器
killTimer 关闭定时器

次数 timerID 类似于 Linux 上的“文件描述符”, 起到身份标识效果。

#include "widget.h"
#include "ui_widget.h"

Widget::Widget(QWidget *parent)
    : QWidget(parent)
    , ui(new Ui::Widget)
{
    ui->setupUi(this);
    // 开启定时器事件
    // 此处 timerID 是一个定时器的身份标识

    _timerID = this->startTimer(1000);
}

Widget::~Widget()
{
    delete ui;
}

void Widget::timerEvent(QTimerEvent *event)
{
    // 如果一个程序中存在多个定时器(startTimer 创建的定时器),此时每个定时器都会触发 timerEvent 函数
    // 先判定一下这次触发是否是想要的定时器触发的
    if (event->timerId() != this->_timerID) {
        // 如果不是我们的定时器触发的,就直接忽略
        // 当前程序中只有这一个定时器
        return;
    }
    // 是咱们自己搞的定时器
    int value = ui->lcdNumber->intValue();
    if (value <= 0) {
        // 停止定时器
        this->killTimer(this->_timerID);
        return;
    }
    value -= 1;
    ui->lcdNumber->display(value);
}

使用 timerEventQTimer 还是要更复杂一点。手动管理 timerID, 还要区分这次函数调用是哪个 timer 引起,后续还是使用 QTimer 即可。

6. 窗口移动事件

moveEvent 窗口移动时触发的事件。
resizeEvent 窗口大小改变时触发的事件。

void Widget::moveEvent(QMoveEvent *event)
{
    qDebug() << event->pos();
}

void Widget::resizeEvent(QResizeEvent *event)
{
    qDebug() << event->size();
}

事件分发 / 事件过滤,属于 Qt 事件机制背后的一些逻辑,Qt 也把这部分内容提供一些 API 让程序员有更多的可操作空间。
事件分发:重写 event 函数,直接获取到所有的事件(杀伤力比较广,不当使用可能对现有的逻辑(现有的事件体系造成一些负面影响))。
当然,有些场景中,比如要禁用用户的某些操作,可以考虑使用事件过滤器机制。
最后还要要优先使用信号槽机制,因为这都是Qt给我们封装好了让我们去使用的。

总结

本文详细介绍了Qt框架中的事件处理机制,包括事件的基本概念、事件的处理方法以及具体的事件类型和处理示例。我们了解到,Qt通过封装操作系统的事件机制,提供了信号槽这一更加方便的事件处理方式。然而,在某些特殊场景下,我们可能需要通过重写事件处理函数来进行更深层次的自定义。文中通过鼠标事件、按键事件、定时器事件以及窗口事件等多个方面的示例,展示了如何在Qt中实现这些事件的处理。

我们学习了如何通过重写enterEvent()leaveEvent()来处理鼠标的进入和离开事件,如何通过mousePressEvent()mouseReleaseEvent()来捕获鼠标的点击和释放,以及如何通过mouseMoveEvent()wheelEvent()来追踪鼠标的移动和滚轮的滚动。此外,还探讨了按键事件的捕获和处理,定时器事件的启动和停止,以及窗口移动和大小改变事件的处理。

最后,文章强调了在使用事件处理机制时,应当优先考虑Qt提供的信号槽机制,因为它更为简洁和安全。但在需要更细致控制的场景下,事件处理函数的重写和事件过滤机制提供了更多的灵活性。通过本文的学习,开发者应该能够更加熟练地运用Qt的事件处理机制,创建出更加丰富和响应灵敏的用户界面。

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

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

相关文章

后端经典三层架构

大家好&#xff0c;这里是教授.F 引入&#xff1a; MVC 全称∶ Model 模型、View 视图、 Controller 控制器。MVC 最早出现在 JavaEE 三层中的 Web 层&#xff0c;它可以有效的指导WEB 层的代码如何有效分离&#xff0c;单独工作。 View 视图∶只负责数据和界面的显示&#…

Python I/O操作笔记

打开文件&#xff1a; 使用 open() 函数&#xff0c;其中文件路径可以是相对路径或绝对路径。 模式除了常见的 r&#xff08;只读&#xff09;、w&#xff08;写入&#xff0c;会覆盖原有内容&#xff09;、a&#xff08;追加&#xff09;外&#xff0c;还有一些其他组合模式&…

小度推出全球首款基于文心大模型的学习机Z30,仅售价6699元

5月27日&#xff0c;小度科技召开新品发布会&#xff0c;全球首款基于文心大模型的学习机——小度学习机Z30重磅发布。 据「TMT星球」了解&#xff0c;该产品基于文心大模型&#xff0c;重新定义了“AI老师”的能力边界&#xff0c;不仅是一款能为孩子提供全面、有效学习辅导的…

LINUX环境基础练习题(附带答案)

&#x1f525; 交流讨论&#xff1a;欢迎加入我们一起学习&#xff01; &#x1f525; 资源分享&#xff1a;耗时200小时精选的「软件测试」资料包 &#x1f525; 教程推荐&#xff1a;火遍全网的《软件测试》教程 &#x1f4e2;欢迎点赞 &#x1f44d; 收藏 ⭐留言 &#x1…

嵩山为什么称为五岳之尊

在此之前&#xff0c;人们心目中的五岳之尊一般是东岳泰山。自此以后&#xff0c;观点一定会改变&#xff1a;五岳之尊是中岳嵩山&#xff01;且听我慢慢道来。 首先将二者进行一下对比—— 中与东的对比&#xff0c;嵩山居中&#xff0c;泰山居东。东方是太阳升起的地方&#…

云原生Kubernetes: 云主机部署K8S 1.30版本 单Master架构

目录 一、实验 1.环境 2.Termius连接云主机 3.网络连通性与安全机制 4.云主机部署docker 5.云主机配置linux内核路由转发与网桥过滤 6.云主机部署cri-dockerd 7.云主机部署kubelet,kubeadm,kubectl 8.kubernetes集群初始化 9.容器网络&#xff08;CNI&#xff09;部署…

牛客NC67 汉诺塔问题【中等 递归 Java/Go/PHP/C++】 lintcode 169 · 汉诺塔

题目 题目链接&#xff1a; https://www.nowcoder.com/practice/7d6cab7d435048c4b05251bf44e9f185 https://www.lintcode.com/problem/169/ 思路 相传在古印度圣庙中&#xff0c;有一种被称为汉诺塔(Hanoi)的游戏。该游戏是在一块铜板装置上&#xff0c;有三根杆(编号A、B、C…

书生·浦语大模型全链路开源体系-作业1

视频链接&#xff1a;书生浦语大模型全链路开源体系_哔哩哔哩_bilibili 1. LLM发展 LLM是近年来人工智能领域的一个重要发展方向。大型语言模型的历史可以追溯到2017年,当时OpenAI推出了GPT-1(Generative Pre-trained Transformer)模型,这是一个基于Transformer架构的语言生成…

论文阅读》学习了解自己:一个粗略到精细的个性化对话生成的人物感知训练框架 AAAI 2023

《论文阅读》学习了解自己&#xff1a;一个粗略到精细的个性化对话生成的人物感知训练框架 AAAI 2023 前言 简介研究现状任务定义模型架构Learning to know myselfLearning to avoid Misidentification损失函数实验结果消融实验 前言 亲身阅读感受分享&#xff0c;细节画图解释…

28【Aseprite 作图】苹果——拆解

1 画苹果框架 左边:第一行 7 第二行 2 第三 四行1 竖着7行 竖着2行 竖着1 、1 行 横着2个 横着4个 苹果可以是左右对称的,完成上述后,水平翻转到右边 2 枝叶 第一行1 左边 2 3 4 行,各1 第5行,竖着4个 再横着3个 右边 竖着3个,然后斜着2个,然后斜着1个 最上面的,两个…

RTDETR结合CVPR2024最新图像增强算法!让你的模型无惧风雨【含端到端推理脚本】

如何有效地探索雨痕的多尺度表示对于图像去雨是很重要的。与现有的基于Transformer的方法相比,这些方法主要依赖于单一尺度的雨痕外观,我们开发了一个端到端的多尺度Transformer,利用各种尺度中潜在有用的特征来促进高质量的图像重建。为了更好地探索空间变化的雨痕的常见退…

九宫格转圈圈抽奖活动,有加速,减速效果

在线访问demo和代码在底部 代码&#xff0c;复制就可以跑 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><tit…

2023idea没有VCS首次提交代码到Git

1、setting 2、vcs------>create git repository 3、右键项目----->Git------>add 4、右键项目------>git------>commit Directory 之后就会显示这个页面(下面写你提交的信息&#xff0c;就是你修改了什么) 点击commit,提交 5、Git--------->push 6、选择…

C++基础:类的继承,public,private,protected

三种继承模式 在上图中: 派生继承 三种继承模式 protected模式中 父类的公有属性和保护属性的成员在子类中都会变为保护属性,只能通过父类或者子类的成员函数调用. 代码示例: #include <iostream> #include <string> using namespace std; //protected class per…

numpy-mkl的下载地址

不要使用pip3直接在终端安装&#xff0c;因为pip3默安装的是numpy&#xff0c;而不是numpymkl。 采用在第三方库中手动下载后&#xff0c;再安装的方式。 第三方库网址&#xff1a;https://www.lfd.uci.edu/~gohlke/pythonlibs/#numpy 如果不能进入就可以选择去git里面&#x…

代码随想录算法训练营第四十一天 | 理论基础、509. 斐波那契数、70. 爬楼梯、746. 使用最小花费爬楼梯

理论基础 代码随想录 视频&#xff1a;从此再也不怕动态规划了&#xff0c;动态规划解题方法论大曝光 &#xff01;| 理论基础 |力扣刷题总结| 动态规划入门_哔哩哔哩_bilibili 动归五部曲 1.dp数组以及下标的含义 2.递推公式 3.dp数组如何初始化 4.遍历顺序(例如先背包再…

Python 求积分

文章目录 Part.I IntroductionPart.II 基础知识Chap.I 特殊量的表示Chap.II integrate 函数简析 Part.III 实例Chap.I 一重积分Chap.II 二重积分 Reference Part.I Introduction 高中及大学时代&#xff0c;苦『积分』久矣&#xff0c;当时如掌握了此『黑科技』&#xff0c;课…

【MiniCPM-V】win10本地部署OCR等性能测试

性能尝试 本地配置如下 --------------------------------------------------------------------------------------- | NVIDIA-SMI 546.80 Driver Version: 546.80 CUDA Version: 12.3 | |-----------------------------------------------------…

设计模式17——模板方法模式

写文章的初心主要是用来帮助自己快速的回忆这个模式该怎么用&#xff0c;主要是下面的UML图可以起到大作用&#xff0c;在你学习过一遍以后可能会遗忘&#xff0c;忘记了不要紧&#xff0c;只要看一眼UML图就能想起来了。同时也请大家多多指教。 模板方法模式&#xff08;Temp…

Java面试八股之自旋是什么意思

Java中的自旋是什么意思 自旋是多线程编程中的一种同步机制&#xff0c;尤其在Java中与锁的实现密切相关。当一个线程尝试获取某个锁&#xff08;如内置锁或显式锁&#xff09;时&#xff0c;如果锁已被其他线程持有&#xff0c;通常的做法是将该线程置于阻塞状态&#xff0c;…