Qt之事件系统

简述

在Qt中,事件就是对象,派生自QEvent抽象类,用来表示在应用程序中发生的事件,或是应用程序需要处理的外部活动产生的事件。

Events可以被任何QObject派生的子类实例对象接收和处理,但它们是关联到特定控件的。下面,我们主要介绍event在典型应用程序中是如何发送及处理的。

  • 简述
  • 事件如何发送
  • 事件类型
  • 事件处理程序
  • 事件过滤器
  • 发送事件
  • 更多参考

事件如何发送

通常情况下,当一个事件发生时,Qt通过构造一个合适的QEvent子类对象来表示事件的发生,然后通过调用event()函数,将该事件对象发送给特定的QObject(或其子类)对象。

此函数不会对事件本身进行处理, 而是首先检查所接受到的事件类型, 然后根据事件类型来调用相应的事件处理程序,事件处理程序在处理完事件之后会返回一个bool值表示该事件是被接受,还是被忽略了。

某些事件,例如:QMouseEvent和QKeyEvent,来自于窗口系统;一些诸如:QTimerEvent,来自于其它的事件源;某些,则来自于应用程序本身。

事件类型

Qt为多数事件类型建立了相应的类,常见的有:QResizeEvent、QPaintEvent、QMouseEvent、QKeyEvent、QCloseEvent。

每一个特定的事件类都继承自QEvent,并添加特定的事件函数。例如:QResizeEvent添加了size()和oldSize()让控件可以发现它们的尺寸发生了的怎么改变.

某些类实际支持不止一种事件类型。QMouseEvent就支持鼠标按键按下事件、双击事件、移动事件、以及其它相关操作所引发的事件。

每一个事件都有一个关联的类型,由QEvent::Type定义,运行时可以很方便的检测每个事件对象的事件类型,以快速的判断该事件对象构造自哪个事件类。

由于程序需要和又多样又复杂的事件进行交互,所以Qt的事件发送机制设计的非常灵活。

QCoreApplication::notify()的文档简洁的说明了整个机制:

bool QCoreApplication::notify(QObject * receiver, QEvent * event) [virtual]

发送事件给接收者:receiver->event(event)。返回从接收者的事件处理程序返回的值。注意这个函数适用于该应用程序中的任何线程中的任何对象。对于特定类型的事件(例如:鼠标和键盘事件)),该事件将被传送到接收者的parent并这样逐级上传。

直到传到顶级对象,如果这些接收者都没有对该事件进行处理的话(比如:返回false)。

共五种处理事件的方式,重写QCoreApplication::notify()只是其中的一种。以下列出了这五种方法:

  1. 重写paintEvent()、mousePressEvent()等。这是最常用、最简单但也是最有限的方式。
  2. 重写QCoreApplication::notify()。这非常强大,可以完全控制事件处理,但一次只可用于一个子类。
  3. 为QCoreApplication::instance()安装一个事件过滤器。这个事件过滤器就能处理所有控件的所有事件,因此这与重写notify()一样强大;此外,可以有不止一个应用程序全局级的事件过滤器,应用程序全局级事件过滤器甚至可以收到已禁用控件的鼠标事件。

    注意:应用程序级事件过滤器仅能用于存活在主线程中的对象。

  4. 重写QObject::event()(像QWidget那样 )。如果你重写了QObject::event(),当Tab键按下时,你就可以在任何控件级事件过滤器捕获这个Tab键按下事件之前处理这个事件。
  5. 给相应的接收对象安装一个事件过滤器。例如:一个捕获所有事件的事件过滤器,包含Tab和Shift+Tab键按下事件,在它们没有改变焦点控件之前。

    另请参考:QObject::event()以及installEventFilter()。

事件处理程序

处理事件的标准方式是调用一个虚函数。例如:QPaintEvent是通过调用QWidget::paintEvent()来处理的。这个虚函数负责进行相应的处理,通常就是重画该控件。如果你在自己实现的虚函数中没有做所有必要的工作,你就有必要调用它的基类实现。

例如,下面的代码处理一个定制checkbox控件的鼠标左键点击事件,并将所有其它点击事件转发给它的基类 QCheckBox:

void MyCheckBox::mousePressEvent(QMouseEvent *event)
{
    if (event->button() == Qt::LeftButton) {
        // 处理鼠标左键
    } else {
        // 传递其它按键给基类
        QCheckBox::mousePressEvent(event);
    }
}

如果你需要替换基类的函数,你应该自己实现所有相关的处理。但是,如果你只想扩展基类的功能,那么你就只需实现需要实现的部分,然后调用基类处理函数来处理你不想处理的情况。

偶尔,你要处理没有相应处理函数的特定事件,或遇到事件处理函数不够用情况。最常见的例子是Tab键按下事件。通常,QWidget截获到Tab键按下事件后,会移动键盘焦点,但是少数控件需要自己来处理这个事件。这些对象可以重写QObject::event()函数,通用的事件处理程序,然后在通常处理过程之前或之后写自己的事件处理过程,或完全替代原处理过程。下面是这样一个很常见的控件:

一个既自己处理Tab事件又自己处理某些按键事件,然后将其它不需自己处理的事件转发给基类处理:

bool MyWidget::event(QEvent *event)
{
    if (event->type() == QEvent::KeyPress) {
    QKeyEvent *ke = static_cast<QKeyEvent *>(event);
    if (ke->key() == Qt::Key_Tab) {
        // 特别的Tab操作
        return true;
    }
    } else if (event->type() == MyCustomEventType) {
    MyCustomEvent *myEvent = static_cast<MyCustomEvent *>(event);
    // 自定义事件处理
    return true;
    }

    return QWidget::event(event);
}

注意:对没有处理的事件仍调用QWidget::event(),并返回该基类调用的返回值以指示事件是否被处理了;若返回true,则将会禁止将该事件再发往其它对象。

事件过滤器

有时候一个对象需要检查,还可能截取发往其它对象的事件。例如:对话框通常需要过滤发往某些控件的事件,比如:更改Enter键按下的事件处理。

通过调用过滤器对象的QObject::installEventFilter()函数,为目标对象设置一个事件过滤器,就可在过滤器对象的QObject::eventFilter()函数中处理发往目标对象的事件。一个事件过滤器在目标对象收到事件之前处理事件,这使得过滤器对象在需要的时候可以检查并丢弃事件。可以调用 QObject::removeEventFilter()来移除已经安装的事件过滤器。

当过滤器的eventFilter()实现被调用时,它可以选择处理该事件,还是转发该事件, 或禁止该事件继续被其它对象处理。若所有的事件过滤器都允许一个事件可被继续处理(每个过滤器处理后都返回 false), 该事件最终将被发送到目标对象。如果其中一个中止了这个流程(通过返回true),则后面的过滤器对象以及目标对象将不会收到该事件。

bool FilterObject::eventFilter(QObject *object, QEvent *event)
{
    if (object == target && event->type() == QEvent::KeyPress) {
        QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event);
        if (keyEvent->key() == Qt::Key_Tab) {
            // 特别的Tab操作
            return true;
        } else
            return false;
    }
    return false;
}

上面代码演示了另外一种截取发往特定对象Tab键事件的方法。在这个例子里,该过滤器处理Tab事件后返回 true来阻止它们被继续处理。所有其它的按键事件将被忽略掉,然后过滤器返回false来允许该事件被已安装的后续过滤器处理,最终发往目标控件.

另外,也可以过滤整个应用程序的所有事件, 只需将过滤器对象安装到QApplication或QCoreApplication 对象上。这样的全局事件过滤器会在任何对象级过滤器调用之前调用.

这是非常强大的,但也拖慢了整个应用程序范围内每个事件的每次处理过程;通常使用其它的技术来代替。

发送事件

许多应用程序需要创建并发送自己的事件。你完全可以模仿Qt自有的事件循环机制, 先构造合适的事件对象, 然调用QCoreApplication::sendEvent()和QCoreApplication::postEvent()把构造好的事件发送给指定的接收者.

sendEvent()立即同步处理要发送的事件。当它返回的时候, 表示相关的事件过滤器 和/或 目标对象已经处理完了该事件。 对于多数的事件类,有一个名为isAccepted()的函数可以用来判别该事件是已被接受处理了,还是被拒绝处理了.

postEvent()将事件提交到一个队列中等待调度。在下一次Qt的主事件循环运行时,就会调度所有提交到队列中的事件,以某种优化的方式。例如,如果有几个大小改变事件,它们就会被压缩成一个事件。这同样适用于绘图事件:QWidget::update()调用postEvent(),以避免多次重画来消除闪烁以及提高速度。

postEvent()也被用于对象的初始化过程,因为提交过的事件通常在相应对象初始化完毕后极短的时间内就会被调度。在实现一个控件的时候,在自定义控件的构造函数中尽早支持事件机制是非常重要的,在可能接受到任何事件之前,确保尽早初始化成员变量。

要创建一个自定义的事件,你需要定义一个事件号,这个事件号应该大于QEvent::User,并且可能需要继承QEvent,以传递关于你自定义的事件类型的特定信息。

更多参考

  • The Event System - Qt助手
时间: 2024-10-12 18:57:14

Qt之事件系统的相关文章

用ISO C++实现自己的信号槽(Qt另类学习)

qtc++objectsignalclassstring 目录(?)[-] Qt信号与槽 引入元对象系统 建立信号槽链接 信号的激活 槽的调用 全家福 零零散散写在后面 Q_OBJECT Connection 其他 有网友抱怨: 哪个大牛能帮帮我,讲解一下信号槽机制的底层实现? 不要那种源码的解析,只要清楚的讲讲是怎么发送信号,怎么去选择相应的槽,再做出反应.也就是类似于一个信号槽的相应流程...求解啊!!! 看了源码,真的是一头雾水...撞墙的心都有了~~~~ 本文使用 ISO C++ 一步一

Qt 状态机框架学习(没学会)

Qt状态机框架是基于状态图XML(SCXML) 实现的.从Qt4.6开始,它已经是QtCore模块的一部分.尽管它本身是蛮复杂的一套东西,但经过和Qt的事件系统(event system).信号槽(signals and slots)及属性系统(property system)深度整合,它使用门槛并不高. 一些概念 Qt的手册中The State Machine Framework一文对Qt状态机框架及使用进行了介绍,可是还是发现看看基本的概念(详见  SCXML   的  第三部分 )更有帮助

Qt版音乐播放器

    Qt版音乐播放器 转载请标明出处:牟尼的专栏 http://blog.csdn.net/u012027907 一.关于Qt 1.1 什么是Qt Qt是一个跨平台应用程序和UI开发框架.使用Qt只需一次性开发应用程序,无需重新编写源代码,便可跨不同桌面和嵌入式操作系统部署这些应用程序. Qt Creator是全新的跨平台Qt IDE,可单独使用,也可与Qt库和开发工具组成一套完整的SDK,其中包括:高级C++代码编辑器,项目和集成管理工具,集成的上下文相关的帮助系统,图形化调试器,代码管理

Qt 5框架介绍

该文章原创于Qter开源社区(www.qter.org),作者 yafeilinux,转载请注明出处! 一.在帮助中查看所有模块 打开 Qt Creator,进入其帮助模式,然后选择目录方式进行查看,打开 "Qt 5.2.0ReferenceDocumentation" 页面.在这里提供了Qt 5.2 的整体介绍,并将其所有内容进行了分类.我们选择右下角的 "All Qt Modules" 来查看所有的 Qt 模块.如下图所示. 在所有模块页面,将 Qt 的模块分为

从 Qt 的 delete 说开来

原地址:http://blog.csdn.net/dbzhang800/article/details/6300025 在C++中学习过程中,我们都知道: delete 和 new 必须 配对使用(一 一对应):delete少了,则内存泄露,多了麻烦更大. Qt作为C++的库,显然是不会违背C++的前述原则的.可是: 在Qt中,我们很多时候都疯狂地用new,却很少用delete,缺少的 delete 去哪儿了?! 注:本文暂不涉及智能指针(smart pointer)相关的东西,你可以考虑 Qt

Qt中的键盘事件,以及焦点的设置(比较详细)

Qt键盘事件属于Qt事件系统,所以事件系统中所有规则对按键事件都有效.下面关注点在按键特有的部分: focus 一个拥有焦点(focus)的QWidget才可以接受键盘事件.有输入焦点的窗口是活动窗口或活动窗口子窗口或子子窗口等. 焦点移动的方式有以下几种: 按下Tab或Shift+Tab 注意:文本编译器(一般需要插入Tab),或者WebView(需要Tab来移动超链接焦点) 等 Qt中,需要输入Tab的地方可以用 Ctrl+Tab 或 Ctrl+Shift+Tab 替代. 点击一个QWidg

Qt程序设计

1.与STL对应Qt库有自己的容器类:QVector<T>.QList<T>.QLinkedList<T>.QStack<T>.QQueue<T>.QMap<K,T>.QHash<K,T>等Qt库的容器的基本操作与STL库基本相同,但是具体实现不同.对于元素的访问支持3种方式:(1)通过下标访问(2)通过迭代器访问(3)通过成员函数访问(比如value(idx)带越界检测的访问,at(idx)不带越界检查的访问)与STL库

Qt delete & deletelater设计

在C++中学习过程中,我们都知道: delete 和 new 必须 配对使用(一 一对应):delete少了,则内存泄露,多了麻烦更大. Qt作为C++的库,显然是不会违背C++的前述原则的.可是: 在Qt中,我们很多时候都疯狂地用new,却很少用delete,缺少的 delete 去哪儿了?! 注:本文暂不涉及智能指针(smart pointer)相关的东西,你可以考虑 Qt 智能指针学习 一文 Qt半自动的内存管理 在Qt中,以下情况下你new出的对象你可以不用 亲自去delete (但你应

Qt 学习之路 :Qt 绘制系统简介

Qt 的绘图系统允许使用相同的 API 在屏幕和其它打印设备上进行绘制.整个绘图系统基于QPainter,QPainterDevice和QPaintEngine三个类. QPainter用来执行绘制的操作:QPaintDevice是一个二维空间的抽象,这个二维空间允许QPainter在其上面进行绘制,也就是QPainter工作的空间:QPaintEngine提供了画笔(QPainter)在不同的设备上进行绘制的统一的接口.QPaintEngine类应用于QPainter和QPaintDevice