调整大小时不要重新绘制窗口

day*_*day 6 qt qml qt-quick qtquick2

我的QML应用程序(Qt 5.4)基于一个Window项目.用户可以调整应用程序的大小.调整应用程序大小时,将分别调整应用程序的内容(使用onWidthChangedonHeightChanged).

这一切都很好.

但是为了避免闪烁,我不希望在应用程序调整大小时更新应用程序的内容.QML中是否有可能检测用户何时实际调整窗口大小(将鼠标按钮按住窗口边框)并且在调整大小完成之前不重新计算内容(释放鼠标按钮)?

Gre*_*cKo 1

编辑:Kuba Ober 建议的更加简单和强大,我仍然会在这里留下我的答案,因为我发现它有点有趣(并且可以修改 C++ 自定义组件方法以按照建议过滤窗口事件)。


请原谅我,但我写了一个快速而丑陋的黑客来看看是否可能,它只涵盖了你问题的第二部分(不更新内容)。我的解决方案会阻止项目的重新绘制,但也会在请求更新时立即隐藏它(这对您来说可能不是问题)。

阅读QQuickItem::updatePaintNode文档,尤其是这句话后

如果用户在项目上设置了 QQuickItem::ItemHasContents 标志,则该函数将作为 QQuickItem::update() 的结果被调用。

我创建了一个 C++ 类来在任意 QQuickItem 上设置/取消设置此标志:

#ifndef ITEMUPDATEBLOCKER_H
#define ITEMUPDATEBLOCKER_H

#include <QObject>
#include <QQuickItem>


class ItemUpdateBlocker : public QObject
{
    Q_OBJECT
    Q_PROPERTY(QQuickItem* target READ target WRITE setTarget NOTIFY targetChanged)
    QQuickItem* m_target;

public:
    explicit ItemUpdateBlocker(QObject *parent = 0) : QObject(parent), m_target(nullptr) {  }
    QQuickItem* target() const { return m_target; }

signals:
    void targetChanged();

private:
    static void blockUpdate(QQuickItem* target)
    {
        if (target)
            target->setFlag(QQuickItem::ItemHasContents, false);
    }

    static void unblockUpdate(QQuickItem* target)
    {
        if (target)
        {
            target->setFlag(QQuickItem::ItemHasContents, true);
            target->update();
        }
    }


public slots:
    void setTarget(QQuickItem* target)
    {
        if (m_target == target)
            return;
        unblockUpdate(m_target);
        blockUpdate(target);
        m_target = target;
        emit targetChanged();
    }
};

#endif // ITEMUPDATEBLOCKER_H
Run Code Online (Sandbox Code Playgroud)

下一步是注册此类,以便可以在 QML 中使用它:

qmlRegisterType<ItemUpdateBlocker>("com.mycompany.qmlcomponents", 1, 0, "ItemUpdateBlocker");
Run Code Online (Sandbox Code Playgroud)

你可以像这样在 QML 中使用它:

import QtQuick 2.4
import QtQuick.Controls 1.3
import QtQuick.Window 2.2
import com.mycompany.qmlcomponents 1.0

ApplicationWindow {

    width: 640
    height: 480
    visible: true

    Rectangle {
        color: "red"
        id: root
        anchors.fill: parent

        Text {
            text: blocker.target ? "Blocked" : "Not Blocked"
        }

        Rectangle {
            color: "white"
            anchors.centerIn: parent
            width: parent.width/2
            height: parent.height/2

            ItemUpdateBlocker {
                id: blocker;
            }

            MouseArea {
                anchors.fill: parent
                onClicked: blocker.target = blocker.target ? null : parent
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

当然,您可以active向阻止程序添加一个属性来简化它的使用(比使用 nulltarget来禁用它更漂亮),但我将把它作为练习。

也许您可以在宽度或高度Window发生变化时启动计时器来使用它,我还没有找到一种直接的方法来查找窗口是否调整了大小。