IscanderChe
IscanderChe22 января 2019 г. 8:45

QSqlTableModel + QTableView + кастомный делегат в виде чекбокса

Qt, QSqlTableModel, QTableView

Всем добрый день.

Суть задачи: надо, чтобы в одной из колонок QTableView вместо хранимого в QSqlTableModel значения выводился чекбокс и при смене состояния чекбокса значения в базе тоже менялось. С помощью ItemIsUserCheckable проблему удалось решить.

Есть ли вариант решения с помощью кастомного делегата? У меня получается только частично: чекбокс виден только при щелчке на ячейку, данные исправно меняет, но при переходе к другой ячейке чекбокс заменяется данными.

class CheckBoxDelegate : public QStyledItemDelegate
{
    Q_OBJECT

public:
    CheckBoxDelegate(QObject* parent = 0);

    QWidget* createEditor(QWidget* parent, const QStyleOptionViewItem& option,
                          const QModelIndex& index) const override;

    void setEditorData(QWidget* editor, const QModelIndex& index) const override;
    void setModelData(QWidget *editor, QAbstractItemModel *model,
                      const QModelIndex &index) const override;

    void updateEditorGeometry(QWidget *editor,
            const QStyleOptionViewItem &option, const QModelIndex &index) const override;
};
QWidget* CheckBoxDelegate::createEditor(QWidget* parent,
    const QStyleOptionViewItem& option,
    const QModelIndex& index) const
{
    if(index.column() == 1)
    {
        QCheckBox* editor = new QCheckBox(parent);
        return editor;
    }

    return QStyledItemDelegate::createEditor(parent, option, index);
}

void CheckBoxDelegate::setEditorData(QWidget* editor,
    const QModelIndex& index) const
{
    if(index.column() == 1)
    {
        int value = index.model()->data(index, Qt::EditRole).toBool();
        QCheckBox* checkBox = static_cast<QCheckBox*>(editor);
        checkBox->setChecked(value != 0);
    }
}

void CheckBoxDelegate::setModelData(QWidget* editor,
    QAbstractItemModel* model, const QModelIndex& index) const
{
    if(index.column() == 1)
    {
        QCheckBox* checkBox = static_cast<QCheckBox*>(editor);
        bool value = checkBox->isChecked();
        model->setData(index, value, Qt::EditRole);
    }
}

void CheckBoxDelegate::updateEditorGeometry(QWidget* editor,
    const QStyleOptionViewItem& option, const QModelIndex& index) const
{
    editor->setGeometry(option.rect);
}
Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

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

9
Evgenii Legotckoi
  • 23 января 2019 г. 15:44
  • (ред.)

Добрый день! Про ваш вопрос помню, вынужден немного отложить, работы много. Чуть позже гляну кое-какие рабочие исходники с делегатами, нечто подобное уже реализовывал.

    IscanderChe
    • 15 февраля 2019 г. 16:22

    Доброй ночи.
    Скромно напоминаю о своём вопросе...

      Evgenii Legotckoi
      • 17 февраля 2019 г. 12:28

      Добрый день. Очень извиняюсь за долгий ответ

      Первое, что нашёл, так это необходимость перерисовать чекбокс.

      void CheckBoxDelegate::paint(QPainter *painter, const QStyleOptionViewItem &option, const QModelIndex &index) const
      {
          if (index.column() == 1)
          {
              painter->save();
              bool checked = index.model()->data(index, Qt::DisplayRole).toBool();
      
              QStyleOptionButton check_box_style_option;
              check_box_style_option.state |= QStyle::State_Enabled;
              if (checked) {
                  check_box_style_option.state |= QStyle::State_On;
              } else {
                  check_box_style_option.state |= QStyle::State_Off;
              }
              check_box_style_option.rect = option.rect;
      
              QApplication::style()->drawControl(QStyle::CE_CheckBox, &check_box_style_option, painter);
              painter->restore();
              return;
          }
      
          QStyledItemDelegate::paint(painter, option, index);
      }
      
        IscanderChe
        • 19 февраля 2019 г. 1:51

        Да, теперь чекбокс виден постоянно. Но для того, чтобы изменить его состояние, прежде нужно выделить ячейку с ним. Это как-то лечится?

          Evgenii Legotckoi
          • 19 февраля 2019 г. 3:34

          Попробуйте включить все триггеры редактирования

          yourView->setEditTriggers(QAbstractItemView::AllEditTriggers)
          
            IscanderChe
            • 19 февраля 2019 г. 7:03

            Тот же эффект.

              Evgenii Legotckoi
              • 20 февраля 2019 г. 9:40

              Думаю, что ещё можно переопределить mouseReleaseEvent(QMouseEvent* event) у QTableView, который содержит модель и немного поиграться с индексом. Если это индекс, который соответствует колонке с чекбоксом, то изменить значение принудительно не дожидаясь перехода ячейки в режим редактирования.

                IscanderChe
                • 20 июня 2019 г. 5:30

                Вернулся к этой задачке только-только, поэтому и не ответил ничего раньше.
                Как переопределить mouseReleaseEvent(QMouseEvent* event) у QTableView, если QTableView задан в ui? Или задавать QTableView только через код?

                  IscanderChe
                  • 21 июня 2019 г. 3:17

                  Всё, с этим разобрался.

                    Комментарии

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

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

                    • Результат:66баллов,
                    • Очки рейтинга-1
                    t

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

                    • Результат:33баллов,
                    • Очки рейтинга-10
                    t

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

                    • Результат:52баллов,
                    • Очки рейтинга-4
                    Последние комментарии
                    G
                    GoattRock3 сентября 2024 г. 20:50
                    Как Копировать Файлы в Linux Задумывались когда-нибудь о том, как мы привыкли доверять свои вещи службам грузоперевозок? Сейчас такие услуги стали неотъемлемой частью нашей жизни, особенно когда речь идет о переездах между …
                    ВР
                    Влад Русоков2 августа 2024 г. 8:47
                    Как Копировать Файлы в Linux Screenshot_20240802-065123.png
                    d
                    dblas55 июля 2024 г. 18:02
                    QML - Урок 016. База данных SQLite и работа с ней в QML Qt Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                    k
                    kmssr9 февраля 2024 г. 2:43
                    Qt Linux - Урок 001. Автозапуск Qt приложения под Linux как сделать автозапуск для флэтпака, который не даёт создавать файлы в ~/.config - вот это вопрос ))
                    АК
                    Анатолий Кононенко5 февраля 2024 г. 9:50
                    Qt WinAPI - Урок 007. Работаем с ICMP Ping в Qt Без строки #include <QRegularExpressionValidator> в заголовочном файле не работает валидатор.
                    Сейчас обсуждают на форуме
                    Evgenii Legotckoi
                    Evgenii Legotckoi24 июня 2024 г. 22:11
                    добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
                    F
                    Fynjy22 июля 2024 г. 11:15
                    при создании qml проекта Kits есть но недоступны для выбора Поставил Qt Creator 11.0.2. Qt 6.4.3 При создании проекта Qml не могу выбрать Kits, они все недоступны, хотя настроены и при создании обычного Qt Widget приложения их можно выбрать. В чем может …
                    BlinCT
                    BlinCT25 июня 2024 г. 8:00
                    Нарисовать кривую в qml Всем привет. Имеется Лист листов с тосками, точки получаны интерполяцией Лагранжа. Вопрос, как этими точками нарисовать кривую? ChartView отпадает сразу, в qt6.7 появился новый элемент…
                    BlinCT
                    BlinCT5 мая 2024 г. 12:46
                    Написать свой GraphsView Всем привет. В Qt есть давольно старый обьект дял работы с графиками ChartsView и есть в 6.7 новый но очень сырой и со слабым функционалом GraphsView. По этой причине я хочу написать х…
                    Evgenii Legotckoi
                    Evgenii Legotckoi2 мая 2024 г. 21:07
                    Мобильное приложение на C++Qt и бэкенд к нему на Django Rest Framework Добрый день. По моему мнению - да, но то, что будет касаться вызовов к функционалу Андроида, может создать огромные трудности.

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