Как заставить изменение размеров indexWidgets вписываться в ячейки Qt5 QTableView - PullRequest
0 голосов
/ 11 декабря 2018

Я запускаю приведенный ниже пример кода с Qt5.11.0 на OSX 10.13.6 (также RHEL 7.6, где проблема также возникает, но не так страшно, как на OSX).Тестовая программа отображает пользовательскую модель в QTableView с indexWidgets, установленным для нескольких столбцов:

#include <QtCore/QDebug>
#include <QtCore/QAbstractItemModel>
#include <QtWidgets/QApplication>
#include <QtWidgets/QMainWindow>
#include <QtWidgets/QHBoxLayout>
#include <QtWidgets/QHeaderView>
#include <QtWidgets/QTableView>
#include <QtWidgets/QPushButton>
#include <QtWidgets/QRadioButton>

class AModel : public QAbstractItemModel
{
    public:
    int            rowCount( const QModelIndex& parent = QModelIndex() ) const override { 
                           return 5;    
                       };
    int            columnCount( const QModelIndex& parent = QModelIndex() ) const override { 
                           return 5;    
                       };
    QModelIndex    parent( const QModelIndex& index ) const override { 
                           return QModelIndex();    
                       };     
    QModelIndex    index( int row, int column, const QModelIndex& parent = QModelIndex() ) const override {
                           if( ( ! parent.isValid() ) &&
                               row >= 0 && row < 5 &&
                               column >= 0 && column < 5 ) {
                               return createIndex( row, column );
                           } else {
                               return QModelIndex();
                           }
                       };
    QVariant       data( const QModelIndex& index, int role = Qt::DisplayRole ) const override {
                           QVariant qval;
                           if( index.column() >= 1 && index.column() < 4 ) { return QVariant(); }
                           switch( role ) {
                           case Qt::DisplayRole:
                               qval = QString( "%1,%2" ).arg( index.row() ).arg( index.column() );
                               break;
                           default:
                               qval = QVariant();
                               break;
                           }
                           return qval;
                       };
};

class AWidget : public QWidget
{
    public:
            AWidget( QWidget* parent ) : QWidget( parent ) {
                    QHBoxLayout* l = new QHBoxLayout();
                    this->setLayout( l );
                    QRadioButton* save = new QRadioButton( "Save" );
                    QRadioButton* del = new QRadioButton( "Delete" );
                    l->addWidget( save );
                    l->addWidget( del );
                };
};

int
main( int argc, char *argv[] ) {
    QApplication   app( argc, argv );
    QMainWindow*   mw = new QMainWindow();
    AModel*        model = new AModel();
    QTableView*    view = new QTableView();

    view->setModel( model );

    // view->verticalHeader()->setDefaultSectionSize( 15 );

    for( int irow = 0; irow < model->rowCount(); irow++ ) {
        QPushButton*   pb = new QPushButton( "Mogrify", mw );
        QRadioButton*  rb = new QRadioButton( "Choose", mw );
        AWidget*       aw = new AWidget( mw );

        QObject::connect( pb, &QPushButton::clicked, [irow](){ qDebug() << "Mogrifying " << irow; } );
        QObject::connect( rb, &QRadioButton::clicked, [irow](){ qDebug() << "Choosing " << irow; } );

        view->setIndexWidget( model->index( irow, 1 ), pb );
        view->setIndexWidget( model->index( irow, 2 ), rb );
        view->setIndexWidget( model->index( irow, 3 ), aw );
    }

    view->resizeColumnsToContents();

    mw->setCentralWidget( view );

    mw->show();

    return app.exec();
}

Если я просто запустите это, как показано выше, результат получится со всеми встроенными в таблицу виджетами, имеющимидостаточно места:

table-embedded widgets sizing properly

Если, однако, я раскомментирую вызов setDefaultSectionSize () в приведенном выше коде, встроенные в таблицу виджеты сами по себе не изменяют размерЯ бы хотел.QPushButton обрезается внизу, QRadioButton заполнен небольшим отступом, а пользовательский составной виджет вообще не отображается:

table-embedded widgets sizing improperly

Я пробовал всевозможные эксперименты QSizeHint, создание подклассов и поиск в Интернете, чтобы заставить эти встроенные виджеты измерять себя в соответствии с пространством, доступным в ячейках таблицы, пока что безрезультатно.Как заставить эти встроенные indexWidgets нарисовать себя так, чтобы они вписывались в пространство ячеек, предоставляемое в QTableView, когда я говорю QTableView, насколько большими должны быть его ячейки?

1 Ответ

0 голосов
/ 12 декабря 2018

Проблема не в QTableView, а в вашем пользовательском виджете.Пользовательский виджет имеет макет, поля которого должны быть равны 0.

class AWidget : public QWidget
{
public:
    AWidget( QWidget* parent=nullptr) :
        QWidget( parent )
    {
        QHBoxLayout* l = new QHBoxLayout(this);
        l->setContentsMargins(0, 0, 0, 0);  // <----
        QRadioButton* save = new QRadioButton( "Save" );
        QRadioButton* del = new QRadioButton( "Delete" );
        l->addWidget( save );
        l->addWidget( del );
    };
};

enter image description here

...