Evgenii Legotckoi
Шілде 5, 2017, 4:05 Т.Қ.

QML сабағы 029. QML жүйесінде "Статикалық" әдістерді пайдалану үшін Singleton нысанын тіркеу

Понятие статических методов, которые используются в QML несколько отличается от классического в C++, когда в классе создаются статические методы, которые можно использовать обращаясь к имени класса, а не к какому-то конкретному объекту. В случае с QML дела обстоят несколько иначе. Для того, чтобы использовать такие методы в QML, которые присутствуют в C++ классе, необходимо зарегистрировать Синглетон объект, который будет предоставлять требуемые методы, причём эти методы уже не должны быть статическими сами по себе. Они должны как минимум быть помечены макросом Q_INVOKABLE , чтобы их можно было использовать в QML.

Для регистрации Синглетона необходимо использовать функцию qmlRegisterSingletonType , в которую помимо стандартных параметров, которые передаются в qmlRegisterType , необходимо также передать статическую функцию singletonProvider , которую также предстоит написать самим.


Структура проекта

Создайте Qt Quick проект, в котором будут содержаться следующие файлы:

  • SingletonQMLCpp.pro - профайл проекта
  • main.cpp - файл с main функцией
  • main.qml - основной файл qml
  • singletonclass.h - заголовочный файл синглетона
  • singletonclass.cpp - файл исходных кодов синглетона

Профайл проекта будет создан по умолчанию и не будет изменяться.

main.cpp

Начнём с файла main.cpp, в котором зарегистрируем объект будущего синглетона.

  1. #include <QGuiApplication>
  2. #include <QQmlApplicationEngine>
  3. #include <QtQml> // Подключаем для использования qmlRegisterSingletonType
  4.  
  5. #include "singletonclass.h" // Подключаем заголовочный файл SingletonClass
  6.  
  7. int main(int argc, char *argv[])
  8. {
  9. QGuiApplication app(argc, argv);
  10.  
  11.   // регистрируем объект синглетона через функцию singletonProvider, то есть передаём указатель на неё в качестве аргумента.
  12. qmlRegisterSingletonType<SingletonClass>("SingletonClass", 1, 0, "SingletonClass", singletonProvider);
  13.  
  14. QQmlApplicationEngine engine;
  15. engine.load(QUrl(QStringLiteral("qrc:/main.qml")));
  16. if (engine.rootObjects().isEmpty())
  17. return -1;
  18.  
  19. return app.exec();
  20. }

singletonclass.h

В заголовочном файле определим enum , который будем использовать для получения определённых сообщений через "статический" метод getMessage в QML слое. В этом же файле определим статическую функцию singletonProvider , которая будет создавать инстанс синглетона.

  1. #ifndef SINGLETONCLASS_H
  2. #define SINGLETONCLASS_H
  3.  
  4. #include <QObject>
  5. #include <QQmlEngine>
  6. #include <QJSEngine>
  7.  
  8. class SingletonClass : public QObject
  9. {
  10. Q_OBJECT
  11. public:
  12. explicit SingletonClass(QObject *parent = nullptr);
  13.  
  14. enum class Message {
  15. Info,
  16. Debug,
  17. Warning,
  18. Error
  19. };
  20. // Регистрация enum для испоьлзования в QML
  21. Q_ENUM(Message)
  22.  
  23. // Чтобы использовать этот метод в QML, объявите его на Q_INVOKABLE
  24. Q_INVOKABLE QString getMessage(Message message);
  25.  
  26. signals:
  27.  
  28. public slots:
  29. };
  30.  
  31. static QObject *singletonProvider(QQmlEngine *engine, QJSEngine *scriptEngine)
  32. {
  33. Q_UNUSED(engine)
  34. Q_UNUSED(scriptEngine)
  35.  
  36. SingletonClass *singletonClass = new SingletonClass();
  37. return singletonClass;
  38. }
  39.  
  40. #endif // SINGLETONCLASS_H

Заметьте, что getMessage не обязан быть статическим, чтобы выступать в качестве статического метода Типа в QML.

singletonclass.cpp

В этом файле дана реализация метода getMessage.

  1. #include "singletonclass.h"
  2.  
  3. SingletonClass::SingletonClass(QObject *parent) : QObject(parent)
  4. {
  5.  
  6. }
  7.  
  8. QString SingletonClass::getMessage(SingletonClass::Message message)
  9. {
  10. switch (message)
  11. {
  12. case Message::Info:
  13. return "This is Info Message";
  14. case Message::Debug:
  15. return "This is Debug Message";
  16. case Message::Warning:
  17. return "This is Warning Message";
  18. case Message::Error:
  19. return "This is Error Message";
  20. default:
  21. return "Nothin not found";
  22. }
  23. }

main.qml

А теперь воспользуемся этим синглетон типом в QML.

  1. import QtQuick 2.5
  2. import QtQuick.Window 2.2
  3. import QtQuick.Controls 1.4
  4.  
  5. import SingletonClass 1.0 // Подключаем Синглетон
  6.  
  7.  
  8. Window {
  9. visible: true
  10. width: 640
  11. height: 480
  12. title: qsTr("Singleton Class")
  13.  
  14. ListView {
  15. anchors.fill: parent
  16. delegate: Item {
  17. height: 48
  18. width: parent.width
  19. Text {
  20. anchors.fill: parent
  21. text: model.text
  22.  
  23. verticalAlignment: Text.AlignVCenter
  24. horizontalAlignment: Text.AlignHCenter
  25. }
  26. }
  27.  
  28. model: listModel
  29. }
  30.  
  31. ListModel {
  32. id: listModel
  33.  
  34.   // Нет возможности использовать методы и функции сразу для property в ListElement, поэтому использован такой хак
  35. Component.onCompleted: {
  36. listModel.append({'text': SingletonClass.getMessage(SingletonClass.Info)})
  37. listModel.append({'text': SingletonClass.getMessage(SingletonClass.Debug)})
  38. listModel.append({'text': SingletonClass.getMessage(SingletonClass.Warning)})
  39. listModel.append({'text': SingletonClass.getMessage(SingletonClass.Error)})
  40. }
  41. }
  42. }

Итог

В результате будет получен приложение, которые выведет четыре сообщения в своём окне.

Таким образом через синглетон объект в QML добавляется возможность использования cpp методов в качестве статических.

Скачать архив с проектом

Ол саған ұнайды ма? Әлеуметтік желілерде бөлісіңіз!

3
  • Қаз. 2, 2018, 1:05 Т.Ж.

In the debugger, I could not manage to see if the constructor is called once for all or if is called at every occurence from QML. Do you know?

Evgenii Legotckoi
  • Қаз. 5, 2018, 6:11 Т.Қ.

For QML constructor will be called once.

It works some differently from classic Singleton, but instance will be one.

Пікірлер

Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
Кіріңіз немесе Тіркеліңіз