zoukankan      html  css  js  c++  java
  • Qt事件系统之五:事件过滤器和事件的发送

    Qt提供了事件过滤器来实现在一个部件中监控其他多个部件的事件。事件过滤器与其他部件不同,它不是一个类,只是由两个函数组成的一种操作,用来完成一个部件对其他部件的事件的监视。这两个函数分别是 installEventFilter() 和 eventFilter(),都是QObject类中的函数。下面通过具体的例子来讲解。

    新建Qt Gui应用,将项目名称更改为myEventFilter,基类选择QWidget,类名保持Widget不变。完成后在设计模式中向界面上拖入一个QTextEdit和一个QSpinBox。在widget.h文件中添加函数声明:

    //事件过滤器
    bool eventFilter(QObject *obj, QEvent *event);
    

    在构造函数中添加代码:

    //为textEdit和spinBox在本窗口上安装事件过滤器
    ui->textEdit->installEventFilter(this);
    ui->spinBox->installEventFilter(this);
    

    要对一个部件使用事件过滤器,那么就要先使用installEventFilter() 函数为该部件安装事件过滤器,这个函数的参数表明了监视对象。比如这里就是为textEdit部件和spinBox部件安装了事件过滤器,其参数this表明要在本部件即Widget中监视textEdit和spinBox的事件。这样,就需要重新实现Widget类的eventFilter()函数,在其中截获并处理两个子部件的事件。

    //事件过滤器
    bool Widget::eventFilter(QObject *obj, QEvent *event)
    {
        //判断部件
        if (obj == ui->textEdit)
        {
            //判断事件
            if (event->type() == QEvent::Wheel)
            {
                // 将event强制转换为发生的事件的类型
                QWheelEvent *wheelEvent = static_cast<QWheelEvent*>(event);
                //缩放textEdit
                if (wheelEvent->delta() > 0)
                    ui->textEdit->zoomIn();
                else
                    ui->textEdit->zoomOut();
    
                return true;
            }
            else //如果是其他事件,可以进行进一步的处理
            {
                return false;
            }
        }
        else if (obj == ui->spinBox)
        {
            if (event->type() == QEvent::KeyPress)
            {
                QKeyEvent *keyEvent = static_cast<QKeyEvent*>(event);
                if (keyEvent->key() == Qt::Key_Space)
                {
                    ui->spinBox->setValue(0);
                    return true;
                }
                else
                {
                    return false;
                }
            }
            else
            {
                return false;
            }
        }
    
        else return QWidget::eventFilter(obj, event);
    }
    

    在这个事件过滤器中先判断部件的类型,然后再判断事件的类型,如果是我们需要的事件就将其进行强制类型转换,然后进行相应的处理。这里需要说明,如果要对一个 特定的事件进行处理,而且不希望它在后面的传递过程中再被处理,那么就返回true, 否则返回false。这个函数中实现了在textEdit部件中使用滚轮进行内容的放大或缩小,在spinBox部件中使用空格来使数值设置为0。

    可以看到,使用事件过滤器可以很容易地处理多个部件的多个事件,如果不使用它,我们就得分别子类化各个部件,然后重新实现它们对应的各个事件处理函数,那样就会很麻烦了。


    Qt也提供发送一个事件的功能,由 QCoreApplication 类的 sendEvent() 函数或者 postEvent() 函数来实现。这两个函数的主要区别是:sendEvent() 会立即处理给定的事件,而 postEvem() 则会将事件放到等待调度队列中,当下一次Qt的主事件循环运行时才会处理它。

    这两个函数还有其他一些区别,比如 sendEvent() 中的QEvent对象参数在事件发送完成后无法自动删除,所以需要在栈上创建QEvent对象;而 postEvent() 中的QEvent对象参数必须在堆上进行创建(例如使用new),当事件被发送后事件队列会自动删除它。

    下面在widget.cpp文件中的构造函数里添加代码来向spinBox部件发送一个向上方向键被按下的事件:

    //向spinBox发送一个向上按钮触发键盘事件,所以可以看到最开始spinBox的值不是0,而是1
    QKeyEvent myEvent(QEvent::KeyPress, Qt::Key_Up, Qt::NoModifier);
    qApp->sendEvent(ui->spinBox, &myEvent); //qApp是Application对象的全局指针
    

    这里使用了 sendEvent() 函数,其中 QKeyEvent 类对象是在栈上创建的。其中的qApp是Q Application对象的全局指针,每一个应用程序中只能使用一个QApplica­tion 对象,这里等价于使用 QApplication:: sendEvent()。 现在运行程序可以发现 spinBox部件中初始值变为了1,这说明已经在这个部件上按下了向上方向键。


    滚轮放大前后的对比图如下:

  • 相关阅读:
    [学习笔记]康托展开
    [模板]平衡树
    [题解]涂色
    [学习笔记]Lucas定理
    欧拉定理及其证明
    一些杂题(排列组合
    swift 动画合集
    UIDynamicAnimator UIKit动力学
    swift 当协议遇见了泛型
    Swift 协议
  • 原文地址:https://www.cnblogs.com/linuxAndMcu/p/11024173.html
Copyright © 2011-2022 走看看