Qt事件处理和传递流程
事件系统的概述
事件的类型
Qt 支持多种事件类型,每种类型代表不同的用户交互或系统事件。常见的事件类型包括:
- 输入事件:如鼠标事件(QMouseEvent)、键盘事件(QKeyEvent)。
- 窗口事件:如窗口大小变化事件(QResizeEvent)、窗口关闭事件(QCloseEvent)。
- 定时器事件:如定时器超时事件(QTimerEvent)。
- 绘图事件:如绘图事件(QPaintEvent)。
- 自定义事件:可以通过继承 QEvent 类创建自定义事件。
QEvent 类
QEvent 类是所有事件的基类。每个事件类型都继承自 QEvent 并扩展了特定的功能。常见的事件类型及其子类有:
- QMouseEvent:处理鼠标相关事件,如鼠标点击、移动、释放等。
- QKeyEvent:处理键盘输入事件,如按键按下和释放。
- QResizeEvent:处理窗口大小变化事件。
- QCloseEvent:处理窗口关闭事件。
- QPaintEvent:处理绘图事件。
- QTimerEvent:处理定时器超时事件。
自定义事件
可以创建自定义事件类型,通过继承 QEvent 类并定义自己的数据和处理逻辑。
class MyCustomEvent : public QEvent {
public:
static const QEvent::Type MyEventType = static_cast<QEvent::Type>(QEvent::User + 1);MyCustomEvent() : QEvent(MyEventType) {}// 自定义数据和方法
QString message;
};
事件处理机制
事件处理函数(event handlers)
每个 QWidget 子类都有一个 event() 方法,它处理所有传递给该对象的事件。Qt 提供了特定类型事件的处理函数,例如 mousePressEvent()、keyPressEvent() 等。这些特定的处理函数通常在 event() 函数中被调用。
void MyWidget::mousePressEvent(QMouseEvent *event) {if (event->button() == Qt::LeftButton) {// 处理鼠标左键按下事件}
}
信号与槽机制
信号与槽机制是 Qt 的核心特性之一,它提供了一种对象之间通信的方式,非常适合用于实现观察者模式。
信号
信号是由对象在特定情况下发出的消息。例如,当按钮被点击时,QPushButton 对象会发出 clicked() 信号。
槽
槽是一个可以被信号连接的函数。当信号被发出时,所有连接到该信号的槽都会被调用。槽可以是任意的成员函数、静态函数,甚至是 lambda 表达式。
连接信号与槽
使用 QObject::connect() 方法将信号连接到槽:
connect(button, &QPushButton::clicked, this, &MyWidget::onButtonClicked);void MyWidget::onButtonClicked() {// 处理按钮点击事件
}
事件传递和处理过程
事件循环
事件循环是 Qt 应用程序的核心部分。它负责调度和分发事件,使得应用程序能够响应用户输入和其他事件。
Qt 应用程序的事件循环由 QCoreApplication::exec() 方法启动。该方法进入一个无限循环,等待事件发生,并将其分发到合适的处理函数。
int main(int argc, char *argv[]) {QApplication app(argc, argv);MyWidget w;w.show();return app.exec();
}
事件在父子窗口之间的传递遵循对象的层次结构。以下是详细的过程:
事件生成:
- 事件(如鼠标点击、键盘输入等)由操作系统或 Qt 自身生成,并加入事件队列。
事件分发:
- QCoreApplication::notify 将事件分发给目标对象(子窗口或父窗口)。
事件过滤器:
- 在 event() 函数中,事件首先传递给事件过滤器。如果事件过滤器处理了事件,则事件传递过程结束。
子窗口处理事件:
- 如果事件首先传递给子窗口,子窗口的 event() 函数会尝试处理该事件。如果子窗口处理了该事件,则事件传递过程结束。
传递给父窗口:
- 如果子窗口未处理该事件,事件会传递给父窗口。父窗口的 event() 函数会尝试处理该事件,或者进一步传递给父窗口的事件过滤器。
逐层传递:
- 这一过程会逐层向上,直到事件被处理或到达顶层窗口(即没有父窗口的窗口)。
notify 函数
notify 函数是事件传递的核心,它在事件循环中被调用,用于将事件分发到相应的对象。
bool QCoreApplication::notify(QObject *receiver, QEvent *event) {if (receiver == nullptr) {qWarning("QCoreApplication::notify: Unexpected null receiver");return false;}// Before delivering the event, we pass it through the event filtersif (receiver->isWidgetType() && static_cast<QWidget*>(receiver)->testAttribute(Qt::WA_SetCursor)) {// Special handling for widgets with set cursor attribute}// Call the event filtersif (receiver->d_func()->sendThroughObjectEventFilters(receiver, event)) {return true;}// Deliver the eventreturn receiver->event(event);
}
在 notify 函数中,事件传递过程如下:
事件过滤器:
- 首先,事件被传递给目标对象的事件过滤器。
- 通过 sendThroughObjectEventFilters 方法调用所有安装在目标对象上的事件过滤器。
- 如果任何一个事件过滤器处理了事件并返回 true,事件传递过程就会终止,notify 返回 true。
事件分发:
- 如果事件过滤器没有处理事件,则调用目标对象的 event 函数。
- 具体的事件处理函数(如 mousePressEvent、keyPressEvent 等)在 event 函数内部被调用。
事件过滤器filter
事件过滤器允许对象在事件到达其目标对象之前对其进行拦截和处理。事件传递过程中,事件过滤器是一个重要的环节。通过在父窗口上安装事件过滤器,可以拦截和处理传递给子窗口的事件。
class MyFilter : public QObject {
Q_OBJECT
protected:
bool eventFilter(QObject *obj, QEvent *event) override {if (event->type() == QEvent::MouseButtonPress) {// 处理鼠标按下事件qDebug() << "Mouse button pressed in object:" << obj;return true; // 事件已被处理,不再传递}return QObject::eventFilter(obj, event); // 传递给默认处理程序
}
};// 在父窗口上安装事件过滤器
MyFilter *filter = new MyFilter;
parentWidget->installEventFilter(filter);
event() 函数处理
每个 QWidget 都有一个 event() 函数,它处理传递给这个窗口部件的所有事件。这个函数会根据事件的类型调用相应的事件处理函数。
bool QWidget::event(QEvent *event) {switch (event->type()) {case QEvent::MouseButtonPress:mousePressEvent(static_cast<QMouseEvent *>(event));break;case QEvent::KeyPress:keyPressEvent(static_cast<QKeyEvent *>(event));break;// 处理其他事件类型default:return QObject::event(event);}return true;
}
事件处理函数Handler
事件处理函数是用户能够控制的的,最底层的事件处理流程,Qt默认实现了很多事件处理函数,用户可以在子类重写这些事件处理函数。
处理鼠标事件
通过重载 mousePressEvent 方法处理鼠标点击事件。
void MyWidget::mousePressEvent(QMouseEvent *event) {if (event->button() == Qt::LeftButton) {qDebug() << "Left mouse button pressed at" << event->pos();}
}
处理键盘事件
通过重载 keyPressEvent 方法处理键盘按键事件。
void MyWidget::keyPressEvent(QKeyEvent *event) {if (event->key() == Qt::Key_Escape) {qDebug() << "Escape key pressed";close(); // 关闭窗口}
}
总结
在 Qt 中,事件在父子窗口之间的传递和处理过程遵循对象的层次结构,通过 notify 函数、事件过滤器和 event() 函数进行管理。事件可以从子窗口传递到父窗口,也可以在父窗口中拦截和处理,确保事件能够在适当的地方被正确处理。
相关文章:
Qt事件处理和传递流程
事件系统的概述 事件的类型 Qt 支持多种事件类型,每种类型代表不同的用户交互或系统事件。常见的事件类型包括: 输入事件:如鼠标事件(QMouseEvent)、键盘事件(QKeyEvent)。窗口事件ÿ…...
基于STM32移植U8g2图形库——OLED显示(HAL库)
文章目录 一、U8g2简介1、特点2、U8g2的使用步骤 二、I2C相关介绍1、I2C的基本原理2、I2C的时序协议 三、OLED屏的工作原理四、汉字点阵显示原理五、建立STM32CubeMX工程六、U8g2移植1、U8g2源码2、移植过程 七、代码编写1、参考博主实现的U82G的demo例程(1…...
C语言概述与历史
引言 C语言是一门历史悠久且影响深远的编程语言。它不仅为后继的许多编程语言奠定了基础,同时因其高效性和灵活性在系统编程和嵌入式开发领域得到了广泛应用。本篇文章将全面介绍C语言的起源与发展、设计目标与理念,以及C语言的标准演化历程,…...
钉钉Stream模式推送程序环境部署
python3.10版本需要openssl1.1.1及以上版本 参考链接:https://blog.csdn.net/weixin_42806458/article/details/110678710 wget https://www.openssl.org/source/openssl-1.1.1q.tar.gz unzip openssl-1.1.1q.tar.gz cd openssl-1.1.1q ./config --prefix/usr/loc…...
c# 二维图形绘制实践
1.等边三角形 1.1 概述 1.2 代码 using System; using System.Drawing; using System.Windows.Forms;public partial class TriangleForm : Form {public TriangleForm(){//InitializeComponent();// 确保窗体大小足够大,以容纳三角形 this.ClientSize new Siz…...
Nvidia TensorRT系列01-TensorRT的功能1
Nvidia TensorRT系列01-TensorRT的功能1 B站:肆十二-的个人空间-肆十二-个人主页-哔哩哔哩视频 (bilibili.com) 博客:肆十二-CSDN博客 问答:(10 封私信 / 72 条消息) 肆十二 - 知乎 (zhihu.com) C和Python API TensorRT的API同时支持C和Pyth…...
Vatee万腾平台:创新科技,助力企业腾飞
在全球化竞争日益激烈的今天,企业如何借助科技力量实现转型升级,已成为摆在众多企业家面前的重大课题。Vatee万腾平台凭借其卓越的创新科技和专业的服务能力,成为众多企业实现腾飞的得力助手。 一、创新科技,引领企业前行 Vatee万…...
搭建k8s集群报错unknown command “\u00a0“ for “kubeadm init“
搭建k8s报错unknown command “\u00a0” for “kubeadm init” 网上搜了一下,是因为复制过来的命令前面包含了空格,将复制的命令放到idea可以清楚看到几个命令前面有空格,删除掉就好了,记录一下...
【数据结构】三路快速排序
1. 简介 传统快速排序用的是双路快速排序,即将大于基准值的部分放到基准值右侧,小于基准值的部分放到基准值左侧,但是这种算法面对过多的重复数据的数组,时间复杂度会增多,于是就有了三路快速排序的思想,其…...
中国菜刀,蚁剑,哥斯拉,冰蝎的流量特征区别
中国菜刀、蚁剑、哥斯拉、冰蝎这四种Webshell连接工具的流量特征各有区别,以下是它们之间的主要差异: 中国菜刀(CaiDao) 流量特征: 请求包: UA头可能伪装为百度、火狐等浏览器的User-Agent。请求体中存在…...
华为OD刷题C卷 - 每日刷题32(执行任务赚积分,计算三叉搜索树的高度)
1、(执行任务赚积分): 这段代码是解决“执行任务赚积分”的问题。它提供了一个Java类Main,其中包含main方法和getResult方法,用于计算在有限的时间内,处理任务可以获得的最多积分。 main方法首先读取任务…...
QT系列教程(11) TextEdit实现Qt 文本高亮
文本高亮 对于textedit里录入的部分单词我们可以实现高亮,实现高亮主要依赖于QSyntaxHighlighter。 我们先创建一个Qt Application类,类名MainWindow, 然后新增一个C类,类名为MySyntaxHighlighter。 #ifndef MYSYNTAXHIGHLIGHTER_H #define …...
蓝队-溯源技巧
溯源技巧 大致思想 通常情况下,接到溯源任务时,获得的信息如下 攻击时间 攻击 IP 预警平台 攻击类型 恶意文件 受攻击域名/IP其中攻击 IP、攻击类型、恶意文件、攻击详情是溯源入手的点。 通过攻击类型分析攻击详情的请求包,看有没有攻击者…...
【5】JDK、JRE和JVM的区别与联系
JDK、JRE和JVM的区别与联系 Java是一种广泛使用的编程语言,它的跨平台特性得益于Java虚拟机(JVM)。然而,在Java的世界里,JDK、JRE和JVM这三个术语常常让人感到困惑。本文将阐述它们各自的功能,以及它们是如…...
【DevOps】Logstash详解:高效日志管理与分析工具
在现代软件开发和运维过程中,日志管理与分析是至关重要的环节。日志可以帮助我们追踪系统行为、诊断问题、优化性能以及确保安全合规。Logstash,作为ELK Stack(Elasticsearch、Logstash、Kibana)的核心组件之一,是一个…...
Vue3 之 Pinia 核心概念(八)
核心概念 State:这是你的应用程序的状态,是一个响应式的对象。 Getters:类似于 Vuex 中的 getters,它们是基于 state 的计算属性。 Actions:类似于 Vuex 中的 mutations 和 actions,它们用于改变 state。但…...
【办公类-04-03】华为助手导出照片视频分类(根据图片、视频的文件名日期分类导出)
背景需求: 用华为手机助手导出的照片视频,只能将jpg照片(exifread读取图片的exif拍摄日期,Png、JPEG、mp4都无法识别到exif信息) 【办公类-04-02】华为助手导出照片(jpg)读取拍摄时间分类导出…...
TVBOX 最新版下载+视频源教程
下载链接 wx 搜索 Geek 前端 发送电视资源进行获取 操作教程...
2024年了,苹果可以通话录音了
人不走空 🌈个人主页:人不走空 💖系列专栏:算法专题 ⏰诗词歌赋:斯是陋室,惟吾德馨 6月11日凌晨,苹果在WWDC24大会上,密集输出了酝酿多时的AI应用更新。苹果对通话、对话、图…...
书生·浦语大模型实战营第二期作业五
1、开发机创建conda环境: 2、安装第三方库: 3、新建pipeline_transformer.py文件,并运行: 4、运行结果: 5、执行模型: 6、与大模型进行对话: 7、默认占有的显存: 8、--cache-max-en…...
别再被‘小样本’难倒了!用Python的PyMC3库实战层次贝叶斯模型
用PyMC3解锁小样本分析:层次贝叶斯建模实战指南 当你的数据集像便利店冰柜里的酸奶——每个品类只有零星几瓶时,传统统计方法往往会束手无策。想象你正分析20个城市的新开门店周销售额,每个城市却只有3-5条数据记录。这时,层次贝叶…...
luci-app-unblockneteasemusic 插件完整技术指南:实现网易云音乐播放限制解除
luci-app-unblockneteasemusic 插件完整技术指南:实现网易云音乐播放限制解除 【免费下载链接】luci-app-unblockneteasemusic [OpenWrt] 解除网易云音乐播放限制 项目地址: https://gitcode.com/gh_mirrors/lu/luci-app-unblockneteasemusic luci-app-unblo…...
联想M920x黑苹果配置指南:从硬件适配到性能优化的完整方案
联想M920x黑苹果配置指南:从硬件适配到性能优化的完整方案 【免费下载链接】M920x-Hackintosh-EFI Hackintosh Opencore EFIs for M920x 项目地址: https://gitcode.com/gh_mirrors/m9/M920x-Hackintosh-EFI 联想M920x作为一款紧凑型商用主机,通过…...
当心“Pin-to-Pin兼容“陷阱:ICM-42688国产替代芯片深度拆解与避坑指南
两句话总结:近期TDK ICM-42688-P价格暴涨至百元且一芯难求,立创商城上出现了华轩阳、Tokmas等"国产替代"。本文通过详细对比三家datasheet数据手册,揭示所谓"兼容"背后的软件陷阱与性能差异。结论可能出乎你意料…...
GSMA:运营商实践AI大模型赋能垂直行业标杆案例集 2025
这份《运营商实践 AI 大模型赋能垂直行业标杆案例集 2025》由 GSMA 发布,聚焦客户服务与运营创新、医疗健康与智慧教育、产业升级与智能制造、公共服务与社会治理四大领域,系统梳理了中国移动、中国电信、中国联通三大运营商携手生态伙伴,将 …...
游戏界面开发与UI框架:零基础上手卡牌游戏界面开发与性能调优
游戏界面开发与UI框架:零基础上手卡牌游戏界面开发与性能调优 【免费下载链接】UiCard Generic UI for card games like Hearthstone, Magic Arena and Slay the Spire... 项目地址: https://gitcode.com/gh_mirrors/ui/UiCard 问题诊断:卡牌UI开…...
NSC_BUILDER:Switch游戏文件管理的全能解决方案
NSC_BUILDER:Switch游戏文件管理的全能解决方案 【免费下载链接】NSC_BUILDER Nintendo Switch Cleaner and Builder. A batchfile, python and html script based in hacbuild and Nuts python libraries. Designed initially to erase titlerights encryption fro…...
移动端视频适配难题:xgplayer的CSS全屏模式实战指南(含16:9与9:16适配技巧)
移动端视频适配难题:xgplayer的CSS全屏模式实战指南(含16:9与9:16适配技巧) 在移动端视频播放场景中,屏幕比例适配一直是开发者面临的棘手问题。传统全屏模式在处理非常规比例视频(如竖屏9:16内容)时往往表…...
建行江门市分行:量身定制金融策 陈皮产业绽新姿
“前期承包土地、购买柑苗已投入大量资金,后续还要设法购买化肥。”眼看资金接续不上,前期投入面临打水漂,流动资金短缺让江门新会某陈皮庄园负责人老李一筹莫展。 获悉老李困境后,建行广东江门分行网点客户经理驱车前往果园实地走…...
java rabbitmq实现消息协作
场景:数据下载采用rpa实现,数据服务采用java springboot实现,需要进行一键数据补录操作1、设置消息承载的通信队列,java 发送任务到rabbitmq和rpa端收到消息(neimeng_data_download)后,将下载结…...
