Qt 基础组件速学 事件过滤器

news2024/10/5 14:41:32

学习目标:理解事件过滤器

前置环境

运行环境:qt creator 4.12

学习内容和效果演示:

Qt 提供了事件过滤器的机制,允许我们在事件到达目标对象之前对事件进行拦截和处理。这在以下情况下非常有用:

  1. 全局事件处理: 我们可以在应用程序级别安装一个事件过滤器,拦截所有控件的事件,实现全局性的事件处理逻辑。

  2. 事件委托: 我们可以让某个对象代理其他对象的事件处理,比如父窗口处理子控件的事件。

  3. 事件转发: 我们可以将事件转发给其他对象处理,比如将鼠标事件转发给子控件处理。

理解总结:在对象和事件增加中间层,实现对象和事件是n:1的形式

使用事件过滤器的步骤如下:

  1. 创建一个事件过滤器对象,实现 bool eventFilter(QObject*, QEvent*) 函数。该函数用于处理被过滤的事件。

  2. 在需要过滤事件的对象上安装事件过滤器,使用 installEventFilter() 函数。

  3. 在事件过滤器的 eventFilter() 函数中处理事件,并返回 true 表示事件已经被处理,否则返回 false 让事件继续传播。

实验

想要实现下列效果,正常思路是给这三个img分别写对应的鼠标事件,功能效果几乎一样。

为了代码简化,我想让这三个img的事件都面向一个事件,则可使用中间层 事件过滤器。

 

 详细主要代码

#include "mainwindow.h"

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
{
    img1.load("C:/Users/jbj/Pictures/test/a.jpg");
    img2.load("C:/Users/jbj/Pictures/test/b.jpg");
    img3.load("C:/Users/jbj/Pictures/test/c.jpg");
    jpg1 =new QLabel;
    jpg2 =new QLabel;
    jpg3 =new QLabel;

    jpg1->setAlignment(Qt::AlignCenter);
    jpg1->setPixmap(QPixmap::fromImage(img1));

    jpg2->setAlignment(Qt::AlignCenter);
    jpg2->setPixmap(QPixmap::fromImage(img2));

    jpg3->setAlignment(Qt::AlignCenter);
    jpg3->setPixmap(QPixmap::fromImage(img3));

    QHBoxLayout* h=new QHBoxLayout;
    h->addWidget(jpg1);
    h->addWidget(jpg2);
    h->addWidget(jpg3);

    QWidget *wgt=new QWidget(this);
    QVBoxLayout* v= new QVBoxLayout(wgt);
    v->addLayout(h);
    status =new QLabel("鼠标按键提示信息!");
    status->setAlignment(Qt::AlignCenter);
    v->addWidget(status);

    setCentralWidget(wgt);//中心控件

    setGeometry(200,200,600,300);

    // 安装事件过滤
    jpg1->installEventFilter(this);
    jpg2->installEventFilter(this);
    jpg3->installEventFilter(this);

}

MainWindow::~MainWindow()
{
}
/*
 * 这个函数是 QObject 类的一个虚函数,需要在自定义的事件过滤器类中进行重写。
 * 它接受两个参数:
QObject* obj: 事件发生的目标对象。QEvent* event: 正在被处理的事件对象。
函数返回值:
如果事件过滤器处理了这个事件,返回 true。
如果事件过滤器没有处理这个事件,返回 false。这样事件就会继续传播给目标对象。
*/
bool MainWindow::eventFilter(QObject* watch,QEvent*e)   {
    if (watch  == jpg1){
        if(e->type() == QEvent::MouseButtonPress){// 鼠标移动区域事件,发生在窗户区域
            QMouseEvent *m =(QMouseEvent*)e;
            if(m->buttons() & Qt::LeftButton) status->setText("鼠标左键被按下:[左边图片]");
            if(m->buttons() & Qt::MidButton) status->setText("鼠标中键被按下:[左边图片]");
            if(m->buttons() & Qt::RightButton) status->setText("鼠标右键被按下:[左边图片]");

            //对当前图片进行放大2倍 创建了一个 QMatrix 对象,用于存储图像变换矩阵。
            QMatrix matrix;
            matrix.scale(2,2);
            QImage img =img1.transformed(matrix);
            jpg1->setPixmap(QPixmap::fromImage(img));
        }
        if(e->type()== QEvent::MouseButtonRelease)//鼠标松开事件
        {
            status->setText("鼠标按键已经释放:[左边图片]");
            jpg1->setPixmap(QPixmap::fromImage(img1));
        }

    }

    if (watch  == jpg2){
        if(e->type() == QEvent::MouseButtonPress){// 鼠标移动区域事件,发生在窗户区域
            QMouseEvent *m =(QMouseEvent*)e;
            if(m->buttons() & Qt::LeftButton) status->setText("鼠标左键被按下:[中间图片]");
            if(m->buttons() & Qt::MidButton) status->setText("鼠标中键被按下:[中间图片]");
            if(m->buttons() & Qt::RightButton) status->setText("鼠标右键被按下:[中间图片]");

            //对当前图片进行放大2倍
            QMatrix matrix;
            matrix.scale(2,2);
            jpg2->setPixmap(QPixmap::fromImage(img2.transformed(matrix)));
        }
        if(e->type()== QEvent::MouseButtonRelease)//鼠标松开事件
        {
            status->setText("鼠标按键已经释放:[中间图片]");
            jpg2->setPixmap(QPixmap::fromImage(img2));
        }

    }

    if (watch  == jpg3){
        if(e->type() == QEvent::MouseButtonPress){// 鼠标移动区域事件,发生在窗户区域
            QMouseEvent *m =(QMouseEvent*)e;
            if(m->buttons() & Qt::LeftButton) status->setText("鼠标左键被按下:[右边图片]");
            if(m->buttons() & Qt::MidButton) status->setText("鼠标中键被按下:[右边图片]");
            if(m->buttons() & Qt::RightButton) status->setText("鼠标右键被按下:[右边图片]");

            //对当前图片进行放大2倍
            QMatrix matrix;
            matrix.scale(2,2);
            jpg3->setPixmap(QPixmap::fromImage(img3.transformed(matrix)));
        }
        if(e->type()== QEvent::MouseButtonRelease)//鼠标松开事件
        {
            status->setText("鼠标按键已经释放:[右边图片]");
            jpg3->setPixmap(QPixmap::fromImage(img3));
        }

    }
    return QMainWindow::eventFilter(watch,e);
}

总结:

Qt 事件过滤器的主要特点和使用场景:

  1. 事件拦截和处理:

    • 事件过滤器允许我们在事件到达目标对象之前对事件进行拦截和处理。
    • 我们可以在 eventFilter() 函数中检查和处理各种类型的事件,如鼠标事件、键盘事件、paint 事件等。
  2. 自定义事件处理逻辑:

    • 通过事件过滤器,我们可以实现各种自定义的事件处理逻辑,如全局性的事件处理、事件委托和事件转发等。
    • 这在需要跨控件或跨对象进行事件处理时非常有用。
  3. 安装和使用:

    • 我们需要创建一个自定义的事件过滤器类,并实现 eventFilter() 函数。
    • 然后在需要过滤事件的对象上安装事件过滤器,使用 installEventFilter() 函数。
  4. 常见使用场景:

    • 全局事件处理: 在应用程序级别安装事件过滤器,拦截所有控件的事件,实现全局性的事件处理逻辑。
    • 事件委托: 让某个对象代理其他对象的事件处理,比如父窗口处理子控件的事件。
    • 事件转发: 将事件转发给其他对象处理,比如将鼠标事件转发给子控件处理。
  5. 返回值含义:

    • 如果事件过滤器处理了这个事件,返回 true
    • 如果事件过滤器没有处理这个事件,返回 false。这样事件就会继续传播给目标对象。

总之,事件过滤器是 Qt 提供的一个强大的机制,可以帮助我们实现各种复杂的事件处理逻辑。它的核心是 eventFilter() 函数,通过对这个函数的实现,我们可以根据需求灵活地拦截和处理事件。

最后附上源代码链接
对您有帮助的话,帮忙点个star

28-event-fillter · jbjnb/Qt demo - 码云 - 开源中国 (gitee.com)

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

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

相关文章

从文本到安全图像:自动提示优化防止不当内容生成

T2I生成技术已经得到了广泛关注,并见证了如GLIDE、Imagen、DALL-E 2、Stable Diffusion等大型生成模型的发展。尽管这些模型能够根据文本描述生成高质量的图像,促进了书籍插图、品牌标识设计、游戏场景创作等多种实际应用,但它们也被恶意用户…

html+js+css做的扫雷

做了个扫雷&#x1f4a3; 88大小 源代码在文章最后 界面 先点击蓝色开局按钮 然后就可以再扫雷的棋盘上玩 0代表该位置没有雷 其他数字代表周围雷的数量 源代码 <!DOCTYPE html> <html lang"en"> <head> <meta charset"UTF-8&qu…

vue事件参数

事件参数 事件参数可以获取event对象和通过事件传递数据 获取event对象 <template> <buttonclick"addCount">点击</button><p>count is: {{ count }}</p><p>{{ coutent_e }}</p> </template> <script>expor…

go 为什么是抢占式调度

GMP 模型 gmp模型是 golang 中用于调度管理 goroutine 的调度器。 调度器的发展史 在 Go 语言中&#xff0c;Goroutine 早期是没有设计成抢占式的&#xff0c;早期 Goroutine 只有读写、主动让出、锁等操作时才会触发调度切换。 这样有一个严重的问题&#xff0c;就是垃圾回…

AI视频生成技术爆发 引领虚拟数字人产业新潮流

2024年刚开局&#xff0c;先有OpenAI的AI视频生成模型Sora惊艳全网&#xff0c;随后阿里巴巴发布EMO&#xff0c;一张照片音频&#xff0c;就能生成具有生动表情和各种头部姿势、口型完全匹配高保真的人声头像动态视频。 技术的革新不仅为内容创作者打开了新世界的大门&#xf…

Spring Boot基础篇

快速上手 SpringBoot是由Pivotal团队提高的全新框架&#xff0c;其设计目的是用来简化Spring应用的初始化搭建以及开发过程 入门案例 在Idea创建 创建时要选择Spring Initializr。 Server URL为要连接的网站&#xff0c;默认为官网start.spring.io&#xff08;访问速度慢&…

为什么建议 MySQL 数据库字段一定要设置 NOT NULL

1. 前言 建议 MySQL 数据库字段一定要设置 NOT NULL 这句建议你可能听好多人讲过&#xff0c;但是有没有仔细想过为什么别人这么说 &#xff1f; 在实际开发中&#xff0c;对使不使用 not null 很多人并没有一个明确的标准&#xff0c;要知道某个字段需不需要添加 not null&a…

【MYSQL】事务隔离级别以及InnerDB底层实现

事务隔离级别 读未提交&#xff08;Read Uncommitted&#xff09; 允许事务读取其他事务未提交的数据&#xff0c;可能会导致脏读。 读已提交&#xff08;Read Committed&#xff09; 一个事务只能看见已经提交的事务所做的更改&#xff0c;可以避免脏读&#xff0c;但可能…

基于CLIP特征的多模态大模型中的视觉短板问题

【论文极速读】 基于CLIP特征的多模态大模型中的视觉短板问题 FesianXu 20240706 at Tencent WeChat search team 前言 今天读到篇CVPR 24’的论文 [1]&#xff0c;讨论了常见的多模态大模型&#xff08;大多都基于CLIP语义特征&#xff0c;以下简称为MLLM&#xff09;中的视觉…

阿里云服务器配置、搭建(针对Spring boot和MySQL项目)

这是一篇极其详细且痛苦的文章&#xff0c;还是在两位后端的大佬手把手教导下、以及我找遍全网所有资料、问了N遍AI、甚至直接申请阿里云工单一对一询问客服一整天、连续清空再上传反复30多次整个项目jar包......总结出来的终极要人命踩坑的问题总结 一、首先购买服务器 其实不…

Apache Seata分布式事务及其三种模式详解

本文来自 Apache Seata官方文档&#xff0c;欢迎访问官网&#xff0c;查看更多深度文章。 本文来自 Apache Seata官方文档&#xff0c;欢迎访问官网&#xff0c;查看更多深度文章。 Apache Seata分布式事务及其三种模式详解 分布式事务 Seata 及其三种模式详解 | Meetup#3 回顾…

基于TCP的在线词典系统(分阶段实现)

1.功能说明 一共四个功能&#xff1a; 注册 登录 查询单词 查询历史记录 单词和解释保存在文件中&#xff0c;单词和解释只占一行, 一行最多300个字节&#xff0c;单词和解释之间至少有一个空格。 2.功能演示 3、分阶段完成各个功能 3.1 完成服务器和客户端的连接 servic…

【深度学习】图形模型基础(5):线性回归模型第四部分:预测与贝叶斯推断

1.引言 贝叶斯推断超越了传统估计方法&#xff0c;它包含三个关键步骤&#xff1a;结合数据和模型形成后验分布&#xff0c;通过模拟传播不确定性&#xff0c;以及利用先验分布整合额外信息。本文将通过实际案例阐释这些步骤&#xff0c;展示它们在预测和推断中的挑战和应用。…

编程上下文Context及其实现原理

编程上下文Context及其实现原理 author:shengfq date:2024-07-06 title:编程上下文Context及其实现原理 category:编程思想1.编程中的上下文Context是指什么? 在编程和软件工程领域&#xff0c;“上下文”&#xff08;Context&#xff09;是一个多义词&#xff0c;其含义可以…

产品经理-​合作的6类干系人(8)

在一个项目中&#xff0c;产品经理并不是一个人在战斗&#xff0c;而是在很多同事的配合下共同完成项目。那产品经理到底要和哪些角色配合&#xff0c;一起完成项目呢 中间的产品经理是一个项目的驱动者。而产品经理的前方是“Boss/Leader”&#xff0c;也就是创业团队中公司的…

地级市数字经济指数、互联网用户数、数字金融普惠指数

2000-2022年地级市数字经济指数&#xff08;含控制变量&#xff09; 目录 数字经济如何改善环境污染 一、引言 二、文献综述 三、实证模型 四、数据来源 五、程序代码 六、运行结果 数字经济如何改善环境污染 摘要&#xff1a; 本论文旨在探讨数字经济对环境污染的改善作…

三级_网络技术_04_中小型网络系统总体规划与设计

1.下列关于路由器技术特征的描述中&#xff0c;正确的是()。 吞吐量是指路由器的路由表容量 背板能力决定了路由器的吞吐量 语音、视频业务对延时抖动要求较低 突发处理能力是以最小帧间隔值来衡量的 2.下列关于路由器技术特征的描述中&#xff0c;正确的是()。 路由器的…

Matlab中collectPlaneWave函数的应用

查看文档如下&#xff1a; 可以看出最多5个参数&#xff0c;分别是阵列对象&#xff0c;信号幅度&#xff0c;入射角度&#xff0c;信号频率&#xff0c;光速。 在下面的代码中&#xff0c;我们先创建一个3阵元的阵列&#xff0c;位置为&#xff1a;&#xff08;-1,0,0&#x…

JavaWeb----JSPJSTL

目录 JSP显隐注释在JSP中写java程序JSP的指令标签JSP中的四大域对象简易版用户登录EL表达式 JSTL条件动作标签if标签 choose\when\otherwise标签迭代标签格式化动作标签 用户登录实例查看是否安装了mysql用户登录界面后台实现 JSP JSP全名是Java Server Pages&#xff0c;它是建…

【后端面试题】【中间件】【NoSQL】MongoDB查询优化3(拆分、嵌入文档,操作系统)

拆分大文档 很常见的一种优化手段&#xff0c;在一些特定的业务场景中&#xff0c;会有一些很大的文档&#xff0c;这些文档有很多字段&#xff0c;而且有一些特定的字段还特别的大。可以考虑拆分这些文档 大文档对MongoDB的性能影响还是很大的&#xff0c;就我个人经验而言&…