如何在单例内创建QML对象?

Sil*_*lex 3 qt qml qt5

我在下面定义了一个QML对象MyQMLObject.qml.此QML文件如下所示:

import QtQuick 2.4

Item {
  id: rootItem
  implicitWidth: LayoutUtils.maxImplicitWidth(children)
  implicitHeight: LayoutUtils.maxImplicitHeight(children)

  Text {
    id: text1
  }
  Text {
    id: text2
  }
  // ...
  Text {
    id: textN
  }
}
Run Code Online (Sandbox Code Playgroud)

应用程序启动时会动态添加文本.对于每种语言,添加不同的文本,其宽度rootItem因所选语言而异.我想以某种方式MyQMLObject在应用程序启动时只创建一次,甚至没有将其可视化并将其实际宽度保存在单例中,例如,因此我可以在整个代码中重用该值,而无需创建MyQMLObject多一次.我怎么能实现这个目标?

现在我有一个单独的QML文件,它包含一个QtObject包含一些常量值的文件.我可以以某种方式MyQMLObject在这个单例中创建一个实例QtObject吗?

我的单身人士Style.qml看起来像这样:

pragma Singleton

import QtQuick 2.4

QtObject {
  readonly property int maxWidth: 400
  // ...
}
Run Code Online (Sandbox Code Playgroud)

Mit*_*tch 12

首先,如果可能,您可以使用a Column而不是手动计算最大宽度:

MyQMLObject.qml

import QtQuick 2.4

Column {
    Text {
        text: "blah"
    }
    Text {
        text: "blahblah"
    }
}
Run Code Online (Sandbox Code Playgroud)

您可以使用动态对象创建来创建临时Column项:

Style.qml

pragma Singleton

import QtQuick 2.4

QtObject {
    readonly property int maxWidth: {
        var component = Qt.createComponent("qrc:/MyQMLObject.qml");
        if (component.status === Component.Error) {
            console.error(component.errorString());
            return 0;
        }

        return component.createObject().width;
    }
}
Run Code Online (Sandbox Code Playgroud)

main.qml

import QtQuick 2.5
import QtQuick.Window 2.2

import App 1.0

Window {
    visible: true

    Component.onCompleted: print(Style.maxWidth)
}
Run Code Online (Sandbox Code Playgroud)

然后,注册单身人士:

main.cpp中

#include <QtGui>
#include <QtQml>

int main(int argc, char *argv[])
{
    QGuiApplication app(argc, argv);

    qmlRegisterSingletonType(QUrl("qrc:///Style.qml"), "App", 1, 0, "Style");

    QQmlApplicationEngine engine;
    engine.load(QUrl(QStringLiteral("qrc:/main.qml")));

    return app.exec();
}
Run Code Online (Sandbox Code Playgroud)

但是,请注意,可以通过计算C++的最大宽度来改进这种方法,从而无需构建项目以将其丢弃.解决这个例子:

#include <QtGui>
#include <QtQml>

class Style : public QObject
{
    Q_OBJECT
    Q_PROPERTY(int maxWidth READ maxWidth CONSTANT)

public:
    Style(QObject* parent = 0) :
        QObject(parent),
        mMaxWidth(0)
    {
        QFontMetrics fontMetrics(qApp->font());
        // Here is where you'd fetch the text...
        QStringList dummyText;
        dummyText << "blah" << "blahblah";
        foreach (const QString &string, dummyText) {
            const int width = fontMetrics.boundingRect(string).width();
            if (width > mMaxWidth)
                mMaxWidth = width;
        }
    }

    int maxWidth() const
    {
        return mMaxWidth;
    }

private:
    int mMaxWidth;
};

static QObject *singletonTypeProvider(QQmlEngine *, QJSEngine *)
{

    Style *style = new Style();
    return style;
}

int main(int argc, char *argv[])
{
    QGuiApplication app(argc, argv);

    qmlRegisterSingletonType<Style>("App", 1, 0, "Style", singletonTypeProvider);

    QQmlApplicationEngine engine;
    engine.load(QUrl(QStringLiteral("qrc:/main.qml")));

    return app.exec();
}

#include "main.moc"
Run Code Online (Sandbox Code Playgroud)

它用于QFontMetrics计算宽度.

main.qml 保持不变.