简述
图形视图对将任何 widget 嵌入到场景中提供了无缝的支持。可以嵌入简单的 widget例如QLineEdit、QPushButton也可以是复杂的 widget例如QTabWidget甚至是完整的主窗口。
要将 widget 嵌入场景中只需要简单地调用 QGraphicsScene::addWidget()或者创建一个 QGraphicsProxyWidget 对象并将 widget 手工地嵌入其中。
QGraphicsProxyWidget
QGraphicsProxyWidget 类提供了一个代理层用于在 QGraphicsScene 中嵌入一个 QWidget。
QGraphicsProxyWidget 嵌入基于 QWidget 的部件例如QPushButton、QFontComboBox甚至 QFileDialog、QGraphicsScene。它在两个对象之间转发事件并在 QWidget 的基于整数的 geometry 和 QGraphicsWidget 的基于 qreal 的 geometry 之间进行转换。
QGraphicsProxyWidget 支持 QWidget 的所有核心功能包括Tab 焦点、键盘输入、拖放和弹出窗口。还可以嵌入复杂的 widget例如带有子 widget 的 widget。
QGraphicsProxyWidget 通过为每个 popup 创建一个子代理来负责嵌入 widget 的 popup 孩子的自动嵌入。这意味着当嵌入的 QComboBox 显示其弹出列表时会自动创建一个新的 QGraphicsProxyWidget嵌入 popup 并正确定位。这只有当 popup 是嵌入的 widget 的孩子时才有效例如 QToolButton::setMenu() 要求 QMenu 实例是 QToolButton 的孩子。
使用 QGraphicsProxyWidget 嵌入 Widget
使用 QGraphicsProxyWidget 嵌入一个 widget 有两种方法
- 将一个 widget 指针和任何相关的 Qt::WindowFlags 传递给 QGraphicsScene::addWidget() - 最常用的方式。
- 创建一个新的 QGraphicsProxyWidget然后调用 setWidget() 嵌入一个 QWidget。
例如在下面的代码段中我们在代理中嵌入了一个 QGroupBox
QGroupBox *pGroupBox = new QGroupBox();
QLabel *pTellLabel = new QLabel();
QLabel *pAddressLabel = new QLabel();
QLineEdit *pTellLineEdit = new QLineEdit();
QLineEdit *pAddressLineEdit = new QLineEdit();
pGroupBox->setTitle(QStringLiteral("联系方式"));
pTellLabel->setText(QStringLiteral("电话号码"));
pAddressLabel->setText(QStringLiteral("居住地址"));
pTellLineEdit->setPlaceholderText(QStringLiteral("手机/固话"));
pAddressLineEdit->setPlaceholderText(QStringLiteral("具体到门牌号"));
QFormLayout *pLayout = new QFormLayout;
pLayout->addRow(pTellLabel, pTellLineEdit);
pLayout->addRow(pAddressLabel, pAddressLineEdit);
pLayout->setSpacing(10);
pLayout->setMargin(20);
pGroupBox->setLayout(pLayout);
QGraphicsScene scene;
QGraphicsProxyWidget *pProxy = scene.addWidget(pGroupBox);
QGraphicsView view(&scene);
view.show();
QGraphicsProxyWidget 与 QWidget 共享所有权因此如果这两个 widget 中的任一个被销毁另一个也将被自动销毁。
同步 Widget 状态
QGraphicsProxyWidget 保持其与嵌入的 widget 状态同步。例如如果代理被隐藏或禁用嵌入的 widget 也将被隐藏或禁用反之亦然。
当 widget 通过调用 addWidget() 被嵌入时QGraphicsProxyWidget 将 widget 的状态例如visible、enabled、geometry、size hints复制到代理中然后两者将尽可能地保持同步。 默认情况下当一个 widget 被嵌入到代理中后widget 和代理都是可见的因为 QGraphicsWidget 在创建时是可见的不必调用 show()。如果显式隐藏嵌入的 widget 代理也将变为不可见。
示例
QGraphicsScene scene;
QLineEdit *edit = new QLineEdit;
QGraphicsProxyWidget *proxy = scene.addWidget(edit);
edit->isVisible(); // 返回 true
proxy->isVisible(); // 也返回 true
edit->hide();
edit->isVisible(); // 返回 false
proxy->isVisible(); // 也返回 false
QGraphicsProxyWidget 保持以下状态的对称性
QWidget 状态 | QGraphicsProxyWidget 状态 | 说明 |
---|---|---|
QWidget::enabled | QGraphicsProxyWidget::enabled | |
QWidget::visible | QGraphicsProxyWidget::visible | 显式状态也是对称的 |
QWidget::geometry | QGraphicsProxyWidget::geometry | 当嵌入的 widget 可见时Geometry 仅保证是对称的 |
QWidget::layoutDirection | QGraphicsProxyWidget::layoutDirection | |
QWidget::style | QGraphicsProxyWidget::style | |
QWidget::palette | QGraphicsProxyWidget::palette | |
QWidget::font | QGraphicsProxyWidget::font | |
QWidget::cursor | QGraphicsProxyWidget::cursor | 嵌入的 widget 覆盖代理光标代理光标的改变依赖嵌入的子 widget 当前在鼠标下。 |
QWidget::sizeHint() | QGraphicsProxyWidget::sizeHint() | 嵌入的 widget 的所有大小提示功能均由代理转发 |
QWidget::getContentsMargins() | QGraphicsProxyWidget::getContentsMargins() | 通过 setWidget() 更新一次 |
QWidget::windowTitle | QGraphicsProxyWidget::windowTitle | 通过 setWidget() 更新一次 |
注意当 widget 被嵌入时QGraphicsScene 保持嵌入的 widget 在一个特殊的状态防止它影响其他 widget嵌入和非嵌入。在此状态下widget 在行为上可能与未嵌入时略有不同。
警告 QGraphicsProxyWidget 类是为了便于桥接 QWidgets 和 QGraphicsItems 而提供不应该用于高性能场景。