Помощь - Поиск - Пользователи - Календарь
Полная версия этой страницы: изменение высоты ячеек QTableView, для которого установлен делегат на основе QTextEdit
Форум на CrossPlatform.RU > Библиотеки > Qt > Qt Модель/Представление
Steklova Olga
Здравствуйте! :) Помогите, пожалуйста, разобраться.
У меня для одной из колонок QTableView установлен делегат на основе QTextEdit.
В процессе увеличения ширины окна с помощью мыши у меня не так, как хотелось бы, меняется отображение многострочного текста в ячейках QTableView. Вот что получается в процессе увеличения ширины окна (см. рис1):

Из всех состояний выглядят так, как я предполагала и хотела, только состояния 1, 3, 4, 8, 10, 12. В этих состояниях текст отображается не сокращенно, а полностью. Как сделать так, чтобы текст всегда отображался полностью - не понимаю. (Безусловно, я учитываю то, что это возможно только в том случае, если места, отведенного для QTableView в окне, хватает для отображения хотя бы одного самого широкого слова из текста всех ячеек таблицы.)

Предполагаю следующее.
Когда из состояния 1 я расширяю окно, то в тот момент, когда текст в какой-то ячейке может уже занять меньшее количество строк, tableView->verticalHeader() пересчитывает высоту строк tableView. Но почему-то при этом текст сначала отображается сокращенно, а позже, при дальнейшем расширении окна, отображается уже полностью.
Потом все это повторятся с какой-нибудь другой ячейкой.


Смотрела тут на форуме темы:
http://www.forum.crossplatform.ru/index.ph...ents+QTableView
("QTableView::resizeColumnsToContents, неожидаемое поведение")
и
http://www.forum.crossplatform.ru/index.ph...leView&st=0
("Изменение высоты и ширины ячеек., QTableView").
Но решения не нашла. Смотреть исходники Qt для меня сложно.

Код у меня такой:
создание делегата: http://www.forum.crossplatform.ru/index.php?showtopic=8385
mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QtGui/QMainWindow>

class QStandardItemModel;
class QTableView;

#include "delegate.h"

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow(QWidget *parent = 0);
    ~MainWindow();
private:
    QStandardItemModel* model;
    QTableView* tableView;
    TextEditDelegate* delegate;
};

#endif // MAINWINDOW_H
mainwindow.cpp
#include <QtGui/QStandardItemModel>
#include <QtGui/QTableView>
#include <QtGui/QHeaderView>

#include "mainwindow.h"

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
{
    model = new QStandardItemModel(5, 2);
    tableView = new QTableView();
    delegate = new TextEditDelegate();
    
    for (int row = 0; row < 5; ++row) {
        for (int column = 0; column < 2; ++column) {
            QModelIndex index = model->index(row, column, QModelIndex());
            switch(column)
            {
            case 0:
                model->setData(index, QVariant(
                        row+1)); break;
            default:
                switch(row)
                {
                case 0:
                    model->setData(index, QVariant(
                            "")); break;
                case 1:
                    model->setData(index, QVariant(
                            "Абв"
                            "  PRIVETPRIVET 1 01234"
                            "  PRIVET 2 01234"
                            "  PRIVET 3 01234"
                            "  PRIVET 4 01234")); break;
                case 2:
                    model->setData(index, QVariant(
                            "PRIVET PRIVET 1 01234"
                            "  Абв"
                            "  PRIVET 2 01234"
                            "  PRIVET 3 01234"
                            "  PRIVET 4 01234")); break;
                case 3:
                    model->setData(index, QVariant(
                            "Абв"
                            "  PRIVETPRIVET 1 01234"
                            "  PRIVET 2 01234"
                            "  PRIVET 3 01234")); break;
                default:
                    model->setData(index, QVariant(
                            "Абв"
                            "  PRIVETPRIVET 1 01234"
                            "  PRIVET 2 01234"
                            "  PRIVET 3 01234"
                            "  PRIVET 4 01234")); break;
                }
                break;
            }
        }
    }

    tableView->setModel(model);
    tableView->setItemDelegateForColumn(1, delegate);
    tableView->setEditTriggers(QAbstractItemView::NoEditTriggers);
    tableView->setSelectionMode(QAbstractItemView::SingleSelection);
    tableView->setSelectionBehavior(QAbstractItemView::SelectRows);

    tableView->resizeColumnToContents(0);
    tableView->horizontalHeader()->setStretchLastSection(true);

    tableView->verticalHeader()->hide();
    tableView->verticalHeader()->setResizeMode(QHeaderView::ResizeToContents);
//    tableView->setTextElideMode(Qt::ElideNone);

    if (model->rowCount() > 0)
        tableView->setCurrentIndex(model->index(0, 0));

    setCentralWidget(tableView);

    setWindowTitle("Text Edit Delegate");
    setGeometry(100, 250, 170, 500);
}

MainWindow::~MainWindow()
{

}

Подумала, что надо использовать
tableView->verticalHeader()->setResizeMode(QHeaderView::ResizeToContents);
чтобы высота строк tableView пересчитывалась автоматически.

Если отремить строку
tableView->setTextElideMode(Qt::ElideNone);
то возникает другая проблема. В этом случае текст всегда отображается не сокращенно, а полностью. Но зато высота строк tableView пересчитывается неверно и возникают состояния, когда высота получается меньше, чем необходимо для отображения всего текста ячеек. Вот так (см. рис2):



Так тоже не получается то, что хочется:
main.cpp
#include <QtCore/QTextCodec>
#include <QtGui/QApplication>

#include "mainwindow.h"

int main(int argc, char *argv[])
{
    QApplication app(argc, argv);
    QTextCodec::setCodecForCStrings(QTextCodec::codecForName("Windows-1251"));

    MainWindow mainWindow;
    mainWindow.show();

    mainWindow.slot_resizeRowsToContents();

    return app.exec();
}
mainwindow.cpp
MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
{
...
//    tableView->verticalHeader()->setResizeMode(QHeaderView::ResizeToContents);
//    tableView->setTextElideMode(Qt::ElideNone);
    connect(this->btn_resizeRowsToContents, SIGNAL(clicked()),
            this, SLOT(slot_resizeRowsToContents()));
...
}

void MainWindow::slot_resizeRowsToContents()
{
    setUpdatesEnabled(false);
    tableView->resizeRowsToContents();
    setUpdatesEnabled(true);

    tableView->update();
}
В этом случае, после нажатия кнопки "resizeRowsToContents", текст иногда продолжает отображаться сокращенно, вот так (см. рис3):


Может быть, я не в том порядке что-то пишу в своем коде? Или вообще что-то не так делаю?
:unknown:
В этом архиве - папка с файлами проекта и картинками экрана:
Нажмите для просмотра прикрепленного файла
Litkevich Yuriy
к сожалению придётся делать свой делегат, в котором нужно будет принимать решение о том как отображать текст при заданной ширине столбца.
Steklova Olga
А я рассчитывала на то, что мне хватит созданного мною примитивного делегата, а вот этот оператор сам за меня все автоматически пересчитает:
tableView->verticalHeader()->setResizeMode(QHeaderView::ResizeToContents);

Цитата
придётся делать свой делегат, в котором нужно будет принимать решение о том как отображать текст при заданной ширине столбца
Я совсем-совсем не понимаю, как это сделать... :unsure: А Вы знаете?
Steklova Olga
Теперь я поняла, что
Цитата
Так как представление у меня редактировать не надо, то делегата для редактирования мне не нужно,
надо просто указать подходящий режим для представления:
tableView->verticalHeader()->setResizeMode(QHeaderView::ResizeToContents);

Жаль только, что он работает, по-моему, кривовато, поэтому вопрос у меня остался открытым, как
Цитата(Litkevich Yuriy)
принимать решение о том, как отображать текст при заданной ширине столбца

У меня подозрение, что в какие-то моменты происходит переключение с установленного мною режима QHeaderView::ResizeToContents на какой-то другой.
Snake174
Попробуй в своей модели установить SizeHintRole:
QVariant Model::data( const QModelIndex &index, int role ) const
{
  if (!index.isValid())
    return QVariant();

  if (index.row() >= static_cast<int>( количество строк )
    return QVariant();

  if (index.column() >= количество столбцов)
    return QVariant();

  if (role == Qt::DisplayRole || role == Qt::ToolTipRole || role == Qt::StatusTipRole)
  {
    for (register int i = 0; i < количество столбцов; ++i)
    {
      if (index.column() == i)
        return текстовые данные;
    }
  }

  if (role == Qt::SizeHintRole)
  {
    QFontMetrics fm( QApplication::font() );
    int h = fm.boundingRect( текстовые данные ).height();
    
    QSize defSize;
    defSize.setHeight( h + 4 );

    return defSize;
  }

  return QVariant();
}


Попробуй в своей модели установить SizeHintRole:
QVariant Model::data( const QModelIndex &index, int role ) const
{
  if (!index.isValid())
    return QVariant();

  if (index.row() >= static_cast<int>( количество строк )
    return QVariant();

  if (index.column() >= количество столбцов)
    return QVariant();

  if (role == Qt::DisplayRole || role == Qt::ToolTipRole || role == Qt::StatusTipRole)
  {
    for (register int i = 0; i < количество столбцов; ++i)
    {
      if (index.column() == i)
        return текстовые данные;
    }
  }

  if (role == Qt::SizeHintRole)
  {
    QFontMetrics fm( QApplication::font() );
    int h = fm.boundingRect( текстовые данные ).height();
    
    QSize defSize;
    defSize.setHeight( h + 4 );

    return defSize;
  }

  return QVariant();
}
wiz29
попробуй просто отображать содержимое делегата для текстовой ячейки

Приведу схематично:

1. Устанавливаешь конкретный делегат для отображения строк.
2. Переопределяешь метод void QItemDelegate::drawDisplay ( QPainter * painter, const QStyleOptionViewItem & option, const QRect & rect, const QString & text ) const
Конфигурация области рисования в нее передается, далее просто отображаешь текст через QPainter c теми параметрами, которые требуются.
wiz29
Вот на основе изначального примера что получилось.

немного поменял файл
Steklova Olga
Спасибочки, wiz29, Вы очень умный! :give_rose:
При просмотре текст теперь отображается так, как было задумано.
Вот только я еще даже не смотрела класс QPainter. Посмотрю.
Надо мне хотя-бы книжку Земскова "Qt на примерах" полностью прочитать, а не писать программу интуитивно, бегая по верхам помощи (по классам Qt), книжек, форума и googlе :)

Есть только маленький вопросик сейчас: что в Вашем примере надо поменять, чтобы при просмотре представления цвет текста в выбранной строке в колонке с делегатом становился (как это принято) белым, а не так, как сейчас (черным) (см. рис)?


1. На будущее, если делать представление редактируемым,
то недостаточно переопределить только createEditor и drawDisplay так, как сейчас.

2. Сейчас мое представление не редактируемое,
поэтому переопределение createEditor в этом варианте я убрала, как совершенно не нужное.

3. Рассказываю дальше, зачем все это надо.
У меня в таблице БД сохраняется переписка двух абонентов (их сообщения друг другу в виде простого текста (не html), время посылки сообщения, ID написавшего сообщение абонента).
В соотв. с моим алгоритмом, в процессе переписки эту таблицу отображать не надо.
После окончания переписки (я знаю, когда это произойдет) надо отобразить таблицу сообщений.
Сначала надо отобразить все сообщения.
После этого можно будет их сортировать / фильтровать (это я знаю, как сделать).

Но вот я думаю, что, так как сообщения могут быть достаточно длинными, хорошо бы было предусмотреть возможность двух вариантов отображения сообщений и переключения между ними с помощью группы кнопок "Сокращенно", "Полностью".
Сначала отобразить все сообщения в первом варианте, сокращенно (каждое - в одну строку, с "..."),
при нажатии кнопки "Полностью" отобразить все сообщения во втором варианте, полностью,
при нажатии кнопки "Сокращенно" вернуться к первому варианту.
Мне кажется, это было бы очень удобно.
:D
Принимаются предложения, как прикрутить к имеющемуся коду сокращенный вариант.

А перерисовку может сократить как-то? Или не стоит об этом беспокоиться?
Например, не постоянно пересчитывать, как отобразить текст в процессе изменения размера окна,
а сделать это сначала при первом отображении таблицы сообщений,
а следующий раз (в случае изменения размера окна) сделать по окончании изменения размера окна, при отпускании мыши.


И еще, пока что, открыт вопрос, как корректно сохранять сообщения неизвестной заранее длины в БД Firebird 2.1 и отображать их потом в представлении (там проблемы с русским текстом и, видимо, с драйвером), об этом моя тема тут.
Если бы этот вопрос решился, то я вместо работы с сообщениями длиной не более 1000 символов,
перешла бы к работе с сообщениями, неограниченными по длине. Так надо по алгоритму.
И тогда эта ветка программы была бы доделана. ;)
wiz29
Цитата(Steklova Olga @ 5.4.2012, 16:07) *
что в Вашем примере надо поменять, чтобы при просмотре представления цвет текста в выбранной строке в колонке с делегатом становился (как это принято) белым, а не так, как сейчас (черным) (см. рис)?


Модификация.
Steklova Olga
Благодарю, Вас, wiz29 ! Теперь просмотр поля типа VARCHAR(1000) в QTableView чудно работает ! :clapping:
P.S. внесла дополнение в свое сообщение 8
Steklova Olga
Snake174, или Вы не правы, или я не знаю...
То, что написано у Вас в коде совсем не подходит. Посмотрите:
mymodel1.h
#ifndef MYMODEL1_H
#define MYMODEL1_H

#include <QStandardItemModel>

class MyModel1 : public QStandardItemModel
{
    Q_OBJECT
public:    
    explicit MyModel1(QObject *parent = 0);

    //второй конструктор (не использую)
    //MyModel1(int rows, int columns, QObject *parent = 0);

    QVariant data(const QModelIndex &index,
                  int role = Qt::DisplayRole) const;
};

#endif // MYMODEL1_H
mymodel1.cpp (привожу два варианта метода data)
#include "mymodel1.h"
#include <QtGui/QFontMetrics> //у него не было
#include <QtGui/QApplication> //у него не было

MyModel1::MyModel1(QObject *parent) :
    QStandardItemModel(parent)
{
}

//Попробуй в своей модели установить SizeHintRole:

//это ваш вариант в том виде, как я его поняла, он не работает
QVariant MyModel1::data(const QModelIndex &index,
                        int role/*у него нет = Qt::DisplayRole*/) const {

  //у него нет QVariant value = QStandardItemModel::data(index, role);

  //у него нет switch

  //у него есть зачем-то
  if (!index.isValid())
    //что это?
    return QVariant();

  //у него так, но это не работает
  //if (index.row() >= static_cast<int>( количество строк )
  //if (index.row() >= static_cast<int>(5)
  if (index.row() >= 5/* количество строк */)
      //что это?
    return QVariant();

  if (index.column() >= 2/*количество столбцов*/)
      //что это?
    return QVariant();

  if (role == Qt::DisplayRole || role == Qt::ToolTipRole || role == Qt::StatusTipRole)
  {
    for (register int i = 0; i < 2/*количество столбцов*/; ++i)
    {
      if (index.column() == i)
        //у него было
        //return текстовые данные;
        return QString("текстовые данные");
    }
  }

  if (role == Qt::SizeHintRole)
  {
    QFontMetrics fm( QApplication::font() );

    //у него было
    //int h = fm.boundingRect( текстовые данные ).height();
    int h = fm.boundingRect(QString("текстовые данные")).height();

    QSize defSize;
    defSize.setHeight( h + 4 );

    return defSize;
  }

  //что это?
  return QVariant();
}



////переделанный мною ваш вариант, он тоже не работает
//QVariant MyModel1::data(const QModelIndex &index,
//                        int role) const {

//  //по-моему, этого здесь не хватает
//  QVariant value = QStandardItemModel::data(index, role);
//  QVariant valueForDisplayRole = QStandardItemModel::data(index, Qt::DisplayRole);

//  //а потом, по-моему, было написано много лишнего,
//  //что я убрала

//  switch (role) {

//  case Qt::SizeHintRole: //
//      QFontMetrics fm( QApplication::font() );
//      QSize defSize;

//      //вы здесь просто берете высоту одной строки текста
//      int h = fm.boundingRect(valueForDisplayRole.toString()).height();

//      //по-моему, этого здесь не хватает
//      defSize = value.toSize();

//      //а здесь вы к высоте одной строки текста + const
//      defSize.setHeight( h + 50 ); //заменила на 50 для наглядности

//      //в результате высота всех строк у вас всегда одинаковая,
//      //равная высоте одной строки текста + const
//      return defSize;

//  } //switch (role)

//  return value;
//}
mainwindow.h
#ifndef MAINWINDOW_H
#define MAINWINDOW_H

#include <QtGui/QMainWindow>

class QStandardItemModel;
class QTableView;

//#include "delegate.h"
#include "mymodel1.h"

class MainWindow : public QMainWindow
{
    Q_OBJECT

public:
    MainWindow(QWidget *parent = 0);
    ~MainWindow();
private:
    //QStandardItemModel* model;
    MyModel1* model;
    QTableView* tableView;
    //TextEditDelegate* delegate;
};

#endif // MAINWINDOW_H
mainwindow.cpp
#include <QtGui/QStandardItemModel>
#include <QtGui/QTableView>
#include <QtGui/QHeaderView>

#include "mainwindow.h"

MainWindow::MainWindow(QWidget *parent)
    : QMainWindow(parent)
{
//    model = new QStandardItemModel(5, 2);

//    model = new QStandardItemModel();
//    model->setRowCount(5);
//    model->setColumnCount(2);

    model = new MyModel1();
    model->setRowCount(5);
    model->setColumnCount(2);

    tableView = new QTableView();
//    delegate = new TextEditDelegate();

    for (int row = 0; row < 5; ++row) {
        for (int column = 0; column < 2; ++column) {
            QModelIndex index = model->index(row, column, QModelIndex());
            switch(column)
            {
            case 0:
                model->setData(index, QVariant(
                        row+1)); break;
            default:
                switch(row)
                {
                case 0:
                    model->setData(index, QVariant(
                            "")); break;
                case 1:
                    model->setData(index, QVariant(
                            "Абв"
                            "  PRIVETPRIVET 1 01234"
                            "  PRIVET 2 01234"
                            "  PRIVET 3 01234"
                            "  PRIVET 4 01234")); break;
                case 2:
                    model->setData(index, QVariant(
                            "PRIVET PRIVET 1 01234"
                            "  Абв"
                            "  PRIVET 2 01234"
                            "  PRIVET 3 01234"
                            "  PRIVET 4 01234")); break;
                case 3:
                    model->setData(index, QVariant(
                            "Абв"
                            "  PRIVETPRIVET 1 01234"
                            "  PRIVET 2 01234"
                            "  PRIVET 3 01234")); break;
                default:
                    model->setData(index, QVariant(
                            "Абв"
                            "  PRIVETPRIVET 1 01234"
                            "  PRIVET 2 01234"
                            "  PRIVET 3 01234"
                            "  PRIVET 4 01234")); break;
                }
                break;
            }
        }
    }

    tableView->setModel(model);
//    tableView->setItemDelegateForColumn(1, delegate);
    tableView->setEditTriggers(QAbstractItemView::NoEditTriggers);
    tableView->setSelectionMode(QAbstractItemView::SingleSelection);
    tableView->setSelectionBehavior(QAbstractItemView::SelectRows);

    tableView->resizeColumnToContents(0);
    tableView->horizontalHeader()->setStretchLastSection(true);

    tableView->verticalHeader()->hide();
    tableView->verticalHeader()->setResizeMode(QHeaderView::ResizeToContents);
    //tableView->setTextElideMode(Qt::ElideNone);

    if (model->rowCount() > 0)
        tableView->setCurrentIndex(model->index(0, 0));

    setCentralWidget(tableView);

    setWindowTitle("Text Edit Delegate");
    setGeometry(100, 250, 170, 500);
}

MainWindow::~MainWindow()
{

}

Steklova Olga
wiz29, :) извините, пожалуйста, но Ваш код все-таки не всегда срабатывает. Посмотрите, что получается при увеличении ширины окна:
wiz29
Нужен код тестового проекта, к сожалению у меня его не сохранилось.
Steklova Olga
wiz29, код тестового примера по прежнему можно взять в Вами прикрепленном файле в 9-ом сообщении этой темы.
Надо только внести следующее изменение в код:
        case 1:
            model->setData(index, QVariant(
                    "ЗДРАВИЯ ЖЕЛАЮ, ТОВАРИЩ КОМАНДИР!"));
                    //"Абв"
                    //"  PRIVETPRIVET 1 01234"
                    //"  PRIVET 2 01234"
                    //"  PRIVET 3 01234"
                    //"  PRIVET 4 01234"));
                    break;
wiz29
Этот вариант рабочий, если где то будет что то криво, то нужно исправлять функцию делегата, вычисляющую предпочтительные размеры для ячейки.
Steklova Olga
Да, wiz29, теперь заработало, ура-а-а! Благодарю! :clapping:

Только в drawDisplay надо внести изменение, которое Вы раньше уже вносили, а сейчас забыли (видимо, взяли не последний прикрепленный файл из сообщения 9, а предпоследний из сообщения 7):
    //было painter->setPen(option.palette.color(cg, QPalette::Text));
    //стало
    if (option.state & QStyle::State_Selected) {
        painter->setPen(option.palette.color(cg, QPalette::HighlightedText));
    }
    else {
        painter->setPen(option.palette.color(cg, QPalette::Text));
    }

И правильнее будет написать
    //было delegate = new TextEditDelegate();
    //стало
    delegate = new TextEditDelegate(this);
wiz29
Да наверное так и есть, скачал не ту версию. :)
Для просмотра полной версии этой страницы, пожалуйста, пройдите по ссылке.
Форум IP.Board © 2001-2024 IPS, Inc.