我正在使用Qt Creator制作计算器。有一个带有gui的默认主窗口形式。在mainwindow.cpp中,我设置了setMouseTracking(true),以在单击时获取鼠标坐标。但是,只有在主窗口窗体中的任何按钮之外单击时,我才能获得鼠标坐标。我想这与以下事实有关:单击按钮时,已经在生成某种鼠标事件来处理该单击。单击按钮内时,我需要获取坐标。我该怎么办?
首先,这是一个坏主意:)
您正在尝试捕获发送到子对象的事件。一种方法是重新实现子对象的事件处理程序。但是,当您使用设计器构造接口时,这会变得有些困难。您必须将所有小部件升级为它们的重新实现版本。
另一种方法是使用Qt的“偶数过滤器”功能。您实现一个基于QObject的类,该类实现一个通用的事件处理函数。然后,将该对象作为“事件过滤器”安装在目标对象上。一个简单的偶数过滤器类看起来像这样;
class Filter : public QObject
{
protected:
bool eventFilter(QObject *obj, QEvent *ev)
{
// obj : original receiver of the event
// ev : any event that is sent to 'obj'
QObject::event(obj, ev);
}
};
及其用法:
Filter filterObject;
targetObject.installEventFilter(filterObject);
通常,必须将此事件过滤器安装到用户界面中的所有对象(窗口小部件)。但是,如果您将此过滤器安装到QApplication
实例上,而不是一一列举,它将捕获应用程序中的所有事件。捕获后,您可以按事件的类型和发起者来过滤事件。
这里是一个更完整的示例,它将捕获应用程序中的所有鼠标按下事件;
class ClickFilter : public QObject
{
protected:
bool eventFilter(QObject *obj, QEvent *ev)
{
if (ev->type() == QEvent::MouseButtonPress && obj->inherits("QWidgetWindow"))
{
QMouseEvent *mouseEvent = static_cast<QMouseEvent *>(ev);
qDebug() << "clicked:" << mouseEvent->pos();
}
return QObject::eventFilter(obj, ev);
}
};
您应该在QApplication
实例上,大概是在main
函数中,安装此事件过滤器对象;
QApplication a(argc, argv);
ClickFilter cf;
a.installEventFilter(&cf);
这将捕获应用程序中的所有事件(包括所有窗口-如果您的应用程序有多个主窗口,然后按类型和发起者对象类型对其进行过滤。
但是此代码存在问题。我已经使用QWidgetWindow
类型来检查事件发起者。那是因为,如果事件未由子窗口小部件处理,则该事件将传播到其父窗口。因此,在某些情况下,同一单击事件会多次调用ClickFilter::eventFilter
。我选择QWidgetWindow
是因为它看起来很简单。问题是; QWidgetWindow
是内部Qt类型。在您的应用程序中使用它不是一个好主意。我确定还有另一种防止重复的方法。
更重要的是,在QApplication
实例上安装事件过滤器不是一个好主意,它可能会导致性能问题。但是,另一方面,在我的应用程序中,我已经使用此方法轻松地将工具栏按钮的键盘快捷键添加到整个应用程序的工具提示中。您可以看到它here。