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个级别的事件处理和过滤: 
1,重新实现事件函数。 比如: mousePressEvent(), keyPress-Event(),   paintEvent() 。 
   这是最常规的事件处理方法。 
2,重新实现QObject::event(). 
   这一般用在Qt没有提供该事件的处理函数时。也就是,我们增加新的事件时。 
3,安装事件过滤器 
4,在 QApplication 上安装事件过滤器。 
   这之所以被单独列出来是因为: QApplication 上的事件过滤器将捕获应用程序的所有事件,而且第一个获得该事件。也就是说事件在发送给其它任何一个event filter之前发送给QApplication的event filter。 
5,重新实现QApplication 的 notify()方法. 
Qt使用 notify()来分发事件。要想在任何事件处理器捕获事件之前捕获事件,唯一的方法就是重新实现QApplication 的 notify()方法。

/****************************************************************************/

Qt创建了QEvent事件对象之后,会调用QObject的event()函数做事件的分发。有时候,你可能需要在调用event()函数之前做一些另外的操作,比如,对话框上某些组件可能并不需要响应回车按下的事件,此时,你就需要重新定义组件的event()函数。如果组件很多,就需要重写很多次event()函数,这显然没有效率。为此,你可以使用一个事件过滤器,来判断是否需要调用event()函数。

  QOjbect有一个eventFilter()函数,用于建立事件过滤器。这个函数的签名如下:

virtual bool QObject::eventFilter ( QObject * watched, QEvent * event )

如果watched对象安装了事件过滤器,这个函数会被调用并进行事件过滤,然后才轮到组件进行事件处理。在重写这个函数时,如果你需要过滤掉某个事件,例如停止对这个事件的响应,需要返回true

bool MainWindow::eventFilter(QObject *obj, QEvent *event) 
   { 
          if (obj == textEdit) { 
              if (event->type() == QEvent::KeyPress) { 
                   QKeyEvent *keyEvent = static_cast(event); 
                   qDebug() << "Ate key press" << keyEvent->key(); 
                    return true; 
                } else { 
                   return false; 
                 } 
             } else { 
             // pass the event on to the parent class 
              return QMainWindow::eventFilter(obj, event); 
          } 
   }

上面的例子中为MainWindow建立了一个事件过滤器。为了过滤某个组件上的事件,首先需要判断这个对象是哪个组件,然后判断这个事件的类型。例如,我不想让textEdit组件处理键盘事件,于是就首先找到这个组件,如果这个事件是键盘事件,则直接返回true,也就是过滤掉了这个事件,其他事件还是要继续处理,所以返回false。对于其他组件,我们并不保证是不是还有过滤器,于是最保险的办法是调用父类的函数。

  在创建了过滤器之后,下面要做的是安装这个过滤器。安装过滤器需要调用installEventFilter()函数。这个函数的签名如下:

 

 void QObject::installEventFilter ( QObject * filterObj )

  这个函数是QObject的一个函数,因此可以安装到任何QObject的子类,并不仅仅是UI组件。这个函数接收一个QObject对象,调用了这个函数安装事件过滤器的组件会调用filterObj定义的eventFilter()函数。例如,textField.installEventFilter(obj),则如果有事件发送到textField组件是,会先调用obj->eventFilter()函数,然后才会调用textField.event()。

  当然,你也可以把事件过滤器安装到QApplication上面,这样就可以过滤所有的事件,已获得更大的控制权。不过,这样做的后果就是会降低事件分发的效率。

  如果一个组件安装了多个过滤器,则最后一个安装的会最先调用,类似于堆栈的行为。

  注意,如果你在事件过滤器中delete了某个接收组件,务必将返回值设为true。否则,Qt还是会将事件分发给这个接收组件,从而导致程序崩溃。

  事件过滤器和被安装的组件必须在同一线程,否则,过滤器不起作用。另外,如果在install之后,这两个组件到了不同的线程,那么,只有等到二者重新回到同一线程的时候过滤器才会有效。

  事件的调用最终都会调用QCoreApplication的notify()函数,因此,最大的控制权实际上是重写QCoreApplication的notify()函数。由此可以看出,Qt的事件处理实际上是分层五个层次:重定义事件处理函数,重定义event()函数,为单个组件安装事件过滤器,为QApplication安装事件过滤器,重定义QCoreApplication的notify()函数。这几个层次的控制权是逐层增大的。

关于事件处理  另外参看

http://blog.csdn.net/xie376450483/archive/2010/08/18/5821970.aspx

参考:http://blog.csdn.net/seanyxie/article/details/5930564

时间: 2024-12-25 18:26:06

QT事件过滤器(全局过滤,省事)的相关文章

Qt入门 - 全局数据类型

<QtGlobal> 1 typedef signed char qint8; /* 8 bit signed */ 2 typedef unsigned char quint8; /* 8 bit unsigned */ 3 typedef short qint16; /* 16 bit signed */ 4 typedef unsigned short quint16; /* 16 bit unsigned */ 5 typedef int qint32; /* 32 bit signe

在ASP.NET Core中通过EF Core实现一个简单的全局过滤查询

前言 不知道大家是否和我有同样的问题: 一般在数据库的设计阶段,会制定一些默认的规则,其中有一条硬性规定就是一定不要对任何表中的数据执行delete硬删除操作,因为每条数据对我们来说都是有用的,并且是值得分析的. 所以我们一般会在每张表中加一个"是否删除IsDeleted"或者"是否有效IsValid"的字段,来标识这条数据的状态是否可用! 那么疑问来了,在写SQL或者Linq的时候我们到底是要加上这个条件还是忽略这个条件呢?答案当然是根据实际业务需求和情况来决定.

asp.net core 2.0 全局过滤(HasQueryFilter)

本文章主要是为了总结自己遇到的问题,同时也是别人的已经最哦好的成果. HasQueryFilter主要用于数据软删除,还习惯用delete的小伙伴,你们out了... 首先设置一个基类,BaseEntity,包含IsDeleted属性: 所有用到软删除的类继承于此类,然后在MyDbContext中的OnModelCreating方法中实现全局过滤: 代码如下: foreach (var entityType in modelBuilder.Model.GetEntityTypes() .Wher

asp.net core MVC 全局过滤器之ExceptionFilter过滤器(一)

本系类将会讲解asp.net core MVC中的内置全局过滤器的使用,将分为以下章节 asp.net core MVC 过滤器之ExceptionFilter过滤器(一) asp.net core MVC 过滤器之ActionFilter过滤器(二) asp.net core MVC 过滤器之ResultFilter过滤器(三) asp.net core MVC 过滤器之ResourceFilter过滤器(四) asp.net core MVC 过滤器之AuthorizationFilter过

vue货币格式化组件、局部过滤功能以及全局过滤功能

在页面中,例如价格数据,不管是后台传递过来的还是前台计算之后显示在页面上的,一般都只是一个数字没有格式,完整的格式应该是 要实现这个其实很简单,vue的过滤功能就很好的能解决这个问题,什么叫做过滤,就是将元数据进行相应的处理在显示出来. 首先建立一个 js 文件 currency.js const digitsRE = /(\d{3})(?=\d)/g /** * value 金额 * currency 货币符号 * decimals 保留位数 */ export function curren

QT 事件过滤器 eventFilter

在监测的代码里执行需要的行为. 这可以用event Filter来达到. 设置一个event filter有两个步骤: 1. 在目标对象上调用installEventFilter(),将监测对象注册到目标对象上.2. 在监测对象的eventFilter()方法里处理目标对象的事件. 在ctor里注册监测对象是一个好地方:CustomerInfoDialog::CustomerInfoDialog(QWidget *parent) :QDialog(parent){    ...    first

qt捕获全局windows消息(使用QAbstractNativeEventFilter,然后注册这个类)

qt  如何捕获全屏的鼠标事件,这个帖子上面主要讲述了下嵌入式qt怎么抓取系统级消息,不过从这篇文章中我也看到了希望,有个回复说winEventFilter支持这种方式,然后我就顺着这个线索找到了nativeEventFilter方法,最终试验成功. 首先是让你自己的类继承自QAbstractNativeEventFilter,然后通过QCoreApplication来注册你的窗口类,代码如下:app.installNativeEventFilter(m_MainWindow); 最后在nati

Qt事件过滤器Event Filter

事件过滤器针对一类或者多种不同类型的对象,定义了重写操作. 简单的说: 1.先对UI对象注册Event Filter 例如:    ui->drawBtn->installEventFilter(this); 2.定义过滤器操作事件 bool BaseWidget::eventFilter(QObject *obj, QEvent *e){ switch(e->type()) { case QEvent::MouseButtonPress: QToolTip::showText(stat

ef 通过反射和表达式树配置全局过滤筛选器

1.软删除接口 1 public interface ISoftDelete 2 { 3 bool IsDeleted { get; set; } 4 } 2.实体类实现 1 public class TestDeleted:ISoftDelete 2 { 3 public int Id { get; set; } 4 5 public string Name { get; set; } 6 public bool IsDeleted { get; set; } 7 8 9 } 3.modelB