将QDialog设置为外星人

Setting a QDialog to be an alien

本文关键字:外星人 设置 QDialog      更新时间:2023-10-16

在一个独立的GUI应用程序中,我没有窗口管理器也没有组合管理器,我想向用户显示一个QDialog来询问值。

这个对话框非常大,所以我想把它设置成半透明的,这样当对话框显示时,用户可以透过它看到应用程序中发生了什么。

问题是,对于X本机窗口的半透明,需要有一个复合管理器。Qt内部小部件可以被涂成半透明,因为它们不对应于本地X窗口(外星人),并且完全只有Qt知道。

有没有一种方法,使一个QDialog的背景半透明,而没有一个复合管理器运行?也许使它成为应用程序主窗口的正常子窗口?有更好的选择吗?

我不知道有什么方法可以把QDialog变成一个正常的子部件。看着Qt的X11代码,我无法找出一种不设置Qt::WindowFlags传递给QWidget(父)构造函数的方法,这样它就会是一个普通的小部件,而不是自己的窗口(但我可能是错的,没有花很多时间)。

一个简单的替代方法是使用普通的QWidget作为容器,而不是QDialog。下面是一个示例"PopupWidget",它绘制了一个半透明的红色背景。

#include <QtGui>
class PopupWidget: public QWidget
{
 Q_OBJECT
 public:
  PopupWidget(QWidget *parent): QWidget(parent)
  {
   QVBoxLayout *vl = new QVBoxLayout;
   QPushButton *pb = new QPushButton("on top!", this);
   vl->addWidget(pb);
   connect(pb, SIGNAL(clicked()), this, SLOT(hide()));
  }
 public slots:
  void popup() {
   setGeometry(0, 0, parentWidget()->width(), parentWidget()->height());
   raise();
   show();
  }
 protected:
  void paintEvent(QPaintEvent *)
  {
   QPainter p(this);
   QBrush b(QColor(255,0,0,128));
   p.fillRect(0, 0, width(), height(), b);
  }
};

为了显示它,调用它的popup()插槽,这将把它提升到小部件堆栈的顶部,使它和它的父级一样大,然后显示它。这将掩盖它后面的所有小部件(你不能用鼠标与它们交互)。当你点击那个按钮时,它会隐藏自己。

事项:

  • 这并不妨碍用户使用选项卡访问下面的小部件。这可以通过切换"正常"小部件容器上的enabled属性来修复。(但是不要禁用PopupWidget的父类:那样会禁用弹出窗口。)
  • 不允许阻塞调用,如QDialog::exec
  • 弹出窗口中的小部件不会是透明的,你必须为你需要的所有小部件类型创建自定义透明背景版本。

但是这可能比在您的环境中集成一个合成管理器要少一些麻烦。