grig_p
grig_p8 декабря 2017 г. 1: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 г. 3:10
  • (ред.)

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

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

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

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

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

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

        Комментарии

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

        C++ - Тест 004. Указатели, Массивы и Циклы

        • Результат:50баллов,
        • Очки рейтинга-4
        m
        • molni99
        • 26 октября 2024 г. 11:37

        C++ - Тест 004. Указатели, Массивы и Циклы

        • Результат:80баллов,
        • Очки рейтинга4
        m
        • molni99
        • 26 октября 2024 г. 11:29

        C++ - Тест 004. Указатели, Массивы и Циклы

        • Результат:20баллов,
        • Очки рейтинга-10
        Последние комментарии
        ИМ
        Игорь Максимов22 ноября 2024 г. 22:51
        Django - Урок 017. Кастомизированная страница авторизации на Django Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
        Evgenii Legotckoi
        Evgenii Legotckoi1 ноября 2024 г. 0:37
        Django - Урок 064. Как написать расширение для Python Markdown Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
        A
        ALO1ZE19 октября 2024 г. 18:19
        Читалка fb3-файлов на Qt Creator Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
        ИМ
        Игорь Максимов5 октября 2024 г. 17:51
        Django - Урок 064. Как написать расширение для Python Markdown Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
        d
        dblas55 июля 2024 г. 21:02
        QML - Урок 016. База данных SQLite и работа с ней в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
        Сейчас обсуждают на форуме
        Evgenii Legotckoi
        Evgenii Legotckoi25 июня 2024 г. 1:11
        добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
        t
        tonypeachey115 ноября 2024 г. 17:04
        google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
        NSProject
        NSProject4 июня 2022 г. 13:49
        Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…
        9
        9Anonim25 октября 2024 г. 19:10
        Машина тьюринга // Начальное состояние 0 0, ,<,1 // Переход в состояние 1 при пустом символе 0,0,>,0 // Остаемся в состоянии 0, двигаясь вправо при встрече 0 0,1,>…

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