关于eventFilter函数无法过滤QTableWidget鼠标事件的处理方式
问题描述
在一个QDialog中有一个QTableWidget,在dialog类中设置table的鼠标事件过滤,代码如下:
cpp
Dialog::Dialog(QWidget *parent)
: QDialog(parent)
, ui(new Ui::AddMultiplePointsDlg)
{
ui->setupUi(this);
ui->tableWidget->installEventFilter(this);
}
bool Dialog::eventFilter(QObject *object, QEvent *event)
{
if(object == ui->tableWidget)
{
if(event->type() == QEvent::MouseButtonRelease)
{
// 其他功能代码
}
}
return QDialog::eventFilter(object, event);
}
在上述代码中,我们想要的效果是,在table上点击鼠标时,程序能过滤到该事件,执行我们的代码。但在实际测试中发现,其他事件都能过滤的到,比如显示事件,键盘事件等。但唯独鼠标事件无法过滤,包括按下和释放时间都无法过滤
问题原因分析
QTableWidget、QTreeWidget等等这类控件,都继承自QAbstractScrollArea。而QAbstractScrollArea类的控件,并不只是本身的一个Wieget,内部还包含一个viewport。viewport也是一个widget,那些表格等item实际上是显示在viewport上的。也就是说,鼠标时间并没有传递给table,而是传递给了viewport。
解决方案
问题原因找到了,解决方法也很简单了,设置事件过滤不要设置table,而是要设置viewport,如果需要获取鼠标的位置,也要使用viewport的坐标,代码如下:
cpp
Dialog::Dialog(QWidget *parent)
: QDialog(parent)
, ui(new Ui::AddMultiplePointsDlg)
{
ui->setupUi(this);
ui->tableWidget->viewport()->installEventFilter(this);
}
bool Dialog::eventFilter(QObject *object, QEvent *event)
{
if(object == ui->tableWidget->viewport())
{
if(event->type() == QEvent::MouseButtonRelease)
{
QMouseEvent *mouseEvent = dynamic_cast<QMouseEvent*>(event);
if(mouseEvent)
{
QPoint pos = mouseEvent->pos();
// 其他功能代码
}
}
}
return QDialog::eventFilter(object, event);
}