ChristoF
ChristoF28 апреля 2020 г. 1:22

Отобразить список с вложенным в него еще одним списком

QAbstractListModel, ListView

Доброго времени суток, В QML есть ListView моделью которого является QAbstractListModel в который из интерфейса для обращения к данным передаётся контейнер std::vector(родительский) c полями. Помимо простых типов данных в нем содержится вложенный контейнер std::vector(дочерний). Для того чтобы "показать" в QML элементы родительского контейнера требуется определить методы QAbstractListModel такие как:
rowCount( )
data( )
roleNames( )
А также создать Enum ролей

Для простых типов, таких как string, int, double нет ничего сложного

int DailyModel::rowCount(const QModelIndex &parent) const
{
    Q_UNUSED(parent)
    return static_cast<int>(m_dailys.size());
}

QVariant DailyModel::data(const QModelIndex &index, int role) const
{
    if (!index.isValid() || index.row() > rowCount(index)) {
            return QVariant();
        }
    const ns1__dayArc *daily = m_dailys.at(index.row());

    switch (role) {
        case DailyRoles::TimeRole: {
            char buffer [80];
            struct tm * timeinfo;
            timeinfo = localtime (daily->Time);
            strftime (buffer,80,"%d/%m/%y %X", timeinfo);
            return QString::fromStdString(buffer);
            }
        case DailyRoles::VbTRole: {
            return QVariant::fromValue(*daily->VbT);
            }

        case DailyRoles::StatusRole: {        
            std::string arr=DailyModel::translator(daily->state);
            return QString::fromStdString(arr);
        }
        case DailyRoles::TRole: {
            return QVariant::fromValue(*daily->T);
            }
        case DailyRoles::K_smtRole: {
            return QVariant::fromValue(*daily->K_USCOREsmt);
            }
        default: {
             return {};
        }
    }
}

QHash<int, QByteArray> DailyModel::roleNames() const
{
    QHash<int, QByteArray> roles;
    roles[DailyRoles::TimeRole] = "time";    
    roles[DailyRoles::StatusRole] = "status";
    roles[DailyRoles::VbTRole] = "VbT";
    roles[DailyRoles::TRole] = "T";
    roles[DailyRoles::K_smtRole] = "K_smt";

    return roles;
}

Но как мне задать роль для std::vector а также вывести его в делегате?

Хочется реализовать что-то подобное

import QtQuick 2.12
import QtQuick.Window 2.12
import QtQuick.Controls 2.5

Window {
    ListModel{
        id: dataModel;

        ListElement{
            color: "skyblue";
            text: "one";
            texts:[
                ListElement{
                    text: "one_000";
                },
                ListElement{
                    text: "one_001";
                },
                ListElement{
                    text: "one_002";
                }
            ]
        }
        ListElement{
            color: "lightgreen";
            text: "two";
            texts:[
                ListElement{
                    text: "two_000";
                },
                ListElement{
                    text: "two_001";
                },
                ListElement{
                    text: "two_002";
                }
            ]
        }
        ListElement{
            color: "orchid";
            text: "three";
            texts:[
                ListElement{
                    text: "three_000";
                },
                ListElement{
                    text: "three_001";
                },
                ListElement{
                    text: "three_002";
                }
            ]
        }
    }
    visible: true
    width: 640
    height: 480

    ListView{
        id: view;
        anchors.fill: parent;
        anchors.margins: 10;
        spacing: 10;
        clip: true;

        model: dataModel;

        delegate: Rectangle{
            width: view.width;
            height: 50;

            color: model.color;
            Row{
                anchors.centerIn: parent;
                anchors.margins: 10;
                spacing: 10;

                Text{
                    renderType: Text.NativeRendering;
                    font.pointSize: 15;
                    text: model.text;
                }
                ComboBox{
                    model: texts;

                }
            }
        }
    }
}

Буду благодарен за помощь

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

Вам это нравится? Поделитесь в социальных сетях!

5
Evgenii Legotckoi
  • 28 апреля 2020 г. 2:55

Добрый день

В этом случае самым адекватным решением, которое напрашивается само по себе, это попробовать возвращать вложенную модель вместо вектора, а в делегате ListElement родительской модели уже отображать вложенную модель ListModel

    ChristoF
    • 28 апреля 2020 г. 5:10
    • (ред.)

    На данный момент придумал полумеру решения данного вопроса, сделать делегаты RadioDelegate внутри котороых вложенный Reapiter,
    при checked отправляю во вложенную модель currentIndex. В результате вложенная модель при каждом клике переопределяется и выводит список соответсвующий данному делегату. Минусом данного подхода является, то что я не могу основываясь на вложеном списке выводить в главный список какую либо информацию (Например количество строк списка), так как при изменении модели вложенного списка он меняется во всех делегатах

    Говоря "возвращать вложенную модель вместо вектора" вы имеете ввиду в методе data() установить роль возвращающую модель?

    QVariant DailyModel::data(const QModelIndex &index, int role) const
    {
        if (!index.isValid() || index.row() > rowCount(index)) {
                return QVariant();
            }
        const ns1__dayArc *daily = m_dailys.at(index.row());
    
        switch (role) {        
            case DailyRoles::VectorRole: {
                return QVariant::fromValue(/* MyModel */); //   либо другая реализация
                }
            default: {
                 return {};
            }
        }
    }
    
      Evgenii Legotckoi
      • 28 апреля 2020 г. 5:12

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

        ChristoF
        • 28 апреля 2020 г. 5:19

        Если возможно, можете подсказать "правильный синтаксис" присвоения ролям значения модели? Мой вариант меня не до конца устраивает и я все же хотел бы выводить вложенный каждый список для всех делегатов одновреммено, так как значения Reapiter-ов я хотел бы использовать в главной моделе

          Evgenii Legotckoi
          • 28 апреля 2020 г. 5:21

          Самому нужно подумать, с наскоку это не сделать, да из головы даже не знаю...

            Комментарии

            Только авторизованные пользователи могут публиковать комментарии.
            Пожалуйста, авторизуйтесь или зарегистрируйтесь
            e
            • ehot
            • 31 марта 2024 г. 21:29

            C++ - Тест 003. Условия и циклы

            • Результат:78баллов,
            • Очки рейтинга2
            B

            C++ - Тест 002. Константы

            • Результат:16баллов,
            • Очки рейтинга-10
            B

            C++ - Тест 001. Первая программа и типы данных

            • Результат:46баллов,
            • Очки рейтинга-6
            Последние комментарии
            k
            kmssr9 февраля 2024 г. 2:43
            Qt Linux - Урок 001. Автозапуск Qt приложения под Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
            АК
            Анатолий Кононенко5 февраля 2024 г. 9:50
            Qt WinAPI - Урок 007. Работаем с ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
            EVA
            EVA25 декабря 2023 г. 18:30
            Boost - статическая линковка в CMake проекте под Windows Ошибка LNK1104 часто возникает, когда компоновщик не может найти или открыть файл библиотеки. В вашем случае, это файл libboost_locale-vc142-mt-gd-x64-1_74.lib из библиотеки Boost для C+…
            J
            JonnyJo25 декабря 2023 г. 16:38
            Boost - статическая линковка в CMake проекте под Windows Сделал всё по-как у вас, но выдаёт ошибку [build] LINK : fatal error LNK1104: не удается открыть файл "libboost_locale-vc142-mt-gd-x64-1_74.lib" Хоть убей, не могу понять в чём дел…
            G
            Gvozdik19 декабря 2023 г. 5:01
            Qt/C++ - Урок 056. Подключение библиотеки Boost в Qt для компиляторов MinGW и MSVC Для решения твой проблемы добавь в файл .pro строчку "LIBS += -lws2_32" она решит проблему , лично мне помогло.
            Сейчас обсуждают на форуме
            a
            a_vlasov14 апреля 2024 г. 13:41
            Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Евгений, добрый день! Такой вопрос. Верно ли следующее утверждение: Любое Android-приложение, написанное на Java/Kotlin чисто теоретически (пусть и с большими трудностями) можно написать и на C+…
            Павел Дорофеев
            Павел Дорофеев14 апреля 2024 г. 9:35
            QTableWidget с 2 заголовками Вот тут есть кастомный QTableView с многорядностью проект поддерживается, обращайтесь
            Mm
            Mind mingles12 апреля 2024 г. 7:42
            ASO Service Forum: Enhancing App Visibility and Reach Welcome to the ASO Service Forum, your ultimate destination for insights, discussions, and strategies revolving around App Store Optimization. ASO (App Store Optimization) is paramoun…
            f
            fastrex4 апреля 2024 г. 11:47
            Вернуть старое поведение QComboBox, не менять индекс при resetModel Добрый день! У нас много проектов в которых используется QComboBox, в версии 5.5.1, когда модель испускает сигнал resetModel, currentIndex не менялся. В версии 5.15 при resetModel происходит try…
            P
            Pisych27 февраля 2023 г. 12:04
            Как получить в массив значения из связанной модели? Спасибо, разобрался:))

            Следите за нами в социальных сетях