Qt事件

一、事件类型

1、事件分类

Qt中事件根据源头分为三种类型:

  • 自发事件:由操作系统产生(例如用户按下鼠标,操作系统会产生一个鼠标事件)然后提交到Qt的event loop

  • Post事件:由应用程序通过postEvent()提交给Qt的event loop的事件

  • Send事件: 由应用程序通过sendEvent()触发的事件,与Post不一样的是他不会提交到Qt的event loop中,是同步执行的。

2、event loop

当在main()函数中调用QApplication::exec()时会自动开启一个event loop循环,每一次循环都会按照如下的顺序处理事件

while (!exit_was_called) {
    while (!posted_event_queue_is_empty) {
        process_next_posted_event();
    }
    while (!spontaneous_event_queue_is_empty) {
        process_next_spontaneous_event();
    }
    while (!posted_event_queue_is_empty) {
        process_next_posted_event();
    }
}

3、postEvent()和sendEvent()

大部分情况下的事件(例如鼠标事件、绘制事件,尺寸调整事件)都是由Qt自动帮我们提交到event loop,然后传递给对应的接受者,我们只需要实现对应的事件hander即可。例如QWidget中对应的各种各样的hander

class MyButton : public QPushButton{
public:
    MyButton(QWidget *parent = nullptr) : QPushButton(parent)
    {
        setStyleSheet("background-color: #ffffff;");
    }

    void mousePressEvent(QMouseEvent *e) override {
        qDebug() << "MyButton::mousePressEvent()";
        QPushButton::mousePressEvent(e);
        e->ignore();
    }

    void keyPressEvent(QKeyEvent *) override
    {
        qDebug() << "MyButton::keyPressEvent()";
    }

    bool event(QEvent *event) override
    {
        if (event->type() == QEvent::MouseButtonPress) {
            qDebug() << "MyButton::event()";
        }
        return QPushButton::event(event);
    }
};

其中event(QEvent *event)的默认实现流程大概如下:

1、根据event->type()的事件类型调用对应的xxxEvent()函数,大部分的系统事件都有对应的hander

2、没有对应xxxEvent()函数的事件则采用默认实现

可以通过QApplication::postEvent()或者QApplication::sendEvent()手动发送一个事件,前者将事件提交到event loop,后者直接触发

// mouseEvent事件必须分配在堆上,内部会持有它,当事件在event loop被使用后,会被自动delete。
auto *mouseEvent = new QMouseEvent(QEvent::MouseButtonPress, QPoint(50, 50), Qt::LeftButton, Qt::LeftButton, Qt::NoModifier);
QApplication::postEvent(myButton, mouseEvent);
// mouseEvent事件会被myButton立即处理,且内部不持有它,所以需要自己delete;或者将mouseEvent分配在栈上
auto *mouseEvent = new QMouseEvent(QEvent::MouseButtonPress, QPoint(50, 50), Qt::LeftButton, Qt::LeftButton, Qt::NoModifier);
QApplication::sendEvent(myButton, mouseEvent);
delete mouseEvent;

二、事件派发

事件响应链

窗口-->父视图-->子视图1-->子视图2....-->子视图n;越在顶端的子视图则优先处理事件

image.png

用户在按钮上点击一下,操作系统会生成一个QMouseEvent事件,添加到Qt的 Event loop,Qt首先找到最终能够响应该鼠标事件的子视图(鼠标点击的位置刚好在子视图上,且该子视图为可见的,处于隐藏状态或者被其它子视图覆盖都算不可见),最后在调用该子视图的event()函数,如果event()返回false,那么事件将传递给事件响应链的上一个视图,否则事件传递终止

三、事件过滤器

正常情况下,发送给一个QObject对象的事件首先会调用event()函数来进行处理,如果想抢在event()函数之前处该事件,可以给该对象添加一个事件过滤器对象,使用方法如下:

class EventFilter:public QObject
{
public:
    bool eventFilter(QObject *watched, QEvent *event) override
    {
        if (event->type() == QEvent::MouseButtonPress) {
            qDebug() << "EventFilter::eventFilter()";
            // 截获鼠标按下事件
            return true;
        }
        else if event->type() == QEvent::KeyPress{
            // 不处理键盘事件
            return false;
        }
        return QObject::eventFilter(watched, event);
    }
}

// 给myWidget对象添加事件过滤器
auto *myWidget = new MyWidget;
myWidget->installEventFilter(new EventFilter);

eventFilter()返回true代表事件直接被截获了,false则代表事件会丢回给原先对象的event()函数中去处理

四、消费事件

在事件传递的过程中,除了通过事件过滤器(即eventFilter())决定事件是否消费外,还可以通过调用event的accept()函数来消耗这个事件,当事件被消耗后,它将不再继续传递。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,088评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,715评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,361评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,099评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 60,987评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,063评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,486评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,175评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,440评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,518评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,305评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,190评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,550评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,880评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,152评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,451评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,637评论 2 335

推荐阅读更多精彩内容

  • 一、QApplication、QGuiApplication、QCoreApplication简介 1、继承关系见...
    379755b27396阅读 1,214评论 0 1
  • 参考:Events and signals in PyQt5 所有的应用都是事件驱动的。事件大部分都是由用户的行为...
    水之心阅读 1,804评论 1 1
  • QObject三大核心功能:信号与槽,内存管理,事件处理 总览 1、谁来产生事件: 最容易想到的是我们的输入设备,...
    上官宏竹阅读 2,595评论 2 4
  • 事件 事件(event)是由系统或者 Qt 本身在不同的时刻发出的。 一些事件在对用户操作做出响应时发出,如键盘事...
    人不知QAQ阅读 258评论 0 0
  • Qt 事件和信号的关系 Qt的事件是windows的底层消息封装而成的。这个消息和MFC里的消息是同一概念,都是指...
    行走的代码阅读 1,451评论 0 0