窗口最小化后隐藏


类型:C++ & Qt4,创建时间:Jan. 1, 2012, 5:49 p.m.

标题无“转载”即原创文章,版权所有。转载请注明来源:http://hgoldfish.com/blogs/article/65/。

通常最小化的窗口会在任务栏上显示一个按钮。当用户按一下这个按钮,窗口就还原了。这是一个被大多数桌面环境,比如Windows,KDE,GNOME所采用的普遍设计。不过,因为任务栏通常只是桌面边上一小行,容纳不了很多按钮,用户通常希望把那些不常用的程序隐藏起来,只在通知栏显示一个小图标,要使用的时候再点击那个小图标恢复窗口。这种作法最典型的例子是QQ和Winamp。

原本以为这个功能是很容易的,今天试着做了一下,原来还有些小麻烦。最基本的思路是拦截窗口的最小化事件,通过阅读Qt文档,可以这么做:

void MyWindow::changeEvent(QEvent* event)
{
    if(event->type()==QEvent::WindowStateChange)
    {
        if(windowState() & Qt::WindowMinimized)
        {
            hide();
            mSystemTrayIcon->show();
        }
    QMainWindow::changeEvent(event);
}

把这个代码拿去试一下,发现不行,任务栏的按钮仍然还在,点击还原后窗口一片空白。仔细想一下,估计是因为这个changeEvent()是在Qt处理最小化事件前调用的。我们把窗口隐藏了,但是Qt还以为窗口正要最小化。仍然运行一遍处理最小化窗口的代码。所以出错了。

解决的思路是先让Qt处理完窗口最小化,等处理完之后再隐藏窗口。这样的话,把hide()那一行改写成:

void MyWindow::changeEvent(QEvent* event)
{
    if(event->type()==QEvent::WindowStateChange)
    {
        if(windowState() & Qt::WindowMinimized)
        {
            QTimer::singleShot(0, this, SLOT(hide())); //修改这一行
            mSystemTrayIcon->show();
        }
    QMainWindow::changeEvent(event);
}

这个是singleShot()的特殊用法,如果第一个参数,也就是等待时间为0,当Qt回到消息循环的时候,通常就是当前处理的事件处理完之后就立即运行this->hide();

点击通知栏图标的处理代码需要多一下setWindowState()

MyWindow::showMe()
{
    show();
    setWindowState(Qt::WindowNoState);
}

标题无“转载”即原创文章,版权所有。转载请注明来源:http://hgoldfish.com/blogs/article/65/。


realerge(Jan. 2, 2012, 8:20 a.m.)

你好,看了您的文章,很有收获。谢谢。

我在实践中按照您的方法操作,最小化窗口没有问题了,但从托盘图标还原到正常大小的过程却始终有问题,现象是点击还原后,图标出现在任务栏,点击后窗口恢复,但是变黑,没有内容了。

如果我不通过最小化事件触发隐藏窗口,那么无论是最小化还是还原都正常,但通过最小化事件触发的话,不用您的方法,则在隐藏窗口的时候出问题,问题和您的描述一样,若用了您的方法,则在还原的时候出现问题。现象非常类似。

问题折磨了很久,实在难以解决,希望您能够帮助我。感谢您的阅读。


何不来发表一下您对本文的看法(使用Markdown语法,分段空两行):