grig_p
grig_p8 декабря 2017 г. 12:31

Модели QAbstractItemModel и сортировка

Здравствуйте!
Возник вопрос вот на какую тему.
Делаю я табличную модель. Наследовался от QAbstractItemModel.
Модель выглядит следующим образом:

// Строка в модели
struct InfoJournalRec
{
int id;
QDateTime created;
QString sourcer;
QString module;
QString msgType;
int priority;
QString detail;
explicit InfoJournalRec() {}
explicit InfoJournalRec(std::tuple<int, QDateTime, QString, QString, QString, int, QString> params)
: id(std::get<0>(params))
, created(std::get<1>(params))
, sourcer(std::get<2>(params))
, module(std::get<3>(params))
, msgType(std::get<4>(params))
, priority(std::get<5>(params))
, detail(std::get<6>(params))
{}
};
Q_DECLARE_METATYPE(InfoJournalRec)

// Список строк typedef QList<InfoJournalRec> QListInfoJournalRec;
typedef QSharedPointer<QListInfoJournalRec> InfoJournalRecListPtr;
Q_DECLARE_METATYPE(InfoJournalRecListPtr) // Сама модель class TableJournalModel : public QAbstractItemModel { Q_OBJECT public: explicit TableJournalModel(QObject *parent = Q_NULLPTR); int rowCount(const QModelIndex &parent = QModelIndex()) const; QVariant data(const QModelIndex &index, int role) const; QVariant headerData(int section, Qt::Orientation orientation, int role = Qt::DisplayRole) const; Qt::ItemFlags flags(const QModelIndex &index) const; QModelIndex index(int row, int column, const QModelIndex &parent = QModelIndex()) const; QModelIndex parent(const QModelIndex &child) const; int columnCount(const QModelIndex &parent = QModelIndex()) const; public slots: void load(); private: InfoJournalRecListPtr m_records = Q_NULLPTR; };

В ней имеется защищенный указатель на список список QList с данными и перекрыт ряд методов, обеспечивающих доступ к ним.
Реализация метода доступа к данным выглядит следующим образом:
QVariant TableJournalModel::data(const QModelIndex &index, int role) const
{
    if (!index.isValid() || (m_records == Q_NULLPTR))
        return QVariant();

    if ((index.row() < 0) || (index.row() >= m_records->count()))
        return QVariant();

    if (role == Qt::DisplayRole || role == Qt::EditRole)
    {
        int n = 0;
        auto di = m_records->begin();
        while (di != m_records->end())
        {
            InfoJournalRec ijr = *di;
            if (n == index.row())
            {
                if (index.column() == 0)
                    return ijr.id;
                else
                if (index.column() == 1)
                    return ijr.created;
                else
                if (index.column() == 2)
                    return ijr.sourcer;
                else
                if (index.column() == 3)
                    return ijr.module;
                else
                if (index.column() == 4)
                    return ijr.msgType;
                else
                if (index.column() == 5)
                    return priorityTitles.value(ijr.priority);
                else
                if (index.column() == 6)
                    return ijr.detail;
            }
            ++n;
            ++di;
        }
        // Если мы не вернули ничего, то вернем пустое значение
        return QVariant();
    }
    else
        return QVariant();
}
Подключение модели осуществляется с помощью QSortFilterProxyModel следующим образом:
    m_model = new TableJournalModel();
m_proxyModel->setSourceModel(m_model); ui->TableView->setModel(m_proxyModel);
Таблица должна:
1. Сортироваться по любому столбцу.
2. Последний столбец detail не должен выводиться в таблице, ибо большой по размеру, а его значение должно выводиться в отдельном поле при выборе строки таблицы.
TableView допускает сортировку. Кроме того, реализована обработка клика на столбце:
    connect(ui->TableView->horizontalHeader(), &QHeaderView::sectionClicked, this, [=](int logicalIndex)// Выделение и активация строки модели
void TableTailPbxLog::onItemActivated(QModelIndex index)
{
    QModelIndex idxDetail = m_model->index(index.row(), 6);
    QString st = m_model->data(idxDetail, Qt::DisplayRole).toString();
    ui->teDetail->setText(st);
}
    {
        ui->TableView->sortByColumn(logicalIndex);
    });
Сортировка работает превосходно.

Выделение и отображение данных из столбца также работает:
// Выделение и активация строки модели
void TableTailPbxLog::onItemActivated(QModelIndex index)
{
    QModelIndex idxDetail = model->index(index.row(), 6);
    QString st = model->data(idxDetail, Qt::DisplayRole).toString();
    ui->teDetail->setText(st);
}
Но, если отсортировать таблицу не в последовательности заполнения, то данные берутся не из тех строк, которые выделены, а из строки, номер которой выбран в отсортированной таблице.
То есть, в процедуре onItemActivated параметр index заполнен данными по отсортированной модели
index.data()->toString(... выводит правильное отображение, а
index.row() содержит номер строки в отсортированной модели, и я не знаю, как получить индексы записей столбцов этой строки, ибо мне нужны индексы строк в исходной модели.

Вопрос, как правильно получить индекс строки в исходной модели?
Не хотелось бы переходить к QStandarditemModel, ибо там много накладных расходов.
Заранее благодарен за ответ.


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

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

3
Evgenii Legotckoi
  • 8 декабря 2017 г. 14:10
  • (ред.)

День добрый!
Вы же используете модель QSortFilterProxyModel для сортировки. А она имеет метод mapToSource, которые будет возвращать индекс исходной модели данных, если в него передать индекс прокси модели. Фактическив в TableView  у вас находится прокси модель, которая мапится на исходную. Ваша строка не соответствует ожидаемому результату потому, что Вы имеете индекс прокси модели. Вам нужно всего лишь преобразовать его в индес исходной модели.

const QModelIndex& sourceIndex = m_proxyModel->mapToSource(proxyIndex);

И если интересно, могу высказать мнение по поводу качества кода в целом. У вас есть ряд огрехов, которые имеет смысл исправить. Это касается код стайла, использования устаревшего подхода, и несколько избыточного и менее читаемого использования некоторых синтаксических конструкций языка.
    grig_p
    • 8 декабря 2017 г. 14:35

    Спасибо!
    Все получилось.
    Неужели все так просто. Вот уж точно "Век живи - век учись"

      Evgenii Legotckoi
      • 8 декабря 2017 г. 14:36

      я бы по-другому сказал )) "Документация - наш друг"

        Комментарии

        Только авторизованные пользователи могут публиковать комментарии.
        Пожалуйста, авторизуйтесь или зарегистрируйтесь
        Ua

        Qt - Тест 001. Сигналы и слоты

        • Результат:84баллов,
        • Очки рейтинга4
        Ua

        Qt - Тест 001. Сигналы и слоты

        • Результат:42баллов,
        • Очки рейтинга-8
        ОК

        Qt - Тест 001. Сигналы и слоты

        • Результат:47баллов,
        • Очки рейтинга-6
        Последние комментарии
        ИМ
        Игорь Максимов22 ноября 2024 г. 21:51
        Django - Урок 017. Кастомизированная страница авторизации на Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
        Evgenii Legotckoi
        Evgenii Legotckoi31 октября 2024 г. 23:37
        Django - Урок 064. Как написать расширение для Python Markdown Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
        A
        ALO1ZE19 октября 2024 г. 17:19
        Читалка fb3-файлов на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
        ИМ
        Игорь Максимов5 октября 2024 г. 16:51
        Django - Урок 064. Как написать расширение для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
        d
        dblas55 июля 2024 г. 20:02
        QML - Урок 016. База данных SQLite и работа с ней в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
        Сейчас обсуждают на форуме
        f
        firstlunoxod15 февраля 2025 г. 13:46
        Рисование на QGraphicsScene при зажатой кнопке мыши Подскажите, пожалуйста! Как данный класс можно дополнить, чтобы созданные объекты можно было перемещать мышкой по сцене?
        Дмитрий
        Дмитрий3 февраля 2025 г. 16:24
        Создание deb-пакета. Как создать ярлык на рабочем столе после установки собственного deb-пакета? Всем привет. Сделал свой deb-пакет с программой. Всё устанавливается и работает. Ставлю по пути /usr/bin/my_application. Как для пользователя при установке пакета сразу создать ярлык на раб…
        NW
        Nayo Wai30 января 2025 г. 19:22
        не запускается компьютер!!! Не запускается компьютер (точнее работает блок , но сам монитор вообще жесть)В общем я ничего с интернета не скачивала в последнее время. На компе никаких левых пр…
        n
        nkly3 января 2025 г. 12:52
        Нужно запретить перемещение только некоторых итемов, остальные перемещать можно. Вопрос решен. Узнать QModelIndex элемента на который мы перетаскиваем другой элемент, можно с помощью функции indexAt(event->position().toPoint()) представления QTreeViev вызываемой в переопр…
        M
        Marsel17 августа 2023 г. 0:26
        OAuth2.0 через VK, получение email Спасибо большое за помощь и простите за то что отнял время своей невнимательностью.

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