gpt4 book ai didi

c++ - 带有动态添加的小部件的 QScrollArea

转载 作者:太空狗 更新时间:2023-10-29 20:09:25 28 4
gpt4 key购买 nike

我在 QScrollArea 中放置自定义小部件时遇到了一些问题。我的自定义小部件在 QGridLayout 中包含 4 个标签来填充它。现在我想将此小部件保留在 QScrollArea 中并能够向其添加更多标签,但我只希望其中的 4 个显示在视口(viewport)中。

这就是带有 4 个标签的小部件在 QScrollArea 中的样子: enter image description here

这是添加了两个标签后 QScrollArea 中的小部件,其中红色矩形是视口(viewport)。 enter image description here

我怎样才能达到这样的结果?

===================================

更新

我最终使用以下代码解决了我的问题。它可能需要一些小的间距修复。

#include "QtGuiApplication2.h"
#include "qscrollarea.h"
#include "CustomWidget.h"

QtGuiApplication2::QtGuiApplication2(QWidget *parent)
: QMainWindow(parent)
{
ui.setupUi(this);

QScrollArea * qScrollArea = new QScrollArea();

CustomWidget * customWidget = new CustomWidget(this);
qScrollArea->setWidget(customWidget);
qScrollArea->setWidgetResizable(true);
qScrollArea->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOn);

ui.mainLayout->addWidget(qScrollArea, 1, 1, 1, 1);
}

CustomWidget 类:

#include "CustomWidget.h"

#include "qlabel.h"

CustomWidget::CustomWidget(QWidget *parent) : QWidget(parent)
{
labelsNum = 4;
rows = 2;
layout = new QGridLayout();
this->setLayout(layout);
QMargins * margin = new QMargins(10, 10, 10, 10);
layout->setContentsMargins(*margin);
layout->setHorizontalSpacing(5);
layout->setVerticalSpacing(5);
initLabels();
addLabels();
}

CustomWidget::~CustomWidget()
{
}

void CustomWidget::initLabels()
{
int cols = labelsNum / rows;

for (int i = 0; i < labelsNum; i++)
{
CustomLabel * label = new CustomLabel(this);
label->setText(QString::number(i));
label->setFrameShape(QFrame::Box);

labels.append(label);
}

for (int i = 0; i < rows; i++)
{
for (int j = 0; j < cols; j++)
{
layout->addWidget(labels.at(i * cols + j), i + 1, j + 1, 1, 1);
}
}
}

void CustomWidget::addLabels()
{

int numLabels = labels.size();

for (int i = 0; i < 2; i++)
{
CustomLabel * label = new CustomLabel(this);
label->setText(QString::number(numLabels + i));
label->setFrameShape(QFrame::Box);

labels.append(label);
}

labelsNum += rows;
int cols = labelsNum / rows;

for (int i = 0; i < rows; i++)
{
for (int j = 0; j < cols; j++)
{
layout->addWidget(labels.at(i * cols + j), i + 1, j + 1, 1, 1);
}
}
}

void CustomWidget::resizeEvent(QResizeEvent * e)
{
QWidget::resizeEvent(e);
QSize size = viewportSize;

//Substract all the spacing from view size
int horizontalSpacing = ((4 / rows) - 1) * layout->horizontalSpacing();
int verticalSpacing = (rows - 1) * layout->verticalSpacing();
size -= QSize(layout->margin() * 2 + horizontalSpacing, layout->margin() * 2 + verticalSpacing);

size *= 0.5;
for (int i = 0; i < labels.size(); i++)
{
labels.at(i)->resizeEvent(e, size);
}
}

和自定义标签:

#include "CustomLabel.h"

CustomLabel::CustomLabel(QWidget *parent): QLabel(parent)
{
setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
setMinimumSize(QSize(50, 50));
}

CustomLabel::~CustomLabel()
{
}

void CustomLabel::resizeEvent(QResizeEvent * e, QSize size)
{
this->setFixedSize(size);
}

最佳答案

要获得您想要的行为,您必须设置大小,为此我创建了以下类,它继承自 QScrollArea 并且内部有一个 QGridLayout。对于此类,您必须确定可见行数和列数以及小部件的固定大小,在您的情况下为 QLabel

#ifndef HORIZONTALSCROLLAREA_H
#define HORIZONTALSCROLLAREA_H

#include <QGridLayout>
#include <QResizeEvent>
#include <QScrollArea>
#include <QScrollBar>

class HorizontalScrollArea : public QScrollArea
{
QWidget *contentWidget;
QGridLayout *grid;
int nRows;
int nColumns;
public:
HorizontalScrollArea(int rows, int cols, QWidget *parent = Q_NULLPTR)
:QScrollArea(parent), nRows(rows), nColumns(cols)
{
setWidgetResizable(true);
contentWidget = new QWidget(this);
setWidget(contentWidget);
grid = new QGridLayout(contentWidget);
setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
}

void addWidget(QWidget *w, int row, int col){
grid->addWidget(w, row, col);
adaptSize();
}

int columnCount() const{
if(grid->count() == 0){
return 0;
}
return grid->columnCount();
}

private:
void adaptSize(){
if(columnCount() >= nColumns ){
int w = 1.0*(width() - grid->horizontalSpacing()*(nColumns+1.6))/nColumns;
int wCorrected = w*columnCount() + grid->horizontalSpacing()*(columnCount()+2);
contentWidget->setFixedWidth(wCorrected);
}
contentWidget->setFixedHeight(viewport()->height());
}
protected:
void resizeEvent(QResizeEvent *event){
QScrollArea::resizeEvent(event);
adaptSize();
}
};

#endif // HORIZONTALSCROLLAREA_H

以下部分为示例:

int main(int argc, char *argv[])
{
QApplication a(argc, argv);
QWidget w;

w.setLayout(new QVBoxLayout);

QPushButton btn("Add", &w);

int nrows = 2;
int ncols = 2;

HorizontalScrollArea scroll(nrows, ncols);

w.layout()->addWidget(&btn);
w.layout()->addWidget(&scroll);

QObject::connect(&btn, &QPushButton::clicked, [&scroll, &nrows](){
int column = scroll.columnCount();
for(int row=0; row < nrows; row++){
QLabel *label = new QLabel(QString("label: %1 %2")
.arg(row)
.arg(column));
label->setFrameShape(QFrame::Box);
label->setAlignment(Qt::AlignCenter);
QColor color(qrand() % 256, qrand() % 256, qrand() % 256);
label->setStyleSheet(QString("QLabel { background-color : %1;}")
.arg(color.name()));
scroll.addWidget(label, row, column);
}
});
w.show();
return a.exec();
}

在下面link是完整的例子。

输出:

enter image description here

关于c++ - 带有动态添加的小部件的 QScrollArea,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46547089/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com