Енді QTableView бағдарламасында делегаттардың көмегімен ұяшықтардың әрекетін теңшеу туралы шағын мақала.
Бұл мақала бір ескі мақаладағы кесте әрекетінің модификациясы болып табылады, атап айтқанда
Qt/C++ – Оқулық 039. QSqlTableModel үлгісіндегі жолды бағандағы мән арқылы толтыру жолы
. Шындығында, бұл ескі мақала бұл мақаланың мазмұнына ешқандай әсер етпейді, бірақ мысал кестесі сол сабақтан алынады.
Тапсырма ретінде тінтуір курсорын апарған кезде жолдың түсі таңдалды.
Бұл келесідей болады
Бұл функцияны іске асыру үшін біз CustomDelegate сыныбын жазамыз, ол ұяшық өңін өзгертуді басқару үшін кестенің бағандарында орнатылады. Жасуша шамдарын қайта бояу.
CustomDelegate.h
Біз QStyledItemDelegate сыныбынан мұра аламыз және paint әдісін қайта анықтаймыз.
#ifndef CUSTOMDELEGATE_H #define CUSTOMDELEGATE_H #include <QStyledItemDelegate> class CustomDelegate : public QStyledItemDelegate { using BaseClass = QStyledItemDelegate; public: CustomDelegate(QObject* parent = nullptr); virtual void paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const override; }; #endif // CUSTOMDELEGATE_H
CustomDelegate.cpp
Бояу әдісінде курсор виджеттің QTableView аймағында қозғалып жатқанда, курсор қайта сызылатын ұяшықтың қатарында бір жерде тұрғанын тексереміз, егер солай болса, ұяшықтың фонын қажетті түспен толтырамыз.
#include "customdelegate.h" #include <QPainter> #include <QAbstractItemView> CustomDelegate::CustomDelegate(QObject* parent) : BaseClass(parent) { } void CustomDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const { if (QAbstractItemView* tableView = qobject_cast<QAbstractItemView*>(this->parent())) { QModelIndex hover = tableView->indexAt(tableView->viewport()->mapFromGlobal(QCursor::pos())); if (hover.row() == index.row()) { painter->fillRect(option.rect, Qt::red); } } BaseClass::paint(painter, option, index); }
QTableView ішінде делегаттарды орнату
Делегатты кестенің әрбір бағанында орнату керек. Сондай-ақ QTableView ішінде тінтуір курсорын бақылауды қосу керек.
ui->tableView->setItemDelegateForColumn(0, new CustomDelegate(ui->tableView)); ui->tableView->setItemDelegateForColumn(1, new CustomDelegate(ui->tableView)); ui->tableView->setItemDelegateForColumn(2, new CustomDelegate(ui->tableView)); ui->tableView->setItemDelegateForColumn(3, new CustomDelegate(ui->tableView)); ui->tableView->setItemDelegateForColumn(4, new CustomDelegate(ui->tableView)); ui->tableView->setMouseTracking(true);
Қорытынды
Бағдарлама архитектурасы тұрғысынан кодтың қай бөлігі не үшін жауапты екенін түсіну өте маңызды.
Тінтуір курсорының қозғалысы тек пайдаланушы әрекеті және деректерге ешқандай қатысы жоқ.
Сондықтан деректер үлгісіне мұндай функционалдылықты қосудан аулақ болуға тырысыңыз, себебі бұл бастапқыда дұрыс емес тәсіл болады.
Деректер мен көріністі араластырғаннан гөрі, сәл көбірек код жазып, делегатты пайдаланған дұрыс.
Яғни, бұл жағдайда бағдарламаның әрекетін өзгерту көріністен немесе көріністің бөлек делегатынан төмен түспеуі керек.
День добрый.
Зачем нужна вот эта строка:
Всё, разобрался. :)
Добрый день. Удобства ради. В больших проектах удобнее вызывать BaseClass, чем постоянно смотреть, от чего конкретно наследован текущий класс. Экономит время.
Код делегата полностью скопировал в свой тестовый проект, но окрашивается не вся строка целиком, а только ячейка, на которую указывает курсор.
Во все колонки установили? Нужно на все колонки устанавливать.
Да, во все колонки.
Полностью скопировал пример - всё правильно работает. Значит, где-то у меня ошибки в тестовом проекте. Буду разбираться. Извините за беспокойство. :)
Разобрался. У вас изначально в проекте были вот эти настройки:
У меня же их не было, поэтому и выделялись отдельные ячейки.
Понятно. Я не обратил внимания на то, что там было в старом коде по настройкам строк :)
Сижу, размышляю: можно ли переписать делегата так, чтобы независимо от настроек строк выделялись строки?
По-моему, смысла в этом нет особого. Если делегат будет игнорировать настройки таблицы, то это приведёт ещё к большему непониманию, что вообще происходит, для программиста, который после вас будет смотреть на ваш проект.