news 2026/4/18 12:04:26

Qt——事件简单介绍

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Qt——事件简单介绍

目录

事件的概念

事件和处理函数的关联

鼠标移入和移出事件

题外话——自定义类如何使用QtDesigner

鼠标按下事件

鼠标释放事件

鼠标双击事件

鼠标移动事件

题外话——setMouseTracking(true)

鼠标滚动事件

键盘事件

定时器事件

窗口移动和大小改变事件


事件的概念

用户进行的各种操作,不仅会产生信号,也会产生事件。我们也可以给事件关联对应的处理函数,当事件触发时就执行到相应代码

实际上,事件本身是操作系统提供的机制Qt将操作系统的事件封装成Qt事件。尽管封装了,Qt事件使用起来还是比较复杂,所以Qt把Qt事件再封装成信号槽

不过Qt信号槽并没有提供所有用户动作对应的信号,因此我们有时还需要利用Qt事件机制来获取并处理用户动作

所有的Qt事件都继承自QEvent:


事件和处理函数的关联

之前我们使用connect函数来把信号和信号槽关联起来。而在Qt事件机制中,我们需要做的是重写Qt内置控件的事件处理函数即创建继承自Qt内置控件类的新控件类并重写其事件处理函数,我们使用新的控件类自然就绑定了新的事件处理函数。

一句话,如果想对某控件的某些事件做出处理,就使用继承自该控件的子类控件并重写父类控件中对应事件的虚函数


鼠标移入和移出事件

#include "label.h" label::label(QWidget * ptr):QLabel(ptr) { this->setStyleSheet("border: 1px solid black; padding: 5px;"); } void label::enterEvent(QEvent *event) { qDebug()<<"鼠标进来了!"; } void label::leaveEvent(QEvent *event) { qDebug()<<"鼠标出去了!"; }


题外话——自定义类如何使用QtDesigner

我们现在的需求是,使用我们自己写的继承自Qt内置控件的控件,但是QtDesigner中的控件都是Qt内置的控件。

实际上我们可以先在QtDesigner拖入一个Qt内置控件,然后将其提升为我们写的子类控件,具体做法如下:

拖入一个内置QLabel控件,通过右侧对象树可知,当前使用的不是子类控件

右击选中的控件,点击提升为:

在上面两处填写你自己编写的控件类名称以及头文件名称并点击添加:

选中出现的行,然后点击提升,我们发现对象树上的类型已经改变了:


鼠标按下事件

#include "label.h" Label::Label(QWidget * ptr):QLabel(ptr) { this->setStyleSheet("border: 1px solid black; padding: 5px;"); } void Label::mousePressEvent(QMouseEvent *event) { if(event->button() == Qt::LeftButton) { qDebug()<<"点击鼠标左键"; } else if(event->button() == Qt::RightButton) { qDebug()<<"点击鼠标右键"; } qDebug()<<"相对于lebel的坐标x:"<< event->x() <<"相对于lebel的坐标y"<<event->y(); qDebug()<<"全局坐标x:"<< event->globalX() <<"全局坐标y"<<event->globalY(); }


鼠标释放事件

#include "label.h" Label::Label(QWidget * ptr):QLabel(ptr) { this->setStyleSheet("border: 1px solid black; padding: 5px;"); } void Label::mousePressEvent(QMouseEvent *event) { if(event->button() == Qt::LeftButton) { qDebug()<<"点击鼠标左键"; } else if(event->button() == Qt::RightButton) { qDebug()<<"点击鼠标右键"; } } void Label::mouseReleaseEvent(QMouseEvent *event) { if(event->button() == Qt::LeftButton) { qDebug()<<"释放鼠标左键"; } else if(event->button() == Qt::RightButton) { qDebug()<<"释放鼠标右键"; } }


鼠标双击事件

#include "label.h" Label::Label(QWidget * ptr):QLabel(ptr) { this->setStyleSheet("border: 1px solid black; padding: 5px;"); } void Label::mousePressEvent(QMouseEvent *event) { if(event->button() == Qt::LeftButton) { qDebug()<<"点击鼠标左键"; } else if(event->button() == Qt::RightButton) { qDebug()<<"点击鼠标右键"; } } void Label::mouseReleaseEvent(QMouseEvent *event) { if(event->button() == Qt::LeftButton) { qDebug()<<"释放鼠标左键"; } else if(event->button() == Qt::RightButton) { qDebug()<<"释放鼠标右键"; } } void Label::mouseDoubleClickEvent(QMouseEvent *event) { if(event->button() == Qt::LeftButton) { qDebug()<<"双击鼠标左键"; } else if(event->button() == Qt::RightButton) { qDebug()<<"双击鼠标右键"; } }

注意:双击事件触发之前,单击事件也是被触发的


鼠标移动事件

#include "widget.h" #include "ui_widget.h" Widget::Widget(QWidget *parent) : QWidget(parent) , ui(new Ui::Widget) { ui->setupUi(this); this->setMouseTracking(true); } void Widget::mouseMoveEvent(QMouseEvent *event) { qDebug()<<"鼠标的x坐标是:"<<event->x()<<"鼠标的y坐标是:"<<event->y(); } Widget::~Widget() { delete ui; }


题外话——setMouseTracking(true)

对于鼠标移动用来说,只要我们轻轻一划,就会产生很多移动事件,这会可能会导致客户端变得卡顿。

因此,控件默认是不追踪鼠标的移动的,在这种情况下,我们只有按住鼠标左键或者右键然后移动鼠标才会触发移动事件。

想要达到鼠标只要移动就触发事件的效果就得告诉Qt你想追踪鼠标移动:setMouseTracking(true)


鼠标滚动事件

Widget::Widget(QWidget *parent) : QWidget(parent) , ui(new Ui::Widget) { ui->setupUi(this); //this->setMouseTracking(true); } void Widget::wheelEvent(QWheelEvent *event) { //显示出鼠标滚动的距离 qDebug()<<event->delta(); }


键盘事件

Widget::Widget(QWidget *parent) : QWidget(parent) , ui(new Ui::Widget) { ui->setupUi(this); } void Widget::keyPressEvent(QKeyEvent *event) { //如果用户按下了普通按键按键,会被存放到这儿 int commonkey = event->key(); //如果用户按下了ctrl,alt等特按键,会被存放到这儿 int specialkey = event->modifiers(); //Qt用数字表示按键类型,但提供给我们一些类型使用,我们使用这些类型区分出用户按下的是什么键 //可以使用specialkey和commonkey判断用户是否按下了组合键(如果两个都不为0就是组合键),按下了什么组合键 if(specialkey == Qt::ControlModifier && commonkey == Qt::Key_A) { qDebug()<<"用户按下了ctrl+A"; } else if(specialkey == 0 && commonkey == Qt::Key_A) { qDebug()<<"用户按下了A"; } else if(specialkey == 0 && commonkey == Qt::Key_B) { qDebug()<<"用户按下了B"; } }


定时器事件

Widget::Widget(QWidget *parent) : QWidget(parent) , ui(new Ui::Widget) { ui->setupUi(this); //开启控件的定时器事件并设置间隔时间,返回定时器ID timer_id = this->startTimer(1000); } //定时器时间处理函数 void Widget::timerEvent(QTimerEvent *event) { //一个控件可能有多个定时器,它们都会触发这个处理函数,所以我们用ID来区分它们 int id = event->timerId(); if(id!=timer_id) { return; } //如果倒计时小于等于0,就停止定时器 int value = ui->lcdNumber->intValue(); if(value<=0) { this->killTimer(id); return; } //否则就让倒计时-1 else { ui->lcdNumber->display(value-1); } }


窗口移动和大小改变事件

Widget::Widget(QWidget *parent) : QWidget(parent) , ui(new Ui::Widget) { ui->setupUi(this); } void Widget::moveEvent(QMoveEvent *event) { qDebug()<<event->pos(); } void Widget::resizeEvent(QResizeEvent *event) { qDebug()<<event->size(); }

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/18 3:25:53

2026年有退款保障的去AIGC痕迹工具:不达标全额退

2026年有退款保障的去AIGC痕迹工具&#xff1a;不达标全额退 花钱处理完还是不达标&#xff0c;找客服退款&#xff0c;客服说"我们不保证效果"。 我同学就遇到过这种事。100多块打水漂了&#xff0c;气死个人。 后来我选工具就只看一条&#xff1a;不达标能不能退…

作者头像 李华
网站建设 2026/4/18 3:29:01

malloc每秒百万次调用扛不住?看Nginx如何用500行代码打造零碎片内存池

一、高并发服务器的内存困局 写过高并发服务器的人,多少都被内存管理折腾过。 我之前做一个长连接网关项目的时候,压测到QPS上万就开始出问题:响应延迟波动剧烈,p99从2ms飙到50ms,GC似的卡顿周期性出现。排查了半天,最后用perf一看,30%的CPU时间花在了malloc/free上。…

作者头像 李华
网站建设 2026/4/17 13:18:02

2026年双引擎技术去AIGC痕迹:为什么效果更好

2026年双引擎技术去AIGC痕迹&#xff1a;为什么效果更好 选去AIGC痕迹工具时&#xff0c;经常看到"双引擎""多引擎"这些词。 到底什么是双引擎&#xff1f;为什么双引擎效果更好&#xff1f; 先说结论&#xff1a;双引擎技术用两套不同的处理方法&#x…

作者头像 李华
网站建设 2026/4/18 11:04:54

CANN端边云协同:从云端训练到边缘推理的全链路智能部署实战

CANN组织链接&#xff1a;https://atomgit.com/cann ops-nn仓库链接&#xff1a;https://atomgit.com/cann/ops-nn 当工业质检模型更新耗时4.2小时导致产线停摆&#xff0c;当弱网环境下模型传输失败率高达38%&#xff0c;当边缘设备推理功耗吞噬83%电池电量引发设备宕机——端…

作者头像 李华
网站建设 2026/4/18 10:08:37

C语言内存函数(二)

一、memset函数的使用1.1.memset函数void * memset ( void * ptr, int value, size_t num );1.2.memset函数的作用填充内存块将指针 ptr 所指向的内存块的前 num 个字节设置为指定的值&#xff08;解释为无符号字符&#xff09;memset是用来设置内存的&#xff0c;将内存中的值…

作者头像 李华