Arrow
ArrowАқп. 28, 2017, 2:41 Т.Ж.

Создание делегата для редактирования данных в 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. Только не совсем как это там реализовать.

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

Рекомендуем хостинг TIMEWEB
Рекомендуем хостинг TIMEWEB
Стабильный хостинг, на котором располагается социальная сеть EVILEG. Для проектов на Django рекомендуем VDS хостинг.

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

8
Evgenii Legotckoi
  • Ақп. 28, 2017, 8:13 Т.Ж.

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

#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
    • Ақп. 28, 2017, 10:12 Т.Ж.

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

    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
      • Ақп. 28, 2017, 11:17 Т.Ж.

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

      model = new QFileSystemModel(this);
      model->setReadOnly(false);
        Arrow
        • Наурыз 1, 2017, 2:18 Т.Ж.

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

          Evgenii Legotckoi
          • Наурыз 1, 2017, 2:31 Т.Ж.
          • Жауап шешім ретінде белгіленді.

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

          ui->mainList->setEditTriggers(QAbstractItemView::AllEditTriggers);
            Arrow
            • Наурыз 1, 2017, 4:03 Т.Ж.

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

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

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

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

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

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

              Evgenii Legotckoi
              • Наурыз 1, 2017, 4:48 Т.Ж.

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

              void MainWindow::on_serchEdit_textChanged(const QString &arg1)
              {
                  model->setFilter(QDir::AllEntries | QDir::Files );
                  model->setNameFilterDisables(false);
                  model->setNameFilters(QStringList("*" + arg1 + "*"));
              }
                Arrow
                • Наурыз 1, 2017, 7:48 Т.Ж.

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

                  Пікірлер

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

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

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

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

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

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

                  • Нәтиже:20ұпай,
                  • Бағалау ұпайлары-10
                  Соңғы пікірлер
                  ИМ
                  Игорь МаксимовҚар. 22, 2024, 7:51 Т.Қ.
                  Django - Оқулық 017. Теңшелген Django кіру беті Добрый вечер Евгений! Я сделал себе авторизацию аналогичную вашей, все работает, кроме возврата к предидущей странице. Редеректит всегда на главную, хотя в логах сервера вижу запросы на правильн…
                  Evgenii Legotckoi
                  Evgenii LegotckoiҚаз. 31, 2024, 9:37 Т.Қ.
                  Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Добрый день. Да, можно. Либо через такие же плагины, либо с постобработкой через python библиотеку Beautiful Soup
                  A
                  ALO1ZEҚаз. 19, 2024, 3:19 Т.Қ.
                  Qt Creator көмегімен fb3 файл оқу құралы Подскажите как это запустить? Я не шарю в программировании и кодинге. Скачал и установаил Qt, но куча ошибок выдается и не запустить. А очень надо fb3 переконвертировать в html
                  ИМ
                  Игорь МаксимовҚаз. 5, 2024, 2:51 Т.Қ.
                  Django - Сабақ 064. Python Markdown кеңейтімін қалай жазуға болады Приветствую Евгений! У меня вопрос. Можно ли вставлять свои классы в разметку редактора markdown? Допустим имея стандартную разметку: <ul> <li></li> <li></l…
                  d
                  dblas5Шілде 5, 2024, 6:02 Т.Қ.
                  QML - Сабақ 016. SQLite деректер қоры және онымен QML Qt-та жұмыс істеу Здравствуйте, возникает такая проблема (я новичок): ApplicationWindow неизвестный элемент. (М300) для TextField и Button аналогично. Могу предположить, что из-за более новой верси…
                  Енді форумда талқылаңыз
                  m
                  moogoҚар. 22, 2024, 3:17 Т.Қ.
                  Mosquito Spray System Effective Mosquito Systems for Backyard | Eco-Friendly Misting Control Device & Repellent Spray - Moogo ; Upgrade your backyard with our mosquito-repellent device! Our misters conce…
                  Evgenii Legotckoi
                  Evgenii LegotckoiМаусым 24, 2024, 10:11 Т.Қ.
                  добавить qlineseries в функции Я тут. Работы оень много. Отправил его в бан.
                  t
                  tonypeachey1Қар. 15, 2024, 2:04 Т.Қ.
                  google domain [url=https://google.com/]domain[/url] domain [http://www.example.com link title]
                  NSProject
                  NSProjectМаусым 4, 2022, 10:49 Т.Ж.
                  Всё ещё разбираюсь с кешем. В следствии прочтения данной статьи. Я принял для себя решение сделать кеширование свойств менеджера модели LikeDislike. И так как установка evileg_core для меня не была возможна, ибо он писался…

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