Меню уровней на QML и C++
Здравствуйте, хочу сделать динамически формируемое меню уровней для игры (то есть меню которое в зависимости от числа уровней имеющихся например в файле XML выводит нужное количество на экран). И не много не понимаю как такое реализовать в купе с QML.
Создал специальный класс для блока уровня (пока выводит лишь треугольники) *.h:
#ifndef MODULEPARTMENUELEMENT_H #define MODULEPARTMENUELEMENT_H #include <QQuickItem> #include <QSGGeometryNode> #include <QSGFlatColorMaterial> class ModulePartMenuElement : public QQuickItem { Q_OBJECT Q_PROPERTY(bool isActive READ isActive WRITE setIsActive NOTIFY isActiveChanged) Q_PROPERTY(QColor activeColor READ activeColor WRITE setActiveColor NOTIFY activeColorChanged) Q_PROPERTY(QColor nonActiveColor READ nonActiveColor WRITE setNonActiveColor NOTIFY nonActiveColorChanged) public: ModulePartMenuElement(QQuickItem *parent = 0); bool isActive() const; QColor activeColor() const; QColor nonActiveColor() const; protected: virtual QSGNode * updatePaintNode(QSGNode *node, UpdatePaintNodeData *updatePaintNodeData) override; signals: void isActiveChanged(bool isActive); void activeColorChanged(QColor activeColor); void nonActiveColorChanged(QColor nonActiveColor); public slots: void setIsActive(const bool isActive); void setActiveColor(const QColor activeColor); void setNonActiveColor(const QColor nonActiveColor); private: //состояние элемента bool _isActive; //фон если активен QColor _activeColor; //фон если не активен QColor _nonActiveColor; }; #endif // MODULEPARTMENUELEMENT_H
*.cpp:
#include "modulepartmenuelement.h" ModulePartMenuElement::ModulePartMenuElement(QQuickItem *parent) : QQuickItem(parent) { setFlag(QQuickItem::ItemHasContents, true); } bool ModulePartMenuElement::isActive() const { return _isActive; } QColor ModulePartMenuElement::activeColor() const { return _activeColor; } QColor ModulePartMenuElement::nonActiveColor() const { return _nonActiveColor; } QSGNode *ModulePartMenuElement::updatePaintNode(QSGNode *oldNode, QQuickItem::UpdatePaintNodeData *updatePaintNodeData) { Q_UNUSED(updatePaintNodeData) QSGGeometryNode *node = static_cast<QSGGeometryNode*>(oldNode); if (!node) { node = new QSGGeometryNode(); node->setFlag(QSGNode::OwnsMaterial, true); node->setFlag(QSGNode::OwnsGeometry, true); } const QRectF rect = boundingRect(); //геометрия QSGGeometry *geometry = new QSGGeometry(QSGGeometry::defaultAttributes_Point2D(), 3); QSGGeometry::Point2D *points = geometry->vertexDataAsPoint2D(); points[0].x = rect.left(); points[0].y = rect.top(); points[1].x = rect.left() + rect.width() / 2.0; points[1].y = rect.bottom(); points[2].x = rect.right(); points[2].y = rect.top(); node->setGeometry(geometry); QSGFlatColorMaterial *material = new QSGFlatColorMaterial; if (_isActive) { material->setColor(_activeColor); } else { material->setColor(_nonActiveColor); } node->setMaterial(material); return node; } void ModulePartMenuElement::setIsActive(const bool isActive) { if (_isActive == isActive) return; _isActive = isActive; emit isActiveChanged(isActive); } void ModulePartMenuElement::setActiveColor(const QColor activeColor) { if (_activeColor == activeColor) return; _activeColor = activeColor; emit activeColorChanged(activeColor); } void ModulePartMenuElement::setNonActiveColor(const QColor nonActiveColor) { if (_nonActiveColor == nonActiveColor) return; _nonActiveColor = nonActiveColor; emit nonActiveColorChanged(nonActiveColor); }
Как я думаю, начинать надо с создания класса модели, наследуемой от абстрактной модели, в которую передается список блоков уровней (Из которого можно убрать всю взаимосвязь с QML, так как он делался под статическое размещение элементов) и модель устанавливается в стандартный вьювер в QML, но, насколько я понимаю, нужно будет еще и создавать класс делегата для отображения блока уровня в QML. (И тут возникает путаница, ведь отображение рисуется уже в созданном мной классе блока уровня или я чего-то не понимаю?) Или может быть все это можно сделать несколько проще?
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.Ол саған ұнайды ма? Әлеуметтік желілерде бөлісіңіз!
Пікірлер
Г
- Геній
- Қыр. 13, 2024, 12:46 Т.Қ.
C++ - Тест 001. Первая программа и типы данных
- Нәтиже:66ұпай,
- Бағалау ұпайлары-1
t
- torgaev_2024
- Қыр. 8, 2024, 6:20 Т.Ж.
C++ - Тест 001. Первая программа и типы данных
- Нәтиже:33ұпай,
- Бағалау ұпайлары-10
Соңғы пікірлер
Linux жүйесінде файлдарды қалай көшіруге болады Задумывались когда-нибудь о том, как мы привыкли доверять свои вещи службам грузоперевозок? Сейчас такие услуги стали неотъемлемой частью нашей жизни, особенно когда речь идет о переездах между …
ВР
Linux жүйесінде файлдарды қалай көшіруге болады Screenshot_20240802-065123.png
Влад РусоковТам. 2, 2024, 1:47 Т.Ж.
QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
Qt Linux - Сабақ 001. Linux астында Autorun Qt қолданбасы как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
АК
Qt WinAPI - Сабақ 007. Qt ішінде ICMP Ping арқылы жұмыс істеу Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
Анатолий КононенкоАқп. 5, 2024, 1:50 Т.Ж.
Енді форумда талқылаңыз
Evgenii LegotckoiМаусым 24, 2024, 3:11 Т.Қ.
при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …
BlinCTМаусым 25, 2024, 1 Т.Ж.
BlinCTМамыр 5, 2024, 5:46 Т.Ж.
Evgenii LegotckoiМамыр 2, 2024, 2:07 Т.Қ.
Может я неправильно понял задачу, но я бы делал все в QML. У ListModel есть методы append и clear. Данные из файла можно считывать в с++ классе и передавать в QML элемент.
Для меня просто qml темный лес), вот и пытаюсь разобраться. Прочитав ваш ответ в голове сформировал алгоритм: 1) при запуске main.qml вызываю метод в с++ классе по обработке файла
Здесь на сайте есть урок QML - 007. Урок про динамическое создание элементов. 1) 2) Можно добавлять элементы при первом открытии меню уровней, можно при создании. 3) Настроить вид элемента можно в делегате. Можно не использовать класс треугольника. В делегат можно добавить MouseArea, в обработчик onClicked, можно добавить изменение к-н свойства элемента делегата. Все есть в уроках на сайте.
В итоге сумел реализовать то что хотел в своем первоначальном вопросе. Но насколько моя реализация получилась грамотной не знаю. Поэтому, кому интересно, прошу посмотреть проект и указать на недочеты или плохую реализацию, если таковые имеются.