您的位置:

深入了解eventfilter

一、eventfilter用法

Qt中提供了event()函数来接收所有的事件,但在某些情况下,我们并不希望所有的事件都被接收,如何才能筛选需要的事件呢?这时,eventfilter便派上用场了。

在Qt中,我们可以使用installEventFilter函数,在一个对象中安装一个事件过滤器,使得该对象只能接收到我们希望它接收到的事件类型。下面是一个简单的示例:

class FilterObject : public QObject
{
protected:
bool eventFilter(QObject *obj, QEvent *event) override {
if (event->type() == QEvent::KeyPress) {
QKeyEvent *keyEvent = static_cast(event);
qDebug() << "Key press event:" << keyEvent->key();
return true;
} else {
// standard event processing
return QObject::eventFilter(obj, event);
}
}
};

int main(int argc, char *argv[])
{
QApplication a(argc, argv);
QWidget widget;
widget.resize(200, 200);
widget.show();

FilterObject *filter = new FilterObject;
qApp->installEventFilter(filter);

return a.exec();
}

  

在这个例子中,我们创建了一个简单的QWidget,安装了一个事件过滤器FilterObject,然后使用QApplication::exec()来启动程序。FilterObject重载了eventFilter函数,只有当接收到QEvent::KeyPress事件时才会处理,并将按键值打印出来。如果接收到其他类型的事件,那么就会继续传递给父对象进行处理。

二、eventfilter不触发

在使用eventfilter的过程中,有时会遇到eventFilter函数不触发的情况,该如何处理呢?

一般来说,eventFilter不会被调用有以下几种情况:

1、事件没有传递到需要接收的对象上。

2、事件被之前已经安装了事件过滤器的对象拦截了。

3、事件循环被堵塞了。

解决以上问题的方法一般有:

1、使用eventFilter继承父类,保证所有事件会向下传递。

2、将事件过滤器安装在最上层的QWidget。

3、检查程序中是否有死循环或递归调用等可能会阻塞事件循环的情况。

三、eventfilter崩溃

在使用eventfilter的过程中,有时会遇到eventFilter函数崩溃的情况,该如何处理呢?

如果eventFilter函数崩溃了,我们可以使用qDebug()进行调试。一种可能的情况是,eventFilter的参数obj或event为空指针,导致eventFilter函数在执行时崩溃。

另一种情况可能是对象已经被销毁了,而事件循环仍在不断地向被销毁的对象发送事件导致程序崩溃。我们可以在eventFilter函数中加入一段判断语句,当obj已经被销毁时,返回false。

bool eventFilter(QObject *obj, QEvent *event) override {
if (!obj) {
return false;
}
// standard event processing
return QObject::eventFilter(obj, event);
}

四、eventfilter qt

在Qt中使用eventfilter是很常见的技巧,常见的使用场景有:

1、在QLineEdit中只允许输入数字;

2、在QTableView中自定义鼠标操作;

3、在QListWidget中自定义鼠标操作。

五、eventfilter返回值

eventFilter函数的返回值决定了事件是否被过滤掉。如果返回true,表示事件已经被过滤了,需要停止传递;如果返回false,表示事件仍然需要传递给其他对象。

六、eventfilterwithquery选取

在某些情况下,我们希望只对某些特定事件进行过滤,可以使用QEvent::registerEventType()函数来自定义事件类型,在eventFilter函数中进行判断。

enum {
MyCustomEventType = QEvent::User + 1
};

class MyWidget : public QWidget
{
public:
MyWidget(QWidget *parent = nullptr);

protected:
bool event(QEvent *event) override;

private:
int mousePressCounter = 0;
};

MyWidget::MyWidget(QWidget *parent)
: QWidget(parent)
{
QApplication::instance()->installEventFilter(this);
QEvent::registerEventType(MyCustomEventType);
}

bool MyWidget::event(QEvent *event)
{
if (event->type() == MyCustomEventType) {
qDebug() << "MyCustomEventType received";
} else {
return QWidget::event(event);
}
}

class MyEventFilter : public QObject
{
protected:
bool eventFilter(QObject *obj, QEvent *event) override {
if (event->type() == QEvent::MouseButtonPress) {
QApplication::postEvent(obj, new QEvent((QEvent::Type)MyCustomEventType));
return true;
} else {
return QObject::eventFilter(obj, event);
}
}
};

int main(int argc, char *argv[])
{
QApplication a(argc, argv);
MyWidget widget;
widget.resize(200, 200);
widget.show();

MyEventFilter *filter = new MyEventFilter;
qApp->installEventFilter(filter);

return a.exec();
}

在这个例子中,我们自定义了一个事件类型MyCustomEventType,并在MyWidget中重载了event函数来接收该事件。在MyEventFilter中,我们监听了鼠标点击事件,当接收到该事件时,我们使用postEvent函数将自定义事件MyCustomEventType发送到接收该事件的对象上。

总结

eventfilter在Qt中是一项非常重要的技术,它可以帮助我们在一个对象中只接收需要的事件类型,从而对事件的处理做到精准控制。要在使用eventfilter的过程中注意处理可能遇到的触发和崩溃问题,同时可以通过自定义事件类型并结合使用QObject::event()函数来扩展更多的应用场景。