【问题标题】:Resizing a QGraphicsItem to take up all space in a QGraphicsView, problems when window is resized调整 QGraphicsItem 的大小以占用 QGraphicsView 中的所有空间,调整窗口大小时出现问题
【发布时间】:2010-10-24 20:28:39
【问题描述】:

我有一个QGraphicsItem(实际上是一个QDeclarativeItem),我希望它占据QGraphicsView 的整个可见空间(同样,它实际上是派生的QDeclarativeView 类),它被添加到其中.通常,您可以使用QDeclarativeView::setResizeMode(QDeclarativeView::SizeRootObjectToView)QDeclarativeView 会自动调整根对象的大小以适应视图。

我遇到的问题是我正在使用手动创建根小部件

QDeclarativeComponent component(declarativeView->engine(), QUrl(qml));
QDeclarativeItem* object = qobject_cast<QDeclarativeItem*>(component.create());
if (object)
{
    declarativeView->scene()->addItem(object);
    ...

而不是让QDeclarativeView 通过调用setSource() 自动执行此操作。我这样做的原因是因为我想在某些事件发生时交换 QML 场景,但我不想破坏之前的场景。调用 setSource() 会删除在调用 setSource() 之前添加的所有项目。因此,我自己创建“根对象”并手动将其添加到场景中。

我正在使用 windows resizeEvent 来调整我的QDeclarativeItem 的大小,如下所示:

void AppWindow::resizeEvent (QResizeEvent* event)
{
    QDeclarativeItem* object = sceneCache.value(sceneName, 0);
    if (object)
    {
        object->setWidth(declarativeView->viewport()->width());
        object->setHeight(declarativeView->viewport()->height());
    }
}

这确实有效!但是,它不是很漂亮。如果您快速调整窗口大小,QDeclarativeItem 的大小调整不够快,您会在它赶上并调整大小之前短暂看到灰色背景。它也不是很流畅。

只有在调整复杂项目的大小时才会发生这种情况(在我的例子中,它是一个 QWebKit 小部件)。它适用于更简单的项目。然而,问题是,如果我让QDeclarativeView 去做,我就没有这些问题:它的大小调整正确且平滑。

我想这不是特定于 QtDeclarative 的东西,而是 QGraphicsView,尽管我可能错了。

有人有什么想法吗?

【问题讨论】:

    标签: c++ qt qgraphicsview qtdeclarative qt-quick


    【解决方案1】:

    我的问题已经解决了。

    问题似乎出在 QML 的 WebView 元素上。我将它切换为从 C++ 注册的 QGraphicsProxyWidget,将小部件设置为普通的 QWebView,所有问题都消失了。它现在的行为完全符合我对 WebView 的预期行为。

    缺点是我需要手动将 QWebView 信号/插槽/属性公开给 QML。 (编辑:我已经通过一个名为object 的只读属性解决了这个问题,该属性返回设置的小部件。由于 QObjects 自动转换为 javascript 可访问对象,因此我可以访问 Web 视图,例如,如下所示:@ 987654322@。为方便起见,我还在 QML 对象上调用了一个 init javascript 函数(如果它存在的话),所以我只需将 web 视图设置在那里 - 效果很好!)

    自从我提出这个问题以来,我所做的另一项更改是将“根对象”从 C++ 切换到 QML 本身,方法是使用 C++ 可调用的 JavaScript 函数创建一个全屏矩形,该函数使用 Qt.createComponent() 加载“实际”QML 文件并使用 anchors.fill 锚定它。这样,逻辑(和对象的缓存)在 QML 中完成,而不是在 C++ 中完成,所以我可以简单地 setSource(),让 Qt 处理大小调整等,而我纯粹在 QML 中工作(无论如何,对于应用程序的这一部分)。

    【讨论】:

      【解决方案2】:

      我不确定这是否会消除闪烁,但您可以在 resizeEvent() 中使用 fitInView

      declarativeView->fitInView(object, Qt::IgnoreAspectRatio);
      

      【讨论】:

      • 谢谢你,我好像错过了那个功能。
      • 它起作用了,虽然我不知道为什么内容在视图中居中显示类似于白色边框的背景颜色(白色)的最小部分。有什么提示吗?
      最近更新 更多