Arrow
Feb. 28, 2017, 1:41 p.m.

Создание делегата для редактирования данных в QListView

Есть объект QListView у которого установлено:

setViewMode(QListView::IconMode);

Отображаемые данные находятся в модели.

Хочется обеспечить возможность редактировать надписи под иконками в QListView. Для этого создал делегата код которого приведен ниже.

// Заголовочный файл *.h

#include <QItemDelegate>
#include <QLineEdit>
#include <QModelIndex>

class LineEditDelegate : public QItemDelegate
{
    Q_OBJECT

public:
    LineEditDelegate(QObject *parent = 0);

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

    void setEditorData(QWidget *editor, const QModelIndex &index) const;

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

    void updateEditorGeometry(QWidget *editor,
                              const QStyleOptionViewItem &option, const QModelIndex &index) const;
};
// Файл реализации *.cpp

#include "lineeditdelegate.h"

LineEditDelegate::LineEditDelegate(QObject *parent): QItemDelegate(parent)
{

}

QWidget *LineEditDelegate::createEditor(QWidget *parent, const QStyleOptionViewItem & /*option*/, const QModelIndex & /*index*/) const
{
     QLineEdit *editor = new QLineEdit(parent);
     return editor;
}

void LineEditDelegate::setEditorData(QWidget *editor, const QModelIndex &index) const
{
    QString value = index.model()->data(index, Qt::DisplayRole).toString();
    QLineEdit *edit = static_cast<QLineEdit*>(editor);
    edit->setText(value);
}

void LineEditDelegate::setModelData(QWidget *editor, QAbstractItemModel *model, const QModelIndex &index) const
{
    QLineEdit *edit = static_cast<QLineEdit*>(editor);
    QString value = edit->text();
    model->setData(index, value, Qt::EditRole);
}

void LineEditDelegate::updateEditorGeometry(QWidget *editor, const QStyleOptionViewItem &option, const QModelIndex &index) const
{
    editor->setGeometry(option.rect);
}

Только теперь остался последний вопрос как его прикрутить к QListView. Пытался сделать так:

MainWindow::MainWindow(QWidget *parent) :
    QMainWindow(parent),
    ui(new Ui::MainWindow)
{
    ui->setupUi(this);

    ui->mainList->setSelectionMode(QAbstractItemView::ExtendedSelection);
    ui->mainList->setDragEnabled(true);
    ui->mainList->setAcceptDrops(true);
    ui->mainList->setDropIndicatorShown(true);
    LineEditDelegate *delegate = new LineEditDelegate(this);
    ui->mainList->setItemDelegate(delegate);
}

и пробовал создавать отдельно процедуру для редактирования данных. Ничего не помогло. С делегатами в Qt до этого не работал и нигде не смог найти как его правильно прикручивать к отображению.

Как вариант возможно нужно создать свой класс на основе QListView. Только не совсем как это там реализовать.

Как правильно это реализовать и менее затратно по ресурсам?

2

Do you like it? Share on social networks!

8
Evgenii Legotckoi
  • Feb. 28, 2017, 7:13 p.m.

Я может быть чего-то не догоняю, только зачем здесь делегат? И так текст редактируется без каких-либо проблем.

#include <QStandardItemModel>
#include <QStandardItem>
#include <QStyle>

Widget::Widget(QWidget *parent) :
    QWidget(parent),
    ui(new Ui::Widget)
{
    ui->setupUi(this);

    ui->mainList->setSelectionMode(QAbstractItemView::ExtendedSelection);
    ui->mainList->setDragEnabled(true);
    ui->mainList->setAcceptDrops(true);
    ui->mainList->setDropIndicatorShown(true);
    ui->mainList->setViewMode(QListView::IconMode);

    QStandardItemModel *model = new QStandardItemModel();

    QStandardItem *item2 = new QStandardItem();
    item2->setIcon(QIcon(style()->standardIcon(QStyle::SP_DirOpenIcon)));
    item2->setText("Item 2");

    QStandardItem *item1 = new QStandardItem();
    item1->setIcon(QIcon(style()->standardIcon(QStyle::SP_DirOpenIcon)));
    item1->setText("Item 1");

    model->appendRow(item1);
    model->appendRow(item2);

    ui->mainList->setModel(model);
}

Вот например моделька с двумя иконками и текстом под ними. Редактирование происходит.

    Arrow
    • Feb. 28, 2017, 9:12 p.m.

    Так да все работает - это я знаю, а если так:

    Widget::Widget(QWidget *parent) :
        QWidget(parent),
        ui(new Ui::Widget)
    {
        ui->setupUi(this);
    
        ui->mainList->setSelectionMode(QAbstractItemView::ExtendedSelection);
        ui->mainList->setDragEnabled(true);
        ui->mainList->setAcceptDrops(true);
        ui->mainList->setDropIndicatorShown(true);
        ui->mainList->setViewMode(QListView::IconMode);
    
        model = new QFileSystemModel(this);
        ui->mainList->setModel(model);
    }
      Evgenii Legotckoi
      • Feb. 28, 2017, 10:17 p.m.

      В данном конкретном случае достаточно лишь отключить режим readOnly у QFileSystemModel .

      model = new QFileSystemModel(this);
      model->setReadOnly(false);
        Arrow
        • March 1, 2017, 1:18 p.m.

        Спасибо, только как перевести модель (model) или представление (QListView) в режим редактирования без двойного клика по элементу, а скажем по нажатию на клавишу?

          Evgenii Legotckoi
          • March 1, 2017, 1:31 p.m.
          • The answer was marked as a solution.

          Включите все флаги редактирования для QListView

          ui->mainList->setEditTriggers(QAbstractItemView::AllEditTriggers);
            Arrow
            • March 1, 2017, 3:03 p.m.

            Спасибо все работает.

            Можно еще один вопрос?

            Пытаюсь организовать поиск данных по введенной пользователем строке.

            Реализовал это так:

            void MainWindow::on_serchEdit_textChanged(const QString &arg1)
            {
                model->setFilter(QDir::AllEntries | QDir::Files);
                model->setNameFilterDisables(false);
                model->setNameFilters(QStringList(arg1));
            }

            Только поиск происходит при полном соответствии введенной строки данным. Как можно организовать поиск по всем вхождениям введенных символов.

              Evgenii Legotckoi
              • March 1, 2017, 3:48 p.m.

              Вот таким вот образом работает

              void MainWindow::on_serchEdit_textChanged(const QString &arg1)
              {
                  model->setFilter(QDir::AllEntries | QDir::Files );
                  model->setNameFilterDisables(false);
                  model->setNameFilters(QStringList("*" + arg1 + "*"));
              }
                Arrow
                • March 1, 2017, 6:48 p.m.

                Спасибо! Я об этом даже и не подумал.

                  Comments

                  Only authorized users can post comments.
                  Please, Log in or Sign up
                  • Last comments
                  • AK
                    April 1, 2025, 11:41 a.m.
                    Добрый день. В данный момент работаю над проектом, где необходимо выводить звук из программы в определенное аудиоустройство (колонки, наушники, виртуальный кабель и т.д). Пишу на Qt5.12.12 поско…
                  • Evgenii Legotckoi
                    March 9, 2025, 9:02 p.m.
                    К сожалению, я этого подсказать не могу, поскольку у меня нет необходимости в обходе блокировок и т.д. Поэтому я и не задавался решением этой проблемы. Ну выглядит так, что вам действитель…
                  • VP
                    March 9, 2025, 4:14 p.m.
                    Здравствуйте! Я устанавливал Qt6 из исходников а также Qt Creator по отдельности. Все компоненты, связанные с разработкой для Android, установлены. Кроме одного... Когда пытаюсь скомпилиров…
                  • ИМ
                    Nov. 22, 2024, 9:51 p.m.
                    Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
                  • Evgenii Legotckoi
                    Oct. 31, 2024, 11:37 p.m.
                    Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup