一个用Qt开发的应用,运行在windows平台,任务栏多了一个窗口标签,显示的标题为“Dialog”。但是点击图标没有在桌面弹出窗口,可以理解为窗口是完全透明的,或者窗口区域完全位于显示器外面。那么如何从窗口图标找到对应的代码呢?
首先,我想到的是spy++。用spy++罗列该进程所有的窗口,找到标题为“Dialog”的可视窗口,幸运的是,刚好有一个,记下他的窗口句柄,窗口类,父类等信息。发现窗口类是qt的默认窗口类,进程中几乎所有的窗口都是这个窗口类,因此没有什么帮助;父类为空,也没有特别的帮助作用;剩下有点用处的就是窗口标题和窗口句柄了,记下来。
既然知道了窗口标题,可以通过查找ui文件来定位。在notepad++中打开任意一个ui文件,然后搜索窗口标题为“Dialog”的ui文件,查找模式为正则表达式,支持换行,规则为:【<property name="windowTitle">.{1,100}?Dialog】(不包括【】)。一轮查找下来,发现标题为“Dialog”的窗口还是很多,怎么进一步缩小范围呢?
如果在窗口显示时把句柄、窗口类、窗口标题等信息打印到日志,然后根据spy++获取的信息进行匹配,不就可以唯一定位到窗口代码了吗?于是,经过一番思索,觉得hook QApplication类,代码如下:
class QMyApplication : public QApplication
{
Q_OBJECT
public:
explicit QMyApplication(int& argc, char* argv[]);
protected:
virtual bool notify(QObject* obj, QEvent* event) override;
};
bool QMyApplication::notify(QObject* obj, QEvent* event)
{
if (event->type() == QEvent::Show) {
QWidget *pwgt = qobject_cast<QWidget*>(obj);
if (pwgt) {
qDebug() << obj << pwgt->winId();
}
}
return QApplication::notify(obj, event);
}
上述代码中,QApplication所有的消息都经过QMyApplication::notify函数进行过滤,我们只关心ShowEvent,在显示窗口是打印为窗口代码类和窗口句柄(pwgt->winId()),然后在日志中查找spy++中获取到的窗口句柄,就能定位到代码类了,此方法非常巧妙。
最后插一句题外话,如果不希望在任务栏显示窗口图标怎么办?这个很简单,一行代码即可解决:setWindowFlags(Qt::Tool | windowFlags());