grig_p
grig_pЖел. 8, 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 Т.Ж.

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

        Пікірлер

        Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
        Кіріңіз немесе Тіркеліңіз
        OI
        • Ora Iro
        • Жел. 24, 2024, 5:38 Т.Қ.

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

        • Нәтиже:40ұпай,
        • Бағалау ұпайлары-8
        AD

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

        • Нәтиже:50ұпай,
        • Бағалау ұпайлары-4
        m
        • molni99
        • Қаз. 26, 2024, 11:37 Т.Ж.

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

        • Нәтиже:80ұпай,
        • Бағалау ұпайлары4
        Соңғы пікірлер
        ИМ
        Игорь МаксимовҚар. 22, 2024, 10:51 Т.Қ.
        Django - Оқулық 017. Теңшелген Django кіру беті Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
        Evgenii Legotckoi
        Evgenii LegotckoiҚар. 1, 2024, 12:37 Т.Ж.
        Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
        A
        ALO1ZEҚаз. 19, 2024, 6:19 Т.Қ.
        Qt Creator көмегімен fb3 файл оқу құралы Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
        ИМ
        Игорь МаксимовҚаз. 5, 2024, 5:51 Т.Қ.
        Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
        d
        dblas5Шілде 5, 2024, 9:02 Т.Қ.
        QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
        Енді форумда талқылаңыз
        n
        nklyЖел. 27, 2024, 10:41 Т.Қ.
        Нужно запретить перемещение только некоторых итемов, остальные перемещать можно. У меня есть Представление QTreeView и древовидная модель QStandardItemModel подключенная к представлению. Итемы в модели QStandardItem. В разных ветках дерева могут быть элементы с одинаковым им…
        Evgenii Legotckoi
        Evgenii LegotckoiМаусым 25, 2024, 1:11 Т.Ж.
        добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
        t
        tonypeachey1Қар. 15, 2024, 5:04 Т.Қ.
        google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
        NSProject
        NSProjectМаусым 4, 2022, 1:49 Т.Қ.
        Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…

        Бізді әлеуметтік желілерде бақылаңыз