Показаны различия между двумя версиями страницы.
Предыдущая версия справа и слева Предыдущая версия Следующая версия | Предыдущая версия | ||
develop:qt:quick3 [2021/08/31 14:26] admin |
develop:qt:quick3 [2021/09/04 10:55] (текущий) admin |
||
---|---|---|---|
Строка 45: | Строка 45: | ||
</ | </ | ||
+ | |||
=== XML-модель === | === XML-модель === | ||
**XmlListModel**- тоже модель списка и используется для XML-данных. Задействует для заполнения данными опросы **XPath** и присваивает данные свойствам.\\ | **XmlListModel**- тоже модель списка и используется для XML-данных. Задействует для заполнения данными опросы **XPath** и присваивает данные свойствам.\\ | ||
Строка 91: | Строка 91: | ||
</ | </ | ||
+ | |||
=== JSON-модель === | === JSON-модель === | ||
---- | ---- | ||
Строка 125: | Строка 125: | ||
</ | </ | ||
+ | |||
==== Представление данных ==== | ==== Представление данных ==== | ||
=== ListView === | === ListView === | ||
Строка 229: | Строка 229: | ||
</ | </ | ||
+ | |||
=== GridView === | === GridView === | ||
---- | ---- | ||
Строка 322: | Строка 322: | ||
</ | </ | ||
+ | |||
=== PathView === | === PathView === | ||
---- | ---- | ||
Строка 411: | Строка 411: | ||
</ | </ | ||
+ | |||
==== Визуальная модель данных ==== | ==== Визуальная модель данных ==== | ||
---- | ---- | ||
Строка 501: | Строка 501: | ||
</ | </ | ||
+ | |||
===== Qt Quick и C++ ===== | ===== Qt Quick и C++ ===== | ||
==== Использование языка QML в C++ ==== | ==== Использование языка QML в C++ ==== | ||
Строка 576: | Строка 576: | ||
</ | </ | ||
+ | |||
=== Взаимодействие из C++ со св-вами элементов QML === | === Взаимодействие из C++ со св-вами элементов QML === | ||
---- | ---- | ||
Строка 619: | Строка 619: | ||
</ | </ | ||
+ | |||
=== Соединение QML-сигналов со слотами C++ === | === Соединение QML-сигналов со слотами C++ === | ||
---- | ---- | ||
В прошлых примерах мы использовали QML внутри виджетов, | В прошлых примерах мы использовали QML внутри виджетов, | ||
- | На этот раз | + | На этот раз вся |
+ | В проект добавлены только 2 модуля: | ||
< | < | ||
- | < | + | < |
+ | **Форма приложения** | ||
<code QML> | <code QML> | ||
+ | import QtQuick 2.8; import QtQuick.Controls 2.2; import QtQuick.Window 2.2 | ||
+ | ApplicationWindow | ||
+ | { | ||
+ | visible: true | ||
+ | width: 150; height: 150; | ||
+ | Column | ||
+ | { | ||
+ | anchors.centerIn: | ||
+ | spacing: 5 | ||
+ | Button | ||
+ | { | ||
+ | signal infoClick(string str) | ||
+ | objectName: " | ||
+ | text: " | ||
+ | onClicked: infoClick(" | ||
+ | } | ||
+ | Button | ||
+ | { | ||
+ | signal quitClick() | ||
+ | objectName: " | ||
+ | text: " | ||
+ | onClicked: quitClick() | ||
+ | } | ||
+ | } | ||
+ | } | ||
</ | </ | ||
- | </ | ||
+ | **Класс со слотами** | ||
+ | <code cpp-qt> | ||
+ | #ifndef MYWIDGET_H | ||
+ | #define MYWIDGET_H | ||
+ | #include < | ||
+ | class CppConnect : public QObject | ||
+ | { | ||
+ | Q_OBJECT | ||
+ | public: | ||
+ | CppConnect(QObject *parent = nullptr) | ||
+ | { | ||
+ | } | ||
+ | |||
+ | public slots: | ||
+ | void slotQuit() | ||
+ | { | ||
+ | qApp-> | ||
+ | } | ||
+ | |||
+ | void slotInfo(const QString str) | ||
+ | { | ||
+ | qDebug() << str; | ||
+ | } | ||
+ | |||
+ | }; | ||
+ | #endif // MYWIDGET_H | ||
+ | </ | ||
+ | |||
+ | **Соединение сигналов со слотами, | ||
+ | <code cpp-qt> | ||
+ | #include < | ||
+ | #include < | ||
+ | #include < | ||
+ | #include " | ||
+ | |||
+ | int main(int argc, char** argv) | ||
+ | { | ||
+ | QGuiApplication app(argc, argv); | ||
+ | QQmlApplicationEngine eng; | ||
+ | QQmlComponent comp(& | ||
+ | CppConnect cc; | ||
+ | QObject *pobj= comp.create(); | ||
+ | |||
+ | QObject *pcmdQuitButton= pobj-> | ||
+ | if(pcmdQuitButton) | ||
+ | { | ||
+ | QObject:: | ||
+ | } | ||
+ | |||
+ | QObject *pcmdInfoButton= pobj-> | ||
+ | if(pcmdInfoButton) | ||
+ | { | ||
+ | QObject:: | ||
+ | } | ||
+ | return app.exec(); | ||
+ | } | ||
+ | </ | ||
+ | </ | ||
+ | |||
+ | |||
==== Использование компонентов C++ в QML ==== | ==== Использование компонентов C++ в QML ==== | ||
---- | ---- | ||
+ | В QML реализована возможность расширения при помощи C++, благодаря этому можно QML расширять новыми элементами из C++.\\ | ||
+ | Существующие технологии импортируются директивой **import**, например " | ||
+ | Работа происходит через класс контекста **QQmlContext**, | ||
+ | Используя этот указатель вы можете ввести в дерево контекста новые объекты классов, | ||
+ | После этого, свойства класса **QObject** станут свойствами QML, а слоты и методы (декларированные с Q_INVOKABLE) смогут вызываться из вашего QML-элемента.\\ | ||
+ | Класс **QQuickWidget** так же содержит объект класса **QQmlEngine** (вызов engine()), который предоставляет среду для исполнения QML.\\ | ||
+ | |||
+ | |||
+ | === Экспорт объектов и виджетов из C++ в QML === | ||
+ | ---- | ||
+ | Пример экспорта, | ||
< | < | ||
- | < | + | < |
+ | **Файл класса myWidget.cpp** | ||
+ | <code cpp-qt> | ||
+ | #include " | ||
+ | |||
+ | MyWidget:: | ||
+ | { | ||
+ | QQuickWidget *pv= new QQuickWidget(this); | ||
+ | pv-> | ||
+ | |||
+ | QVBoxLayout *pvbx= new QVBoxLayout(this); | ||
+ | pvbx-> | ||
+ | setLayout(pvbx); | ||
+ | |||
+ | QQmlContext *pcon= pv-> | ||
+ | QStringList lst; | ||
+ | for(int i= 0; i < 100; i++) | ||
+ | lst << " | ||
+ | QStringListModel *pmodel= new QStringListModel(this); | ||
+ | pmodel-> | ||
+ | |||
+ | pcon-> | ||
+ | pcon-> | ||
+ | pcon-> | ||
+ | pcon-> | ||
+ | } | ||
+ | |||
+ | void MyWidget:: | ||
+ | { | ||
+ | QMessageBox:: | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | |||
+ | **Форма QML** | ||
<code QML> | <code QML> | ||
+ | import QtQuick 2.8 | ||
+ | Rectangle | ||
+ | { | ||
+ | width: 200; height: 200; | ||
+ | color: myColor | ||
+ | Text | ||
+ | { | ||
+ | anchors.centerIn: | ||
+ | text: myText | ||
+ | } | ||
+ | ListView | ||
+ | { | ||
+ | anchors.fill: | ||
+ | model: myModel | ||
+ | delegate: Text { text: model.display } | ||
+ | } | ||
+ | MouseArea | ||
+ | { | ||
+ | anchors.fill: | ||
+ | onPressed: | ||
+ | { | ||
+ | myWidget.setWindowTitle(" | ||
+ | myWidget.slotDisplayDialog(); | ||
+ | } | ||
+ | } | ||
+ | } | ||
</ | </ | ||
</ | </ | ||
+ | |||
+ | === Использование зарегистр объектов C++ в QML === | ||
+ | ---- | ||
+ | Как я понял, это передача в QML только **конкретных свойств и методов**, | ||
+ | Прежде всего производим регистрацию нашего класса, | ||
+ | < | ||
+ | < | ||
+ | **Calculate.h** | ||
+ | <code cpp-qt> | ||
+ | #ifndef CALCULATION_H | ||
+ | #define CALCULATION_H | ||
+ | #include < | ||
- | === Экспорт объектов и виджетов из C++ в QML === | + | class Calculation : public QObject |
- | ---- | + | { |
+ | Q_OBJECT | ||
+ | private: | ||
+ | Q_PROPERTY(qulonglong input WRITE setInputValue | ||
+ | READ inputValue | ||
+ | NOTIFY inputValueChanged) // Уведомление об изменении | ||
+ | Q_PROPERTY(qulonglong result READ resultValue | ||
+ | | ||
+ | qulonglong m_nInput; | ||
+ | | ||
+ | public: | ||
+ | explicit Calculation(QObject *parent = nullptr); | ||
- | < | + | Q_INVOKABLE qulonglong factorial(const qulonglong &n); |
- | < | + | |
- | <code QML> | + | void setInputValue(const qulonglong& |
+ | qulonglong resultValue() const; | ||
+ | signals: | ||
+ | void inputValueChanged(qulonglong); | ||
+ | void resultValueChanged(qulonglong); | ||
+ | }; | ||
+ | |||
+ | #endif // CALCULATION_H | ||
</ | </ | ||
- | </ | ||
+ | **Calculate.cpp** | ||
+ | <code cpp-qt> | ||
+ | #include " | ||
- | === Использов зарегистр объектов C++ в QML === | + | Calculation:: |
- | ---- | + | { |
+ | } | ||
+ | qulonglong Calculation:: | ||
+ | { | ||
+ | return n ? (n * factorial(n-1)) : 1; | ||
+ | } | ||
- | <details> | + | qulonglong Calculation:: |
- | <summary>:!: Пример: </ | + | { |
+ | return m_nInput; | ||
+ | } | ||
+ | |||
+ | qulonglong Calculation:: | ||
+ | { | ||
+ | return m_nResult; | ||
+ | } | ||
+ | |||
+ | void Calculation:: | ||
+ | { | ||
+ | m_nInput= n; | ||
+ | m_nResult= factorial(m_nInput); | ||
+ | emit inputValueChanged(m_nInput); | ||
+ | emit resultValueChanged(m_nResult); | ||
+ | } | ||
+ | </code> | ||
+ | |||
+ | **main.cpp** | ||
+ | <code cpp-qt> | ||
+ | #include < | ||
+ | #include < | ||
+ | #include " | ||
+ | |||
+ | int main(int argc, char** argv) | ||
+ | { | ||
+ | QGuiApplication app(argc, argv); | ||
+ | qmlRegisterType< | ||
+ | QQmlApplicationEngine engine; | ||
+ | engine.load(QUrl(" | ||
+ | |||
+ | return app.exec(); | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | |||
+ | **Два варианта** реализации, | ||
<code QML> | <code QML> | ||
+ | import QtQuick 2.8; import QtQuick.Controls 2.2; import QtQuick.Layouts 1.3; | ||
+ | import my.class.Calc 1.0 | ||
+ | ApplicationWindow | ||
+ | { | ||
+ | title: " | ||
+ | visible: true | ||
+ | Calculation | ||
+ | { | ||
+ | id: calc | ||
+ | } | ||
+ | ColumnLayout | ||
+ | { | ||
+ | anchors.fill: | ||
+ | RowLayout // Первый подход, | ||
+ | { | ||
+ | SpinBox | ||
+ | { | ||
+ | id: sbx | ||
+ | value: 0 | ||
+ | } | ||
+ | Text | ||
+ | { | ||
+ | text: " | ||
+ | } | ||
+ | } | ||
+ | RowLayout // Второй подход, | ||
+ | { | ||
+ | SpinBox | ||
+ | { | ||
+ | value: 0 | ||
+ | onValueChanged: | ||
+ | } | ||
+ | Text | ||
+ | { | ||
+ | text: " | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | } | ||
</ | </ | ||
- | </ | ||
+ | **Альтернативный подход** с использованием сигналов | ||
+ | <code QML> | ||
+ | import QtQuick 2.8; import QtQuick.Controls 2.2; import QtQuick.Layouts 1.3; | ||
+ | import my.class.Calc 1.0 | ||
+ | ApplicationWindow | ||
+ | { | ||
+ | title: " | ||
+ | visible: true | ||
+ | Calculatoin | ||
+ | { | ||
+ | input: sbx.value | ||
+ | onResultValueChanged: | ||
+ | } | ||
+ | RowLayout | ||
+ | { | ||
+ | SpinBox | ||
+ | { | ||
+ | id: sbx | ||
+ | value: 0 | ||
+ | } | ||
+ | Text | ||
+ | { | ||
+ | id: txt | ||
+ | } | ||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | </ | ||
+ | |||
+ | |||
=== Реализация визуальных элементов QML на C++ === | === Реализация визуальных элементов QML на C++ === | ||
---- | ---- | ||
+ | Все базовые **элементы QML** (такие как Rectangle или Text), **реализованы на C++** (QQuickRectangle, | ||
< | < | ||
- | < | + | < |
- | < | + | **Ellipse.h** |
+ | < | ||
+ | #ifndef ELLIPSE_H | ||
+ | #define ELLIPSE_H | ||
+ | #include < | ||
+ | |||
+ | class QPainter; | ||
+ | |||
+ | class Ellipse: public QQuickPaintedItem | ||
+ | { | ||
+ | Q_OBJECT | ||
+ | |||
+ | private: | ||
+ | Q_PROPERTY(QColor color WRITE setColorValue | ||
+ | READ colorValue) | ||
+ | QColor m_color; | ||
+ | |||
+ | public: | ||
+ | Ellipse(QQuickItem *parent= nullptr); | ||
+ | // Отвечает рисование элемента, | ||
+ | void paint(QPainter *ppainter); | ||
+ | QColor colorValue() const; | ||
+ | void setColorValue(const QColor& | ||
+ | }; | ||
+ | |||
+ | #endif // ELLIPSE_H | ||
</ | </ | ||
- | </ | ||
+ | **Ellipse.hpp** | ||
+ | <code cpp-qt> | ||
+ | #include " | ||
+ | #include < | ||
+ | Ellipse:: | ||
+ | { | ||
+ | } | ||
- | === Класс QQuickImageProvider ==== | + | void Ellipse:: |
- | ---- | + | { |
+ | ppainter-> | ||
+ | ppainter-> | ||
+ | ppainter-> | ||
+ | ppainter-> | ||
+ | } | ||
+ | QColor Ellipse:: | ||
+ | { | ||
+ | return this-> | ||
+ | } | ||
- | <details> | + | void Ellipse:: |
- | <summary>:!: Пример: | + | { |
+ | this-> | ||
+ | } | ||
+ | </code> | ||
+ | |||
+ | |||
+ | **main.hpp** | ||
+ | <code cpp-qt> | ||
+ | #include < | ||
+ | #include < | ||
+ | #include " | ||
+ | |||
+ | int main(int argc, char** argv) | ||
+ | { | ||
+ | QGuiApplication app(argc, argv); | ||
+ | qmlRegisterType< | ||
+ | QQmlApplicationEngine engine; | ||
+ | engine.load(QUrl(" | ||
+ | |||
+ | return app.exec(); | ||
+ | } | ||
+ | </code> | ||
+ | |||
+ | |||
+ | **main.qml** | ||
<code QML> | <code QML> | ||
+ | import QtQuick 2.8; import QtQuick.Controls 2.2 | ||
+ | import my.class.Ellipse 1.0 | ||
+ | ApplicationWindow | ||
+ | { | ||
+ | title: "Paint Element"; | ||
+ | visible: true | ||
+ | Ellipse | ||
+ | { | ||
+ | anchors.fill: | ||
+ | color: " | ||
+ | } | ||
+ | } | ||
</ | </ | ||
</ | </ | ||
+ | |||
+ | === Класс QQuickImageProvider ==== | ||
+ | ---- | ||
+ | Этот класс является неким объектом, | ||
+ | Т.к. ImageProvider возвращает (методом request) изображение в как таковом виде (тип данных image/ | ||
< | < | ||
- | < | + | < |
+ | |||
+ | **ImageProvider.h** | ||
+ | <code cpp-qt> | ||
+ | #ifndef IMAGEPROVIDER_H | ||
+ | #define IMAGEPROVIDER_H | ||
+ | #include < | ||
+ | #include < | ||
+ | #include < | ||
+ | |||
+ | class ImageProvider: | ||
+ | { | ||
+ | private: | ||
+ | QImage brightness(const QImage & | ||
+ | |||
+ | public: | ||
+ | ImageProvider(); | ||
+ | QImage requestImage(const QString&, | ||
+ | }; | ||
+ | |||
+ | #endif // IMAGEPROVIDER_H | ||
+ | </ | ||
+ | |||
+ | |||
+ | **ImageProvider.hpp** | ||
+ | <code cpp-qt> | ||
+ | #include " | ||
+ | |||
+ | ImageProvider:: | ||
+ | { | ||
+ | } | ||
+ | |||
+ | QImage ImageProvider:: | ||
+ | { | ||
+ | QImage imgTemp= imgOrig; | ||
+ | qint32 nHeigt= imgTemp.height(); | ||
+ | qint32 nWidth= imgTemp.width(); | ||
+ | |||
+ | for(qint32 y= 0; y < nHeigt; ++y) | ||
+ | { | ||
+ | QRgb *tempLine= reinterpret_cast< | ||
+ | for(qint32 x= 0; x < nWidth; ++x) | ||
+ | { | ||
+ | int r= qRed(*tempLine) + n; | ||
+ | int g= qGreen(*tempLine) + n; | ||
+ | int b= qBlue(*tempLine) + n; | ||
+ | int a= qAlpha(*tempLine); | ||
+ | |||
+ | *tempLine++ = qRgba(r > 255 ? 255 : r < 0 ? 0:r, | ||
+ | g > 255 ? 255 : g < 0 ? 0:g, | ||
+ | b > 255 ? 255 : b < 0 ? 0:b, | ||
+ | a); | ||
+ | } | ||
+ | } | ||
+ | return imgTemp; | ||
+ | } | ||
+ | |||
+ | QImage ImageProvider:: | ||
+ | { | ||
+ | QStringList lst= strId.split(";" | ||
+ | bool bOk= false; | ||
+ | int nBrightness= lst.last().toInt(& | ||
+ | QImage img= brightness(QImage(":/" | ||
+ | |||
+ | if(ps) | ||
+ | *ps= img.size(); | ||
+ | |||
+ | return img; | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | |||
+ | **main.hpp** | ||
+ | <code cpp-qt> | ||
+ | #include < | ||
+ | #include < | ||
+ | #include " | ||
+ | |||
+ | int main(int argc, char** argv) | ||
+ | { | ||
+ | QGuiApplication app(argc, argv); | ||
+ | QQmlApplicationEngine engine; | ||
+ | engine.addImageProvider(QLatin1String(" | ||
+ | engine.load(QUrl(" | ||
+ | |||
+ | return app.exec(); | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | |||
+ | **main.qml** | ||
<code QML> | <code QML> | ||
+ | import QtQuick 2.8; import QtQuick.Controls 2.2 | ||
+ | ApplicationWindow | ||
+ | { | ||
+ | title: " | ||
+ | visible: true | ||
+ | Column | ||
+ | { | ||
+ | id: controls | ||
+ | Image | ||
+ | { | ||
+ | id: img | ||
+ | source: " | ||
+ | } | ||
+ | Slider | ||
+ | { | ||
+ | id: sld | ||
+ | width: img.width | ||
+ | value: 0.75 | ||
+ | stepSize: 0.01 | ||
+ | property int brightnessValue: | ||
+ | } | ||
+ | Text | ||
+ | { | ||
+ | width: img.width | ||
+ | text: "< | ||
+ | } | ||
+ | } | ||
+ | } | ||
</ | </ | ||
</ | </ | ||
- |