为什么延迟加载的 QWidget 会显示出来,而存储的 QWidget 却不会?

Why does a lazy-loaded QWidget get displayed but a stored one does not?

我有一组出现在不同地方的小弹出窗口小部件,但每次只出现一个。对于简单的功能,new-to-show 和 delete-to-hide 是可以的,并且可以正常工作,但是当它们开始处理自己的数据时,我可以看到内存泄漏。

因此,因为我只需要每种类型中的一个,所以我想我应该在父构造函数中预先创建所有这些,然后根据需要显示和隐藏它们。据我所知,这应该有效,但 popup->show() 没有显示。此示例所基于的复杂应用程序显示弹出窗口确实存在于正确的位置并且可以与用户交互...除了它是不可见的。

这是显示的惰性版本:

#ifndef MAIN_H
#define MAIN_H

#include <QtWidgets>

class Popup : public QLabel
{
    Q_OBJECT
public:
    explicit Popup(int x, int y, int width, int height, QWidget* parent = 0);
};

class MainWindow : public QMainWindow
{
    Q_OBJECT
public:
    MainWindow(QWidget* parent = 0);
    ~MainWindow() {}
    void mousePressEvent(QMouseEvent* ev);
private:
    Popup* popup;
};

#endif // MAIN_H

/***************
*** main.cpp ***
***************/
#include "main.h"
#include <QApplication>

int main(int argc, char* argv[])
{
    QApplication a(argc, argv);
    MainWindow w;
    w.show();
    return a.exec();
}



MainWindow::MainWindow(QWidget* parent)
    : QMainWindow(parent)
{
    popup = 0;

    QWidget* cWidget = new QWidget(this);
    cWidget->setStyleSheet("background-color: lightgray");
    setCentralWidget(cWidget);

    showMaximized();
}

void MainWindow::mousePressEvent(QMouseEvent* ev)
{
    if(popup != 0)
    {
        if(!popup->geometry().contains(ev->x(), ev->y()))
        {
            delete popup;
            popup = 0;
        }
    }
    else
    {
        popup = new Popup(ev->x(), ev->y(), 100, 100, this);
        popup->show();
    }
    ev->accept();
}



Popup::Popup(int x, int y, int width, int height, QWidget* parent) :
    QLabel(parent)
{
    setStyleSheet("background-color: black");
    setGeometry(
                x - (width  / 2),   // Left
                y - (height / 2),   // Top
                width ,             // Width
                height              // Height
                );
}

这里是未显示的预创建版本:

#ifndef MAIN_H
#define MAIN_H

#include <QtWidgets>

class Popup : public QLabel
{
    Q_OBJECT
public:
    explicit Popup(QWidget* parent = 0);
    void setup(int x, int y, int width, int height);
};

class MainWindow : public QMainWindow
{
    Q_OBJECT
public:
    MainWindow(QWidget* parent = 0);
    ~MainWindow() {}
    void mousePressEvent(QMouseEvent* ev);
private:
    Popup* popup;
};

#endif // MAIN_H

/***************
*** main.cpp ***
***************/
#include "main.h"
#include <QApplication>

int main(int argc, char* argv[])
{
    QApplication a(argc, argv);
    MainWindow w;
    w.show();
    return a.exec();
}



MainWindow::MainWindow(QWidget* parent)
    : QMainWindow(parent)
{
    popup = new Popup(this);

    QWidget* cWidget = new QWidget(this);
    cWidget->setStyleSheet("background-color: lightgray");
    setCentralWidget(cWidget);

    showMaximized();
}

void MainWindow::mousePressEvent(QMouseEvent* ev)
{
    if(popup->isVisible())
    {
        if(!popup->geometry().contains(ev->x(), ev->y()))
        {
            popup->hide();
        }
    }
    else
    {
        popup->setup(ev->x(), ev->y(), 100, 100);
        popup->show();
    }
    ev->accept();
}



Popup::Popup(QWidget* parent) :
    QLabel(parent)
{
    setStyleSheet("background-color: black");
}

void Popup::setup(int x, int y, int width, int height)
{
    setGeometry(
                x - (width  / 2),   // Left
                y - (height / 2),   // Top
                width ,             // Width
                height              // Height
                );
}

我错过了什么?

您从未为 popup 小部件设置 window 标志(例如 Qt::Popup)。
它实际上是 MainWindow w 的子窗口小部件,应该显示在某处。

在 Qt 中,没有布局的 QWidget 只是简单地堆叠在一起。出现的 z 顺序取决于实例化的顺序。这就是为什么惰性 QLabel 可见而另一个不可见的原因。

实用在cWidget之后实例化弹出窗口可能就足够了:

MainWindow::MainWindow(QWidget* parent)
    : QMainWindow(parent)
{
    // popup = new Popup(this); // not here

    QWidget* cWidget = new QWidget(this);
    cWidget->setStyleSheet("background-color: lightgray");
    setCentralWidget(cWidget);

    popup = new Popup(this); // but here (untested, but should work)

    showMaximized();
}

但是要以 Qt 方式执行此操作,您应该避免堆叠小部件,而是给您的 popup 一个真正的弹出窗口外观:

popup->setWindowFlags(Qt::Popup); // this lets popup have its own window

Martin 提到了 z 顺序,我还没有与之建立联系,促使我进行了更多谷歌搜索,很快找到了 QWidget::raise() 方法。现在它按我预期的方式工作。

实例化顺序确实有效,但如果您动态创建更多小部件,则无效。弹出窗口仍会出现在动态小部件下方。

我确定有办法让它工作(这个 我的第一个 Qt 项目,所以我完全希望在几年后将它命名为 "ugly"现在尽管我尽了最大的努力),但是 window 标志打破了弹出窗口和应用程序已经编写的方式之间的耦合。如果我从 window 标志开始,我可能会让它工作得很好甚至更好,但我没有。

这是我现在正在做的,有效:

#ifndef MAIN_H
#define MAIN_H

#include <QtWidgets>

class Popup : public QLabel
{
    Q_OBJECT
public:
    explicit Popup(QWidget* parent = 0);
    void setup(int x, int y, int width, int height);
};

class MainWindow : public QMainWindow
{
    Q_OBJECT
public:
    MainWindow(QWidget* parent = 0);
    ~MainWindow() {}
    void mousePressEvent(QMouseEvent* ev);
private:
    Popup* popup;
};

#endif // MAIN_H

/***************
*** main.cpp ***
***************/
#include "main.h"
#include <QApplication>

int main(int argc, char* argv[])
{
    QApplication a(argc, argv);
    MainWindow w;
    w.show();
    return a.exec();
}



MainWindow::MainWindow(QWidget* parent)
    : QMainWindow(parent)
{
    popup = new Popup(this);

    QWidget* cWidget = new QWidget(this);
    cWidget->setStyleSheet("background-color: lightgray");
    setCentralWidget(cWidget);

    showMaximized();
}

void MainWindow::mousePressEvent(QMouseEvent* ev)
{
    if(popup->isVisible())
    {
        if(!popup->geometry().contains(ev->x(), ev->y()))
        {
            popup->hide();
        }
    }
    else
    {
        popup->setup(ev->x(), ev->y(), 100, 100);
        popup->raise();     /*** THIS IS WHAT I WAS MISSING!!! ***/
        popup->show();
    }
    ev->accept();
}



Popup::Popup(QWidget* parent) :
    QLabel(parent)
{
    setStyleSheet("background-color: black");
}

void Popup::setup(int x, int y, int width, int height)
{
    setGeometry(
                x - (width  / 2),   // Left
                y - (height / 2),   // Top
                width ,             // Width
                height              // Height
                );
}