无法自动调整QScrollArea的大小

4

我的环境:

  • Windows 10, 64位
  • Microsoft Visual Studio 2015
  • Qt 5.6.2
  • Qt Creator 4.8.1

我有一个对话框,其中包含一个滚动区域。滚动区域最初是空的:

empty scroll area

当用户单击复选框时,小部件会添加到滚动区域中:

widgets added, scroll area not resized

如上所示,只显示了滚动区域的一部分。我想自动扩大它(以及包含它的对话框),以便结果看起来像这样:

widgets added, manually resized

我的代码缺少什么?

我将问题简化为一个最小的示例程序(以上图片是从中获取的)。以下是代码:

scroll_area.pro:

QT += widgets
TEMPLATE = app
SOURCES += main.cpp my_dialog.cpp
HEADERS += my_dialog.h my_widget.h
TARGET = ScrollArea

main.cpp:

#include "my_dialog.h"
#include <QApplication>

int main(int argc, char* argv[])
{
    QApplication app(argc, argv);
    MyDialog myDialog;
    myDialog.show();
    return app.exec();
}

my_dialog.h:

#ifndef MY_DIALOG_H
#define MY_DIALOG_H

#include <QBoxLayout>
#include <QDialog>
#include <QScrollArea>

class MyDialog : public QDialog
{
    Q_OBJECT

public:
    MyDialog();

private:
    QScrollArea* m_scrollArea;
    QHBoxLayout* m_hLayout;

private slots:
    void changeScrollArea(int newState);
};

#endif // MY_DIALOG_H

my_dialog.cpp:

#include "my_dialog.h"
#include "my_widget.h"
#include <QCheckBox>
#include <QDialogButtonBox>

MyDialog::MyDialog() :
    QDialog(nullptr), m_scrollArea(new QScrollArea(this)), m_hLayout(new QHBoxLayout())
{
    setWindowTitle("My Dialog");

    QVBoxLayout* vlayout = new QVBoxLayout(this);

    // Set up the scroll area.
    m_scrollArea->setBackgroundRole(QPalette::ColorRole::Light);
    m_scrollArea->setHorizontalScrollBarPolicy(Qt::ScrollBarAsNeeded);
    vlayout->addWidget(m_scrollArea);

    // Set up the scroll contents.
    QWidget* scrollContents = new QWidget(this);
    scrollContents->setLayout(m_hLayout);
    m_scrollArea->setWidget(scrollContents);

    // Set up the check box.
    QCheckBox* checkbox = new QCheckBox("Check to add widgets", this);
    connect(checkbox, SIGNAL(stateChanged(int)), this, SLOT(changeScrollArea(int)));
    vlayout->addWidget(checkbox);
}

void MyDialog::changeScrollArea(int newState)
{
    // Set up the horizontal layout.
    m_hLayout->deleteLater();
    m_hLayout = new QHBoxLayout();

    // Set up the scroll contents.
    QWidget* scrollContents = new QWidget(this);
    scrollContents->setLayout(m_hLayout);
    if (newState == Qt::Checked)
    {
        m_hLayout->addWidget(new MyWidget(scrollContents));
        m_hLayout->addWidget(new MyWidget(scrollContents));
    }
    m_scrollArea->setWidget(scrollContents);
}

my_widget.h

#ifndef MY_WIDGET_H
#define MY_WIDGET_H

#include <QGridLayout>
#include <QPushButton>

class MyWidget : public QWidget
{
    Q_OBJECT

public:
    MyWidget(QWidget* parent) : QWidget(parent)
    {
        QGridLayout* gridLayout = new QGridLayout(this);

        for (int row = 0; row < 5; ++row)
        {
            for (int column = 0; column < 1; ++column)
            {
                QPushButton* button = new QPushButton("FOO");
                gridLayout->addWidget(button, row, column);
            }
        }
    }
};

#endif // MY_WIDGET_H
1个回答

2

QScrollArea不会根据其内容更改大小,因为其任务是建立一个可以在视口中具有大尺寸的小部件,因此解决方案是使用代码更改大小:

mydialog.h

"最初的回答"

#ifndef MYDIALOG_H
#define MYDIALOG_H

#include <QDialog>

class QScrollArea;
class QHBoxLayout;

class MyDialog : public QDialog
{
    Q_OBJECT
public:
    MyDialog();
private:
    QScrollArea* m_scrollArea;
    QHBoxLayout* m_hLayout;
private slots:
    void changeScrollArea(int newState);
    void adjust();
};
#endif // MYDIALOG_H

mydialog.cpp

#include "mydialog.h"
#include "mywidget.h"

#include <QCheckBox>
#include <QHBoxLayout>
#include <QScrollArea>
#include <QScrollBar>
#include <QTimer>

MyDialog::MyDialog() :
    QDialog(nullptr), m_scrollArea(new QScrollArea), m_hLayout(new QHBoxLayout())
{
    setWindowTitle("My Dialog");
    QVBoxLayout* vlayout = new QVBoxLayout(this);
    // Set up the scroll area.
    m_scrollArea->setBackgroundRole(QPalette::ColorRole::Light);
    m_scrollArea->setHorizontalScrollBarPolicy(Qt::ScrollBarAsNeeded);
    m_scrollArea->setWidgetResizable(true);
    vlayout->addWidget(m_scrollArea);
    // Set up the scroll contents.
    QWidget* scrollContents = new QWidget;
    scrollContents->setLayout(m_hLayout);
    m_scrollArea->setWidget(scrollContents);
    // Set up the check box.
    QCheckBox* checkbox = new QCheckBox("Check to add widgets");
    connect(checkbox, &QCheckBox::stateChanged, this, &MyDialog::changeScrollArea);
    vlayout->addWidget(checkbox);
}
void MyDialog::changeScrollArea(int newState)
{
    m_hLayout->deleteLater();
    // Set up the scroll contents.
    QWidget* scrollContents = new QWidget;
    // Set up the horizontal layout.
    m_hLayout = new QHBoxLayout(scrollContents);
    if (newState == Qt::Checked)
    {
        m_hLayout->addWidget(new MyWidget(scrollContents));
        m_hLayout->addWidget(new MyWidget(scrollContents));
    }
    m_scrollArea->setWidget(scrollContents);
    m_scrollArea->resize(m_scrollArea->minimumSizeHint());
    resize(sizeHint());
    QTimer::singleShot(0, this, &MyDialog::adjust);
}

void MyDialog::adjust()
{
    const QSize max_size(1000, 1000);
    const int step = 5;
    while (m_scrollArea->verticalScrollBar()->isVisible() && height() < max_size.height())
        resize(width(), height() + step);
    while (m_scrollArea->horizontalScrollBar()->isVisible() && width() < max_size.width())
        resize(width()+step, height());
}

它有效。感谢您抽出时间发布这个。 - user1387866

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接