Evgenii Legotckoi
Маусым 4, 2020, 3:30 Т.Ж.

Qt/C++ - 091-сабақ. Кестедегі жолды бөлектеуді басқаратын теңшелетін делегат қалай жазылады

Енді 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);

Қорытынды

Бағдарлама архитектурасы тұрғысынан кодтың қай бөлігі не үшін жауапты екенін түсіну өте маңызды.
Тінтуір курсорының қозғалысы тек пайдаланушы әрекеті және деректерге ешқандай қатысы жоқ.
Сондықтан деректер үлгісіне мұндай функционалдылықты қосудан аулақ болуға тырысыңыз, себебі бұл бастапқыда дұрыс емес тәсіл болады.
Деректер мен көріністі араластырғаннан гөрі, сәл көбірек код жазып, делегатты пайдаланған дұрыс.
Яғни, бұл жағдайда бағдарламаның әрекетін өзгерту көріністен немесе көріністің бөлек делегатынан төмен түспеуі керек.

Осы тақырып бойынша ұсынылатын мақалалар

Мақала бойынша сұралады0сұрақтар(лар)

4

Ол саған ұнайды ма? Әлеуметтік желілерде бөлісіңіз!

IscanderChe
  • Маусым 4, 2020, 4:30 Т.Қ.
  • (өңделген)

День добрый.
Зачем нужна вот эта строка:

using BaseClass = QStyledItemDelegate;

Всё, разобрался. :)

Evgenii Legotckoi
  • Маусым 4, 2020, 5:12 Т.Қ.
  • (өңделген)

Добрый день. Удобства ради. В больших проектах удобнее вызывать BaseClass, чем постоянно смотреть, от чего конкретно наследован текущий класс. Экономит время.

IscanderChe
  • Маусым 4, 2020, 5:59 Т.Қ.

Код делегата полностью скопировал в свой тестовый проект, но окрашивается не вся строка целиком, а только ячейка, на которую указывает курсор.

Evgenii Legotckoi
  • Маусым 4, 2020, 6:08 Т.Қ.

Во все колонки установили? Нужно на все колонки устанавливать.

IscanderChe
  • Маусым 4, 2020, 6:10 Т.Қ.

Да, во все колонки.

IscanderChe
  • Маусым 4, 2020, 10:10 Т.Қ.

Полностью скопировал пример - всё правильно работает. Значит, где-то у меня ошибки в тестовом проекте. Буду разбираться. Извините за беспокойство. :)

IscanderChe
  • Маусым 5, 2020, 12:27 Т.Қ.

Разобрался. У вас изначально в проекте были вот эти настройки:

ui->tableView->setSelectionBehavior(QAbstractItemView::SelectRows);
ui->tableView->setSelectionMode(QAbstractItemView::SingleSelection);

У меня же их не было, поэтому и выделялись отдельные ячейки.

Evgenii Legotckoi
  • Маусым 5, 2020, 12:31 Т.Қ.

Понятно. Я не обратил внимания на то, что там было в старом коде по настройкам строк :)

IscanderChe
  • Маусым 5, 2020, 12:34 Т.Қ.

Сижу, размышляю: можно ли переписать делегата так, чтобы независимо от настроек строк выделялись строки?

Evgenii Legotckoi
  • Маусым 5, 2020, 12:39 Т.Қ.

По-моему, смысла в этом нет особого. Если делегат будет игнорировать настройки таблицы, то это приведёт ещё к большему непониманию, что вообще происходит, для программиста, который после вас будет смотреть на ваш проект.

Пікірлер

Тек рұқсаты бар пайдаланушылар ғана пікір қалдыра алады.
Кіріңіз немесе Тіркеліңіз