Алексей Внуков
Алексей Внуков25. Juli 2019 17:31

QML Model & C++Model

qml, qt, model

Доброго времни суток! есть модель на C++,которая берет данные из sqlite базы. в файле QML есть listview, в который должны загружаться данные, с листом очень много действий и в такой ситуации не удобно работать с С++ моделью, поскольку нужно регулярно делать апдейт модели данные в делегатах перезатираются что сильно затрудняет работу, а из самой модели данные нужны только на старте. Собственно вопрос, можно ли при старте програмы С++ модель копировать в QML модель чтоб с ней работать динамично, чтоб только в некоторых ситуациях данные записывались в базу и отобразились только при следующем старте программы.
п.с если не понятно описание задавайте вопросы

Рекомендуємо хостинг TIMEWEB
Рекомендуємо хостинг TIMEWEB
Stabiles Hosting des sozialen Netzwerks EVILEG. Wir empfehlen VDS-Hosting für Django-Projekte.

Magst du es? In sozialen Netzwerken teilen!

4
Evgenii Legotckoi
  • 26. Juli 2019 04:09

Добрый день!

А можно код того, как модель C++ регистрируется в QML и как она добавлена в ListView?

    Алексей Внуков
    • 26. Juli 2019 04:32
    • (bearbeitet)

    main.cpp

     PageModel *page_model=new PageModel();
         engine.rootContext()->setContextProperty("page_model",page_model);
    
    

    Pagemodel.h

    class PageModel : public QSqlQueryModel
    {
        Q_OBJECT
    public:
        explicit PageModel(QObject *parent = nullptr);
    
        enum Roles
        {
            idRole= Qt::UserRole+1,
            urlRole,
            nameSiteRole,
            screenRole
        };
    
        Q_INVOKABLE QVariant data(const QModelIndex & index, int role = Qt::DisplayRole) const;
        Q_INVOKABLE QVariantMap get(int idx) const;
    
    signals:
    
    protected:
        QHash<int,QByteArray> roleNames()const;
    
    public slots:
        void updateModel();
        int getId(int row);
    };
    
    

    PageModel.cpp

    PageModel::PageModel(QObject *parent) : QSqlQueryModel (parent)
    {
        this->updateModel();
    }
    
    QVariant PageModel::data(const QModelIndex &index, int role) const
    {
        int columnId = role - Qt::UserRole - 1;
        // Создаём индекс с помощью новоиспечённого ID колонки
        QModelIndex modelIndex = this->index(index.row(), columnId);
    
        /* И с помощью уже метода data() базового класса
         * вытаскиваем данные для таблицы из модели
         * */
        return QSqlQueryModel::data(modelIndex, Qt::DisplayRole);
    }
    
    QVariantMap PageModel::get(int idx) const
    {
        QVariantMap map;
        foreach(int k, roleNames().keys()) {
            map[roleNames().value(k)] = data(index(idx, 0), k);
        }
        return map;
    }
    
    QHash<int, QByteArray> PageModel::roleNames() const
    {
        QHash<int, QByteArray> roles;
        roles[idRole]="id";
        roles[urlRole]="url";
        roles[nameSiteRole]="namesite";
        roles[screenRole]="screen";
        return roles;
    }
    
    void PageModel::updateModel()
    {
        this->setQuery("select id, url, namesite, screen from openTabTable");
    }
    
    int PageModel::getId(int row)
    {
        return this->data(this->index(row, 0), idRole).toInt();
    }
    
    

    QML

    import QtQuick 2.12
    import QtQuick.Controls 2.12
    import QtWebView 1.13
    
    Item {
        id: idListView1Form
        width: 400
        height: 700
    
        Rectangle{
            id: rectangleForListView
            y: parent.height * 0.15
            height: parent.height * 0.7
            anchors.left: parent.left
            anchors.right: parent.right
    
            ListView
            {
                id: myListView1
                highlightRangeMode: ListView.StrictlyEnforceRange
                anchors.fill: parent
                orientation : ListView.Horizontal
                snapMode: ListView.NoSnap
                contentX: 100
                contentWidth: 100
                model: page_model
    
                delegate:ItemDelegate
                {
                    id: itemDelegate
                    width : myListView1.width < myListView1.height ? myListView1.width * 0.7 : myListView1.height * 1
                    height: myListView1.width < myListView1.height ? myListView1.width * 1 : myListView1.height * 0.7
                    Rectangle
                    {
    
                        WebView1
                        {
                            id:webView
                        }
    
                        anchors.fill: parent
                        anchors.leftMargin: myListView1.width * 0.06
                        Label
                        {
                            id:site_name
                            anchors.top: parent.top
                            anchors.centerIn: parent
                            text: namesite
                        }
    
                        Rectangle {
                            id: rec1
                            color: "grey"
                            anchors
                            {
                                top:site_name.bottom
                                left: parent.left
                                right:parent.right
                                bottom:parent.bottom
                            }
                            Label
                            {
                                anchors.centerIn: parent
                                text: url
                            }
                        }
    
    
                    }
                    MouseArea
                    {
                        id: ma_click
                        anchors.fill: parent
                        onClicked:
                        {
                            webView.url=url
                            stack.push(webView)
                        }
                    }
                }
            }
        }
    }
    
    
      Evgenii Legotckoi
      • 26. Juli 2019 05:10
      • Die Antwort wurde als Lösung markiert.

      Я не уверен, но думаю. что можно ещё переписать метод setData в QSqlQueryModel, в котором записывать данные в базу данных и сразу делать апдейт модели после записи данных в базу данных. Тогда проблема с перезатиранием данных в делегатах будет уже менее актуальна.

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

      Ещё можно попробовать с помощью QVariantMap протолкнуть данные до QML и там уже записать в QML модель данных.

      Ещё можно попробовать взять данные из базы данных прямо в QML

      import QtQuick 2.0
      
      Rectangle {
          color: "white"
          width: 200
          height: 100
      
          Text {
              text: "?"
              anchors.horizontalCenter: parent.horizontalCenter
              function findGreetings() {
                  var db = openDatabaseSync("QDeclarativeExampleDB", "1.0", "The Example QML SQL!", 1000000);
      
                  db.transaction(
                      function(tx) {
                          // Create the database if it doesn't already exist
                          tx.executeSql('CREATE TABLE IF NOT EXISTS Greeting(salutation TEXT, salutee TEXT)');
      
                          // Add (another) greeting row
                          tx.executeSql('INSERT INTO Greeting VALUES(?, ?)', [ 'hello', 'world' ]);
      
                          // Show all added greetings
                          var rs = tx.executeSql('SELECT * FROM Greeting');
      
                          var r = ""
                          for (var i = 0; i < rs.rows.length; i++) {
                              r += rs.rows.item(i).salutation + ", " + rs.rows.item(i).salutee + "\n"
                          }
                          text = r
                      }
                  )
              }
              Component.onCompleted: findGreetings()
          }
      }
      

      Тогда манипуляций в C++ делать не нужно будет.

        Алексей Внуков
        • 26. Juli 2019 05:59

        благодарю, интересное решение

          Kommentare

          Nur autorisierte Benutzer können Kommentare posten.
          Bitte Anmelden oder Registrieren
          Letzte Kommentare
          A
          ALO1ZE19. Oktober 2024 08:19
          Fb3-Dateileser auf Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
          ИМ
          Игорь Максимов5. Oktober 2024 07:51
          Django – Lektion 064. So schreiben Sie eine Python-Markdown-Erweiterung Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
          d
          dblas55. Juli 2024 11:02
          QML - Lektion 016. SQLite-Datenbank und das Arbeiten damit in QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
          k
          kmssr8. Februar 2024 18:43
          Qt Linux - Lektion 001. Autorun Qt-Anwendung unter Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
          Qt WinAPI - Lektion 007. Arbeiten mit ICMP-Ping in Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
          Jetzt im Forum diskutieren
          J
          JacobFib17. Oktober 2024 03:27
          добавить qlineseries в функции Пользователь может получить любые разъяснения по интересующим вопросам, касающимся обработки его персональных данных, обратившись к Оператору с помощью электронной почты https://topdecorpro.ru…
          JW
          Jhon Wick1. Oktober 2024 15:52
          Indian Food Restaurant In Columbus OH| Layla’s Kitchen Indian Restaurant If you're looking for a truly authentic https://www.laylaskitchenrestaurantohio.com/ , Layla’s Kitchen Indian Restaurant is your go-to destination. Located at 6152 Cleveland Ave, Colu…
          КГ
          Кирилл Гусарев27. September 2024 09:09
          Не запускается программа на Qt: точка входа в процедуру не найдена в библиотеке DLL Написал программу на C++ Qt в Qt Creator, сбилдил Release с помощью MinGW 64-bit, бинарнику напихал dll-ки с помощью windeployqt.exe. При попытке запуска моей сбилженной программы выдаёт три оши…
          F
          Fynjy22. Juli 2024 04:15
          при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …

          Folgen Sie uns in sozialen Netzwerken