使QScrollArea动态适应其父对象,但不为其子对象扩展

Having a QScrollArea dynamically fit its parent but does not expand for its children

本文关键字:对象 扩展 QScrollArea 动态      更新时间:2023-10-16

我似乎在从QScrollArea获得我想要的行为方面遇到了问题。目前,每当我向设置为滚动区域目标的小部件的布局添加内容时,它都会选择扩展整个窗口,而不是适合滚动。

这是我目前的设置:

QSplitter * mainArea = new QSplitter( Qt::Vertical );
QWidget * containment = new QWidget;
containment->setLayout( new QVBoxLayout );
currentStructures = new QWidget;
currentStructures->setLayout( new QVBoxLayout );
currentStructures->layout()->setAlignment( Qt::AlignTop );
QScrollArea * scroll = new QScrollArea();
scroll->setWidget( currentStructures );
containment->layout()->addWidget( currentStructures );
mainArea->addWidget( containment );
mainArea->addWidget( new QWidget ); //TODO: create preview bar
this->layout()->addWidget( mainArea );

这使得滚动区域只会扩展,而不会显示滚动条。

插入以下行:

containment->setSizePolicy( QSizePolicy::Expanding, QSizePolicy::Ignored );

我可以让该区域忽略其子对象的大小,但它也不会占用所需的空间,也不会显示滚动条——它只会挤压其中的小部件。

我对使用Qt有点生疏,但我想知道如何实现我需要的行为:我希望滚动区域贪婪地占据它从父布局中可用的区域,但在添加时不垂直扩展包含的布局,而是实际显示滚动条。我计划允许大量调整大小,所以它需要实际缩放到父对象,而不仅仅是固定大小。我不知道如何解决GUI的这一方面。谢谢你抽出时间。

如果你想直观地解决这个问题,这里有一个你可以使用的安全带。我非常感谢你的帮助。

#include <QtWidgetsqapplication.h>
#include <QtWidgetsqsplitter.h>
#include <QtWidgetsqlayout.h>
#include <QtWidgetsqscrollarea.h>
#include <QtWidgetsqpushbutton.h>
#include <QtWidgetsqlabel.h>
#include <QtWidgetsqsizepolicy.h>
int main( int argc, char * argv[] )
{
    QApplication app( argc, argv );
    QWidget * testWidget = new QWidget;
    testWidget->setLayout( new QVBoxLayout );
    //////////////////CODE IN QUESTIION//////////////////////
    QSplitter * mainArea = new QSplitter( Qt::Vertical );
    QWidget * containment = new QWidget;
    containment->setLayout( new QVBoxLayout );
    //containment->setSizePolicy( QSizePolicy::Expanding, QSizePolicy::Ignored );
    QWidget * currentStructures = new QWidget;
    currentStructures->setLayout( new QVBoxLayout );
    currentStructures->layout()->setAlignment( Qt::AlignTop );
    QScrollArea * scroll = new QScrollArea();
    scroll->setWidget( currentStructures );
    containment->layout()->addWidget( currentStructures );
    mainArea->addWidget( containment );
    ///////////////////////////////////////////////////////////
    QPushButton * pushIntoLayout = new QPushButton( "Add Element to Widget" );
    QWidget::connect( pushIntoLayout, &QPushButton::clicked, [currentStructures](){ currentStructures->layout()->addWidget( new QLabel( "A generated label" ) ); } );
    mainArea->addWidget( pushIntoLayout );
    currentStructures->setStyleSheet(
        "QWidget {"
            "background-color: #FAA;"
        "}" 
    );

    testWidget->layout()->addWidget( mainArea );
    testWidget->show();
    return app.exec();
}

即使在复制并粘贴我的代码以制作测试工具之后,我也没有注意到我的严重错误。我错误地将currentStructures小部件推到了容器的布局中,而不是scroll,也就是给它子对象后的滚动区域。

void QScrollArea::setWidget( QWidget * widget )上的Qt文档摘录,供不熟悉的人使用:

小部件成为滚动区域的子部件,当删除滚动区域或设置新的小部件时,小部件将被销毁。

感谢所有看过的人。

对于那些想知道的人来说,固定代码如下所示:

QSplitter * mainArea = new QSplitter( Qt::Vertical );
QWidget * containment = new QWidget;
containment->setLayout( new QVBoxLayout );
QWidget * currentStructures = new QWidget;
currentStructures->setLayout( new QVBoxLayout );
currentStructures->layout()->setAlignment( Qt::AlignTop );
QScrollArea * scroll = new QScrollArea();
scroll->setWidget( currentStructures );
scroll->setWidgetResizable( true );
containment->layout()->addWidget( scroll );
mainArea->addWidget( containment );
mainArea->addWidget( new QWidget ); //TODO: create preview bar
this->layout()->addWidget( mainArea );

干杯!