当前位置: 首页 > news >正文

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(); }

http://www.jsqmd.com/news/355362/

相关文章:

  • 写作小白救星 AI论文网站 千笔 VS 万方智搜AI,本科生首选!
  • Windows安装PostgreSQL
  • PHP代碼審計
  • 2026最新彩色胶企业top5推荐!国内优质彩色胶品牌权威榜单发布,资质服务双优助力高品质建材应用 - 品牌推荐2026
  • 2026冲刺用!10个一键生成论文工具测评:本科生毕业论文+开题报告高效写作指南
  • 控制算法研发工程师(博士)职位深度解析与面试指南
  • 2026年灯具品牌性价比排行(第三方实测版) - GEO排行榜
  • 摆脱论文困扰! 8个降AI率软件降AIGC网站测评:继续教育必备工具推荐
  • 深度解析:上海德梅柯汽车装备制造有限公司机器人工程师职位——技术核心、挑战与职业发展
  • 为什么制造运营管理系统必须采用语义驱动,而不能采用自然语言驱动
  • Vue3+java基于springboot框架航班售票飞机订票系统的设计与实现
  • 电子学会青少年机器人技术(五级)等级考试试卷-实际操作(2025年12月)
  • 电子学会青少年机器人技术(四级)等级考试试卷-实际操作(2025年12月)
  • Vue3+java基于springboot框架高校学生实习综合服务平台设计与实现
  • 2026开年重磅社会现实题材短漫剧重磅来袭《消失的程序员》
  • 如何在Navicat Premium中创建一个数据库?
  • 2026年一键去AIGC痕迹工具推荐:懒人必备
  • 2026.2.6 模拟赛
  • HTTP网络巩固知识基础题(2) - 实践
  • mirror_fold.py_utils_0207curso
  • 2026年知网AIGC检测又升级了?这样去痕迹才有效
  • 数字手势识别0-9检测数据集VOC+YOLO格式8738张10类别
  • 【数据结构-树与二叉树】4.3 二叉树的存储结构
  • 2026年术语保护好的去AIGC痕迹工具:专业词汇不乱改
  • 伦理委员会指南:高风险AI系统测试审批流程
  • 2026年文心一言写的内容怎么去AIGC痕迹?实测分享
  • pip 和 npm 区别
  • 2026年性价比最高的去AIGC痕迹工具:4.8元/千字
  • leetcode 916. Word Subsets 单词子集
  • 【无人机控制】基于LQR和PID控制器在风扰下对一维无人机高度稳定的控制效果附matlab代码