QT-事件分发器、事件过滤器

原文章:https://blog.csdn.net/qq_31073871/article/details/80661064

事件分发器:

教程中告诉我们要子类化QEvent,派生出my_Event,然后在my_Event中定义事件类型。

比如,我需要对PushButton按钮进行事件处理。首先呢,我们需要定义一个my_label类是继承于QLabeln,然后在my_label类中重写事件分发器,通过事件分发器来拦截事件。

//鼠标按下
void my_label::mousePressEvent(QMouseEvent *ev)
{
        QString str = QString(QString::fromLocal8Bit("鼠标按下 X = %1  Y= %2").arg(ev->x()).arg(ev->y()));
        ev->ignore();//使事件继续向父控件传递
        qDebug()<<str;
}

在widget上添加一个Qlabel,然后提升为my_label,然后也重写一下widget的鼠标按下事件

直接在widget.h里声明

//声明鼠标按下事件
void mousePressEvent(QMouseEvent *event);

在widget里重写鼠标按下事件

void Widget::mousePressEvent(QMouseEvent *event)
{
    qDebug()<<"123";
}

当鼠标按下的时候,子控件处理完之后继续向父控件进行传递。

如果不让事件继续向父控件进行传递,则到子控件处理完之后就结束了。

 相当于子控件在窗口上占的位置,处理完之后,就忽略label的位置,点击就变成了widget上。

 事件传递的方向是子控件先接收到事件,父控件后接收到控件。

事件过滤器:

事件分发器是子控件先控制,然后在向父控件进行传递。

假如说我们想先让父控件进行控制,然后再传递给子控件就要用到事件过滤器。

事件过滤器可以提前拦截其他任意对象的事件,兰街道的事件都会在本对象的eventFilter函数中接收到,我们可以决定拦截处理完之后是否继续放行。事件过滤器在事件分发器触发之前进行拦截。

举例:一个按钮被点击,本应是按钮先收到点击事件,通过事件过滤器,可以让窗体先收到这个事件,窗体再决定是否把这个事件继续传播给按钮。

实现事件过滤器需要两个步骤:

1.给控件安装事件过滤器

在widget.cpp的构造函数中写上

    //事件过滤器
    //label_3设置widget作为自己的事件过滤器
    ui->label_3->installEventFilter(this);

2.重写 eventFilter(QObject *watched, QEvent *event)函数

代码解析:

1.如果监视的对象时label_3

2.如果事件类型是鼠标点击

3.因为事件过滤器中event对象是继承于QEvent,鼠标是继承于QMouseEvent。所以要把QEvent强转成QMouseEvent。

如果鼠标按下的是左键,return true拦截事件并且不往下分发,反之继续传播。

//label的所有事件都要先流到这里,这里放行后label才能收到事件
bool Widget::eventFilter(QObject *watched, QEvent *event)
{
    if(watched == ui->label_3)//确认被监视的对象
    {
        if(event->type() == QEvent::MouseButtonPress)//确认事件的类型
        {
            QMouseEvent *ev = static_cast<QMouseEvent *>(event);//前面已经确认过事件类型为鼠标类型,所以这里可以放心的进行静态转换
            qDebug()<<"Filter mainwindow MouseButtonPress";
            if(ev->button()==Qt::LeftButton)
                return true;//true 拦截
            else
                return false;//false 继续传播
        }
    }
    return false;//继续传播
}

可以看到:

1、按钮并没有收到左键被按下事件,因为这一事件被监视对象给拦截后,没有放行;

2、按钮收到了右键按下事件,这一事件依次被以下函数执行:监视对象的eventFilter函数-->按钮的mousePressEvent()函数-->窗口的mousePressEvent()函数
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Qt事件过滤器 Qt事件模型一个真正强大的特色是一个QObject 的实例能够管理另一个QObject 实例的事件。 让我们试着设想已经有了一个CustomerInfoDialog的小部件。CustomerInfoDialog 包含一系列QLineEdit. 现在,我们想用空格键来代替Tab,使焦点在这些QLineEdit间切换。 一个解决的方法是类化QLineEdit,重新实现keyPressEvent(),并在keyPressEvent()里调用focusNextChild()。像下面这样: void MyLineEdit::keyPressEvent(QKeyEvent *event) { if (event->key() == Qt::Key_Space) { focusNextChild(); } else { QLineEdit::keyPressEvent(event); } } 但这有一个缺点。如果CustomerInfoDialog里有很多不同的控件(比如QComboBox,QEdit,QSpinBox),我们就必须类化这么多控件。这是一个烦琐的任务。 一个更好的解决办法是: 让CustomerInfoDialog去管理他的部件的按键事件,实现要求的行为。我们可以使用事件过滤器。 一个事件过滤器的安装需要下面2个步骤: 1, 调用installEventFilter()注册需要管理的对象。 2,在eventFilter() 里处理需要管理的对象的事件。 一般,推荐在CustomerInfoDialog的构造函数中注册被管理的对象。像下面这样: CustomerInfoDialog::CustomerInfoDialog(QWidget *parent) : QDialog(parent){ ... firstNameEdit->installEventFilter(this); lastNameEdit->installEventFilter(this); cityEdit->installEventFilter(this); phoneNumberEdit->installEventFilter(this); } 一旦,事件管理被注册,发送到firstNameEdit,lastNameEdit,cityEdit,phoneNumberEdit的事件将首先发送到eventFilter()。 下面是一个 eventFilter()函数的实现: bool CustomerInfoDialog::eventFilter(QObject *target, QEvent *event) { if (target == firstNameEdit || target == lastNameEdit || target == cityEdit || target == phoneNumberEdit) { if (event->type() == QEvent::KeyPress) { QKeyEvent *keyEvent = static_cast<QKeyEvent *>(event); if (keyEvent->key() == Qt::Key_Space) { focusNextChild(); return true; } } } return QDialog::eventFilter(target, event); } 在上面的函数中,我们首先检查目标部件是否是 firstNameEdit,lastNameEdit,cityEdit,phoneNumberEdit。接着,我们判断事件是否是按键事件。如果事件是按键事件,我们把事件转换为QKeyEvent。接着,我们判断是否按下了空格键,如果是,我们调用focusNextChild(),把焦点传递给下一个控件。然后,返回,true通知Qt,我们已经处理了该事件。 如果返回false的话,Qt继续将该事件发送给目标控件,结果是一个空格被插入到QLineEdit中。 如果目标控件不是 QLineEdit,或者按键不是空格键,我们将把事件传递给基类的eventFilter()函数。 Qt提供5个级别

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值