Evgenii Legotckoi
Nov. 27, 2015, 8:54 p.m.

QML - Lesson 018. Loader in QML Qt – The working with the dynamic components

For the organization of the dynamic components of the change is convenient to use a Loader component, which is included in QML QtQuick and is a container for components in your application, let's say that you need to periodically replace the interface.

To draw an analogy, for example, with the development of Java for Android , there is a system of fragments, which can also be replaced in the container for them, following the logic of your application. For example, we click on the button and in a particular container, we replaced one fragment of another, and if we click on another button, then there is a third fragment, which replaces the second fragment is.

Therefore, make an application that will have 5 buttons and pressing each button in the Loader fragments will vary.

The structure of the project to work with the Loader

  • QmlLoader.pro - the profile of the project;
  • main.cpp - the main file of the application source code;
  • main.qml - basic qml code file;
  • Fragment1.qml - the first fragment for replacement Loader;
  • Fragment2.qml - second fragment;
  • Fragment3.qml - a third fragment.

main.cpp is created by default and is not subject to change, so it will not be given a listing, as well as the project profile.

main.qml

Installing components in the Loader can be done in several ways:

  1. A source property - in this case the component is a QML file and its contents is limited scope in terms of interaction with other elements in main.qml file;
  2. Through sourceComponent property - in this case, set Component objects that are contained in the file main.qml and not limited scope when interacting with other objects in main.qml;
  3. Installation via setSource() method - in this case, it is possible to specify additional options for an object, such as opacity, but do not forget to reset these settings, or they persist even when you install a new component.

In the following the above, there is a layer with five buttons, the Loader object, which is a rectangular container in which to put our pieces. Working with its location in the application window is made as if it were an object of type Rectangle .

  1. import QtQuick 2.5
  2. import QtQuick.Controls 1.4
  3. import QtQuick.Layouts 1.1
  4.  
  5. ApplicationWindow {
  6. visible: true
  7. width: 640
  8. height: 480
  9. title: qsTr("Hello World")
  10.  
  11. // Layer with buttons that will change the fragments
  12. RowLayout {
  13. id: rowLayout
  14. anchors.top: parent.top
  15. anchors.left: parent.left
  16. anchors.right: parent.right
  17. anchors.margins: 5
  18.  
  19. Button {
  20. text: qsTr("Fragment 1")
  21. // Download the component from a file
  22. onClicked: loader.source = "Fragment1.qml"
  23. }
  24.  
  25. Button {
  26. text: qsTr("Fragment 2")
  27. // Loading setSource component through the method of installing the fragment parameters
  28. onClicked: loader.setSource("Fragment2.qml", {"opactity": 1 })
  29. }
  30.  
  31. Button {
  32. text: qsTr("Fragment 3")
  33. // Loading setSource component through the method of installing the fragment parameters
  34. onClicked: loader.setSource("Fragment3.qml", {"opacity": 0.5})
  35. }
  36.  
  37. Button {
  38. text: qsTr("Fragment 4")
  39. // Installing a fragment from the Component
  40. onClicked: loader.sourceComponent = fragment4
  41. }
  42.  
  43. Button {
  44. text: qsTr("Fragment 5")
  45. // Installing a fragment from the Component
  46. onClicked: loader.sourceComponent = fragment5
  47. }
  48. }
  49.  
  50. Loader {
  51. id: loader
  52. anchors.top: rowLayout.bottom
  53. anchors.left: parent.left
  54. anchors.right: parent.right
  55. anchors.bottom: parent.bottom
  56. anchors.topMargin: 5
  57. source: "Fragment1.qml"
  58. }
  59.  
  60. // The fourth fragment as component
  61. Component {
  62. id: fragment4
  63.  
  64. Rectangle {
  65. anchors.fill: parent
  66. color: "blue"
  67.  
  68. Text {
  69. text: "Fragment 5"
  70. color: "white"
  71. anchors.top: parent.top
  72. anchors.right: parent.right
  73. anchors.margins: 50
  74. font.pixelSize: 30
  75.  
  76. renderType: Text.NativeRendering
  77. }
  78.  
  79. }
  80. }
  81.  
  82. // The fifth fragment as component
  83. Component {
  84. id: fragment5
  85.  
  86. Rectangle {
  87. anchors.fill: parent
  88. color: "black"
  89.  
  90. Text {
  91. text: "Fragment 5"
  92. color: "white"
  93. anchors.top: parent.top
  94. anchors.right: parent.right
  95. anchors.margins: 50
  96. font.pixelSize: 30
  97.  
  98. renderType: Text.NativeRendering
  99. }
  100.  
  101. }
  102. }
  103. }

Fragment1.qml and the remaining fragments

And other code fragments Fragment1.qml identical except that they all have a different background color.

  1. import QtQuick 2.5
  2.  
  3. Rectangle {
  4. anchors.fill: parent
  5. color: "green"
  6.  
  7. Text {
  8. text: "Fragment 1"
  9. color: "white"
  10. anchors.top: parent.top
  11. anchors.right: parent.right
  12. anchors.margins: 50
  13. font.pixelSize: 30
  14.  
  15. renderType: Text.NativeRendering
  16. }
  17.  
  18. }

Conclusion

Video

Do you like it? Share on social networks!

S
  • Oct. 18, 2017, 7:11 p.m.

Добрый день интересует такой вопрос. С помощью QML возможно ли создавать многооконные приложения? Есть ли аналоги MDI area? Гугл не помогает, в документации только однооконные приложения с регистрацией в main.cpp. Какая практика вообще существует использования qml в создании больших приложений (многооконных).

Evgenii Legotckoi
  • Oct. 18, 2017, 7:28 p.m.

День добрый!!

В QML нет аналога MDI area, но можно сделать стандартный проект на QWidget, добавить в главное окно QMdiArea, а в него добавлять QML-ки в качестве виджетов через QQuickWidget. Вот статья для примера, как отобразить виджет с QML в проекте на QWidget .

 
То есть получается некий костыль. В принципе, можно сделать и собственную MDIArea для QML, теоретически... это не должно составить труда. Но нужно подумать. Там по идее, нужно лишь сделать возможности для обработки изменения размера объектов, которые будут представлять из себя эти самые окна. А также реализовать логику перетаскивания окон и скроллинга для этой MDI area.
 
Просто многооконные приложение можно без проблем создавать на QML, вопрос лишь в том, насколько удачно можно реализовать некоторые сложные структуры интерфейса наподобие древовидных списков и этой самой MDI area.
S
  • Oct. 18, 2017, 7:37 p.m.

Вот и я тоже пришел к такому же выводу, стандартное QWidget как обертка для QuickWidget-ов. А вообще практика применения QML для десктопов видимо такая же, я не нашел информации на этот счет вообще .

Evgenii Legotckoi
  • Oct. 18, 2017, 7:43 p.m.

Я не встречался с повсеместным использованием QML для десктопа, если приложение также не затачивается на работу под мобильные платформы. Видимо, поэтому и MDI area до сих пор не существует для QML, поскольку имеет крайне низкий приоритет.

t
  • Nov. 1, 2019, 9:28 p.m.
  • (edited)

Добрый день! Благодарю за хороший пример, только его бы дополнить немного: что если мне нужно из фрагментов "Fragment 1.qml" - "Fragment 2.qml" посылать сигналы, причём набор сигналов разный. Как это лучше организовать, какие тут есть варианты?

P.S. Сам спросил - сам отвечаю: :)

  1. Loader
  2. {
  3. onLoaded: {
  4. if(item.objectName == "Fragment1") { // Set object name to an item beforehand
  5. item.mySignal1.connect(targetObject.function1);
  6. item.mySignal2.connect(targetObject.function2);
  7. }
  8. }
  9. }

Another variant: move above logic inside

  1. Fragment1.qml
and call these functions via property like this:

  1. property var LoaderParent: parent.parent
  2. ...
  3. onMySignal1:{
  4. LoaderParent.function1(params);
  5. }
Evgenii Legotckoi
  • Nov. 4, 2019, 5:16 p.m.

Добрый день. Не успел Вам ответить ))
Для ответа на ваш вопрос, по моему мнению лучше было бы объединить всю информациб о сигналах и слотах, которая присутсвует на сайте.
Поэтому я первоначально потратил время на подготовку статьи, касательно вашего вопроса.
Можете прочитать здесь QML - Урок 036. Работа с сигналами и слотами в QML .
Надеюсь, что найдёте ещё что-нибудь дополнительное для себя.

t
  • Nov. 4, 2019, 8:34 p.m.

Благодарю!

T
  • Feb. 5, 2021, 3:21 p.m.

Всем привет. А есть ли возможность динамически загрузить qml файл, который скачивается во время работы программы и помещается в определённую папку на диске C?

Evgenii Legotckoi
  • July 2, 2021, 5:39 p.m.

Да можно, нужно указать относительный путь к qml файлу

добрый день, грамотно ли использовать loader для загрузки небольших диалоговых окон по клику из меню? и если да, то возникает проблема: загрузили первое диалоговое окно, потом его закрыли, а открыть снова получается только после згразки другого д.о. Проблема в свойтве status?

Лично для меня loader - это компонень, который загружает какую-то часть внутри окна, поэтому с этой точки зрения я бы не стал рассматривать использование loader лоя открытия окон, только для заполнения этих окон контентом.

Comments

Only authorized users can post comments.
Please, Log in or Sign up
  • Last comments
  • AK
    April 1, 2025, 11:41 a.m.
    Добрый день. В данный момент работаю над проектом, где необходимо выводить звук из программы в определенное аудиоустройство (колонки, наушники, виртуальный кабель и т.д). Пишу на Qt5.12.12 поско…
  • Evgenii Legotckoi
    March 9, 2025, 9:02 p.m.
    К сожалению, я этого подсказать не могу, поскольку у меня нет необходимости в обходе блокировок и т.д. Поэтому я и не задавался решением этой проблемы. Ну выглядит так, что вам действитель…
  • VP
    March 9, 2025, 4:14 p.m.
    Здравствуйте! Я устанавливал Qt6 из исходников а также Qt Creator по отдельности. Все компоненты, связанные с разработкой для Android, установлены. Кроме одного... Когда пытаюсь скомпилиров…
  • ИМ
    Nov. 22, 2024, 9:51 p.m.
    Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
  • Evgenii Legotckoi
    Oct. 31, 2024, 11:37 p.m.
    Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup