mirror of
https://github.com/crystalidea/qt6windows7.git
synced 2025-01-24 21:05:05 +08:00
1478 lines
44 KiB
C++
1478 lines
44 KiB
C++
|
// Copyright (C) 2016 The Qt Company Ltd.
|
||
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
||
|
|
||
|
|
||
|
#include <QTest>
|
||
|
#include <qlayout.h>
|
||
|
#include <qapplication.h>
|
||
|
#include <qwidget.h>
|
||
|
#include <qproxystyle.h>
|
||
|
#include <qsizepolicy.h>
|
||
|
|
||
|
#include <QtWidgets/QCheckBox>
|
||
|
#include <QtWidgets/QLabel>
|
||
|
#include <QtWidgets/QLineEdit>
|
||
|
#include <QtWidgets/QPushButton>
|
||
|
|
||
|
#include <private/qdialog_p.h>
|
||
|
|
||
|
#include <QStyleFactory>
|
||
|
#include <QSharedPointer>
|
||
|
|
||
|
#include <QtTest/private/qtesthelpers_p.h>
|
||
|
|
||
|
using namespace QTestPrivate;
|
||
|
|
||
|
#include <qformlayout.h>
|
||
|
|
||
|
// ItemRole has enumerators for numerical values 0..2, thus the only
|
||
|
// valid numerical values for storing into an ItemRole variable are 0..3:
|
||
|
constexpr QFormLayout::ItemRole invalidRole = QFormLayout::ItemRole(3);
|
||
|
|
||
|
struct QFormLayoutTakeRowResultHolder {
|
||
|
QFormLayoutTakeRowResultHolder(QFormLayout::TakeRowResult result) noexcept
|
||
|
: labelItem(result.labelItem),
|
||
|
fieldItem(result.fieldItem)
|
||
|
{
|
||
|
}
|
||
|
~QFormLayoutTakeRowResultHolder()
|
||
|
{
|
||
|
// re-use a QFormLayout to recursively reap the QLayoutItems:
|
||
|
QFormLayout disposer;
|
||
|
if (labelItem)
|
||
|
disposer.setItem(0, QFormLayout::LabelRole, labelItem);
|
||
|
if (fieldItem)
|
||
|
disposer.setItem(0, QFormLayout::FieldRole, fieldItem);
|
||
|
}
|
||
|
QFormLayoutTakeRowResultHolder(QFormLayoutTakeRowResultHolder &&other) noexcept
|
||
|
: labelItem(other.labelItem),
|
||
|
fieldItem(other.fieldItem)
|
||
|
{
|
||
|
other.labelItem = nullptr;
|
||
|
other.fieldItem = nullptr;
|
||
|
}
|
||
|
QFormLayoutTakeRowResultHolder &operator=(QFormLayoutTakeRowResultHolder &&other) noexcept
|
||
|
{
|
||
|
swap(other);
|
||
|
return *this;
|
||
|
}
|
||
|
|
||
|
void swap(QFormLayoutTakeRowResultHolder &other) noexcept
|
||
|
{
|
||
|
qSwap(labelItem, other.labelItem);
|
||
|
qSwap(fieldItem, other.fieldItem);
|
||
|
}
|
||
|
|
||
|
QLayoutItem *labelItem;
|
||
|
QLayoutItem *fieldItem;
|
||
|
};
|
||
|
|
||
|
class tst_QFormLayout : public QObject
|
||
|
{
|
||
|
Q_OBJECT
|
||
|
|
||
|
private slots:
|
||
|
void cleanup();
|
||
|
void rowCount();
|
||
|
#if QT_CONFIG(shortcut)
|
||
|
void buddies();
|
||
|
#endif
|
||
|
void getItemPosition();
|
||
|
void wrapping();
|
||
|
void spacing();
|
||
|
void contentsRect();
|
||
|
|
||
|
void setFormStyle();
|
||
|
void setFieldGrowthPolicy();
|
||
|
void setRowWrapPolicy();
|
||
|
void setLabelAlignment();
|
||
|
void setFormAlignment();
|
||
|
|
||
|
/*
|
||
|
void setHorizontalSpacing(int spacing);
|
||
|
int horizontalSpacing() const;
|
||
|
void setVerticalSpacing(int spacing);
|
||
|
int verticalSpacing() const;
|
||
|
*/
|
||
|
|
||
|
void addRow();
|
||
|
void insertRow_QWidget_QWidget();
|
||
|
void insertRow_QWidget_QLayout();
|
||
|
void insertRow_QString_QWidget();
|
||
|
void insertRow_QString_QLayout();
|
||
|
void insertRow_QWidget();
|
||
|
void insertRow_QLayout();
|
||
|
void removeRow();
|
||
|
void removeRow_QWidget();
|
||
|
void removeRow_QLayout();
|
||
|
void takeRow();
|
||
|
void takeRow_QWidget();
|
||
|
void takeRow_QLayout();
|
||
|
void setWidget();
|
||
|
void setLayout();
|
||
|
void hideShowRow();
|
||
|
void showWithHiddenRow();
|
||
|
void hiddenRowAndStretch();
|
||
|
|
||
|
/*
|
||
|
QLayoutItem *itemAt(int row, ItemRole role) const;
|
||
|
void getItemPosition(int index, int *rowPtr, ItemRole *rolePtr) const;
|
||
|
void getLayoutPosition(QWidget *widget, int *rowPtr, ItemRole *rolePtr) const;
|
||
|
void getItemPosition(QLayoutItem *item, int *rowPtr, ItemRole *rolePtr) const;
|
||
|
QWidget *labelForField(QWidget *widget) const;
|
||
|
QWidget *labelForField(QLayoutItem *item) const;
|
||
|
|
||
|
void addItem(QLayoutItem *item);
|
||
|
*/
|
||
|
|
||
|
void itemAt();
|
||
|
void takeAt();
|
||
|
void layoutAlone();
|
||
|
void replaceWidget();
|
||
|
/*
|
||
|
void setGeometry(const QRect &rect);
|
||
|
QSize minimumSize() const;
|
||
|
QSize sizeHint() const;
|
||
|
|
||
|
bool hasHeightForWidth() const;
|
||
|
int heightForWidth(int width) const;
|
||
|
Qt::Orientations expandingDirections() const;
|
||
|
*/
|
||
|
|
||
|
void taskQTBUG_27420_takeAtShouldUnparentLayout();
|
||
|
void taskQTBUG_40609_addingWidgetToItsOwnLayout();
|
||
|
void taskQTBUG_40609_addingLayoutToItself();
|
||
|
|
||
|
};
|
||
|
|
||
|
void tst_QFormLayout::cleanup()
|
||
|
{
|
||
|
QTRY_VERIFY(QApplication::topLevelWidgets().isEmpty());
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::rowCount()
|
||
|
{
|
||
|
QWidget w;
|
||
|
QFormLayout *fl = new QFormLayout(&w);
|
||
|
|
||
|
fl->addRow(tr("Label 1"), new QLineEdit);
|
||
|
fl->addRow(tr("Label 2"), new QLineEdit);
|
||
|
fl->addRow(tr("Label 3"), new QLineEdit);
|
||
|
QCOMPARE(fl->rowCount(), 3);
|
||
|
|
||
|
fl->addRow(new QWidget);
|
||
|
fl->addRow(new QHBoxLayout);
|
||
|
QCOMPARE(fl->rowCount(), 5);
|
||
|
|
||
|
fl->insertRow(1, tr("Label 0.5"), new QLineEdit);
|
||
|
QCOMPARE(fl->rowCount(), 6);
|
||
|
|
||
|
//TODO: remove items
|
||
|
}
|
||
|
|
||
|
#if QT_CONFIG(shortcut)
|
||
|
|
||
|
void tst_QFormLayout::buddies()
|
||
|
{
|
||
|
QWidget w;
|
||
|
QFormLayout *fl = new QFormLayout(&w);
|
||
|
|
||
|
//normal buddy case
|
||
|
QLineEdit *le = new QLineEdit;
|
||
|
fl->addRow(tr("Label 1"), le);
|
||
|
QLabel *label = qobject_cast<QLabel *>(fl->labelForField(le));
|
||
|
QVERIFY(label);
|
||
|
QWidget *lew = le;
|
||
|
QCOMPARE(label->buddy(), lew);
|
||
|
|
||
|
//null label
|
||
|
QLineEdit *le2 = new QLineEdit;
|
||
|
fl->addRow(0, le2);
|
||
|
QWidget *label2 = fl->labelForField(le2);
|
||
|
QVERIFY(!label2);
|
||
|
|
||
|
//no label
|
||
|
QLineEdit *le3 = new QLineEdit;
|
||
|
fl->addRow(le3);
|
||
|
QWidget *label3 = fl->labelForField(le3);
|
||
|
QVERIFY(!label3);
|
||
|
|
||
|
//TODO: empty label?
|
||
|
}
|
||
|
|
||
|
#endif // QT_CONFIG(shortcut)
|
||
|
|
||
|
void tst_QFormLayout::getItemPosition()
|
||
|
{
|
||
|
QWidget w;
|
||
|
QFormLayout *fl = new QFormLayout(&w);
|
||
|
|
||
|
QList<QLabel*> labels;
|
||
|
QList<QLineEdit*> fields;
|
||
|
for (int i = 0; i < 5; ++i) {
|
||
|
labels.append(new QLabel(QLatin1String("Label " ) + QString::number(i + 1)));
|
||
|
fields.append(new QLineEdit);
|
||
|
fl->addRow(labels[i], fields[i]);
|
||
|
}
|
||
|
|
||
|
//a field
|
||
|
{
|
||
|
int row;
|
||
|
QFormLayout::ItemRole role;
|
||
|
fl->getWidgetPosition(fields[3], &row, &role);
|
||
|
QCOMPARE(row, 3);
|
||
|
QCOMPARE(role, QFormLayout::FieldRole);
|
||
|
}
|
||
|
|
||
|
//a label
|
||
|
{
|
||
|
int row;
|
||
|
QFormLayout::ItemRole role;
|
||
|
fl->getWidgetPosition(labels[2], &row, &role);
|
||
|
QCOMPARE(row, 2);
|
||
|
QCOMPARE(role, QFormLayout::LabelRole);
|
||
|
}
|
||
|
|
||
|
//a layout that's been inserted
|
||
|
{
|
||
|
QVBoxLayout *vbl = new QVBoxLayout;
|
||
|
fl->insertRow(2, "Label 1.5", vbl);
|
||
|
int row;
|
||
|
QFormLayout::ItemRole role;
|
||
|
fl->getLayoutPosition(vbl, &row, &role);
|
||
|
QCOMPARE(row, 2);
|
||
|
QCOMPARE(role, QFormLayout::FieldRole);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::wrapping()
|
||
|
{
|
||
|
QWidget w;
|
||
|
QFormLayout *fl = new QFormLayout(&w);
|
||
|
fl->setRowWrapPolicy(QFormLayout::WrapLongRows);
|
||
|
|
||
|
QLineEdit *le = new QLineEdit;
|
||
|
QLabel *lbl = new QLabel("A long label which is actually long enough to trigger wrapping,"
|
||
|
" even on Android and even if it is executed on a tiling window"
|
||
|
" manager which forces the window into fullscreen mode.");
|
||
|
le->setMinimumWidth(200);
|
||
|
fl->addRow(lbl, le);
|
||
|
|
||
|
w.setFixedWidth(240);
|
||
|
w.setWindowTitle(QTest::currentTestFunction());
|
||
|
w.show();
|
||
|
|
||
|
QCOMPARE(le->geometry().y() > lbl->geometry().y(), true);
|
||
|
|
||
|
//TODO: additional tests covering different wrapping cases
|
||
|
}
|
||
|
|
||
|
class CustomLayoutStyle : public QProxyStyle
|
||
|
{
|
||
|
Q_OBJECT
|
||
|
public:
|
||
|
CustomLayoutStyle() : QProxyStyle(QStyleFactory::create("windows"))
|
||
|
{
|
||
|
hspacing = 5;
|
||
|
vspacing = 10;
|
||
|
}
|
||
|
|
||
|
virtual int pixelMetric(PixelMetric metric, const QStyleOption * option = nullptr,
|
||
|
const QWidget * widget = nullptr) const override;
|
||
|
|
||
|
int hspacing;
|
||
|
int vspacing;
|
||
|
};
|
||
|
|
||
|
int CustomLayoutStyle::pixelMetric(PixelMetric metric, const QStyleOption * option /*= nullptr*/,
|
||
|
const QWidget * widget /*= nullptr*/ ) const
|
||
|
{
|
||
|
switch (metric) {
|
||
|
case PM_LayoutHorizontalSpacing:
|
||
|
return hspacing;
|
||
|
case PM_LayoutVerticalSpacing:
|
||
|
return vspacing;
|
||
|
break;
|
||
|
default:
|
||
|
break;
|
||
|
}
|
||
|
return QProxyStyle::pixelMetric(metric, option, widget);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::spacing()
|
||
|
{
|
||
|
//TODO: confirm spacing behavior
|
||
|
QWidget w;
|
||
|
QScopedPointer<CustomLayoutStyle> style(new CustomLayoutStyle);
|
||
|
style->hspacing = 5;
|
||
|
style->vspacing = 10;
|
||
|
w.setStyle(style.data());
|
||
|
QFormLayout *fl = new QFormLayout(&w);
|
||
|
QCOMPARE(style->hspacing, fl->horizontalSpacing());
|
||
|
QCOMPARE(style->vspacing, fl->verticalSpacing());
|
||
|
|
||
|
//QCOMPARE(fl->spacing(), -1);
|
||
|
fl->setVerticalSpacing(5);
|
||
|
QCOMPARE(5, fl->horizontalSpacing());
|
||
|
QCOMPARE(5, fl->verticalSpacing());
|
||
|
//QCOMPARE(fl->spacing(), 5);
|
||
|
fl->setVerticalSpacing(-1);
|
||
|
QCOMPARE(style->hspacing, fl->horizontalSpacing());
|
||
|
QCOMPARE(style->vspacing, fl->verticalSpacing());
|
||
|
|
||
|
style->hspacing = 5;
|
||
|
style->vspacing = 5;
|
||
|
//QCOMPARE(fl->spacing(), 5);
|
||
|
|
||
|
fl->setHorizontalSpacing(20);
|
||
|
//QCOMPARE(fl->spacing(), -1);
|
||
|
style->vspacing = 20;
|
||
|
QCOMPARE(fl->horizontalSpacing(), 20);
|
||
|
QCOMPARE(fl->verticalSpacing(), 20);
|
||
|
//QCOMPARE(fl->spacing(), 20);
|
||
|
fl->setHorizontalSpacing(-1);
|
||
|
//QCOMPARE(fl->spacing(), -1);
|
||
|
style->hspacing = 20;
|
||
|
//QCOMPARE(fl->spacing(), 20);
|
||
|
|
||
|
|
||
|
|
||
|
// Do not assert if spacings are negative (QTBUG-34731)
|
||
|
style->vspacing = -1;
|
||
|
style->hspacing = -1;
|
||
|
QLabel *label = new QLabel(tr("Asserts"));
|
||
|
QCheckBox *checkBox = new QCheckBox(tr("Yes"));
|
||
|
fl->setWidget(0, QFormLayout::LabelRole, label);
|
||
|
fl->setWidget(1, QFormLayout::FieldRole, checkBox);
|
||
|
w.resize(200, 100);
|
||
|
w.setWindowTitle(QTest::currentTestFunction());
|
||
|
w.show();
|
||
|
QVERIFY(QTest::qWaitForWindowExposed(&w));
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::contentsRect()
|
||
|
{
|
||
|
QWidget w;
|
||
|
setFrameless(&w);
|
||
|
QFormLayout form;
|
||
|
w.setLayout(&form);
|
||
|
form.addRow("Label", new QPushButton(&w));
|
||
|
w.setWindowTitle(QTest::currentTestFunction());
|
||
|
w.show();
|
||
|
QVERIFY(QTest::qWaitForWindowExposed(&w));
|
||
|
int l, t, r, b;
|
||
|
form.getContentsMargins(&l, &t, &r, &b);
|
||
|
QRect geom = form.geometry();
|
||
|
|
||
|
QCOMPARE(geom.adjusted(+l, +t, -r, -b), form.contentsRect());
|
||
|
}
|
||
|
|
||
|
|
||
|
class DummyMacStyle : public QCommonStyle
|
||
|
{
|
||
|
public:
|
||
|
virtual int styleHint ( StyleHint hint, const QStyleOption * option = 0, const QWidget * widget = nullptr, QStyleHintReturn * returnData = 0 ) const override
|
||
|
{
|
||
|
switch(hint) {
|
||
|
case SH_FormLayoutFormAlignment:
|
||
|
return Qt::AlignHCenter | Qt::AlignTop;
|
||
|
case SH_FormLayoutLabelAlignment:
|
||
|
return Qt::AlignRight;
|
||
|
case SH_FormLayoutWrapPolicy:
|
||
|
return QFormLayout::DontWrapRows;
|
||
|
case SH_FormLayoutFieldGrowthPolicy:
|
||
|
return QFormLayout::FieldsStayAtSizeHint;
|
||
|
default:
|
||
|
return QCommonStyle::styleHint(hint, option, widget, returnData);
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
class DummyQtopiaStyle : public QCommonStyle
|
||
|
{
|
||
|
public:
|
||
|
virtual int styleHint ( StyleHint hint, const QStyleOption * option = 0, const QWidget * widget = nullptr, QStyleHintReturn * returnData = 0 ) const override
|
||
|
{
|
||
|
switch(hint) {
|
||
|
case SH_FormLayoutFormAlignment:
|
||
|
return Qt::AlignLeft | Qt::AlignTop;
|
||
|
case SH_FormLayoutLabelAlignment:
|
||
|
return Qt::AlignRight;
|
||
|
case SH_FormLayoutWrapPolicy:
|
||
|
return QFormLayout::WrapLongRows;
|
||
|
case SH_FormLayoutFieldGrowthPolicy:
|
||
|
return QFormLayout::AllNonFixedFieldsGrow;
|
||
|
default:
|
||
|
return QCommonStyle::styleHint(hint, option, widget, returnData);
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
void tst_QFormLayout::setFormStyle()
|
||
|
{
|
||
|
QWidget widget;
|
||
|
QFormLayout layout;
|
||
|
widget.setLayout(&layout);
|
||
|
|
||
|
#if 0 // QT_NO_STYLE_PLASTIQUE
|
||
|
widget.setStyle(new QPlastiqueStyle());
|
||
|
|
||
|
QCOMPARE(layout.labelAlignment(), Qt::AlignRight);
|
||
|
QVERIFY(layout.formAlignment() == (Qt::AlignLeft | Qt::AlignTop));
|
||
|
QCOMPARE(layout.fieldGrowthPolicy(), QFormLayout::ExpandingFieldsGrow);
|
||
|
QCOMPARE(layout.rowWrapPolicy(), QFormLayout::DontWrapRows);
|
||
|
#endif
|
||
|
|
||
|
const QScopedPointer<QStyle> windowsStyle(QStyleFactory::create("windows"));
|
||
|
widget.setStyle(windowsStyle.data());
|
||
|
|
||
|
QCOMPARE(layout.labelAlignment(), Qt::AlignLeft);
|
||
|
QVERIFY(layout.formAlignment() == (Qt::AlignLeft | Qt::AlignTop));
|
||
|
QCOMPARE(layout.fieldGrowthPolicy(), QFormLayout::AllNonFixedFieldsGrow);
|
||
|
QCOMPARE(layout.rowWrapPolicy(), QFormLayout::DontWrapRows);
|
||
|
|
||
|
/* can't directly create mac style or qtopia style, since
|
||
|
this test is cross platform.. so create dummy styles that
|
||
|
return all the right stylehints.
|
||
|
*/
|
||
|
DummyMacStyle macStyle;
|
||
|
widget.setStyle(&macStyle);
|
||
|
|
||
|
QCOMPARE(layout.labelAlignment(), Qt::AlignRight);
|
||
|
QVERIFY(layout.formAlignment() == (Qt::AlignHCenter | Qt::AlignTop));
|
||
|
QCOMPARE(layout.fieldGrowthPolicy(), QFormLayout::FieldsStayAtSizeHint);
|
||
|
QCOMPARE(layout.rowWrapPolicy(), QFormLayout::DontWrapRows);
|
||
|
|
||
|
DummyQtopiaStyle qtopiaStyle;
|
||
|
widget.setStyle(&qtopiaStyle);
|
||
|
|
||
|
QCOMPARE(layout.labelAlignment(), Qt::AlignRight);
|
||
|
QVERIFY(layout.formAlignment() == (Qt::AlignLeft | Qt::AlignTop));
|
||
|
QCOMPARE(layout.fieldGrowthPolicy(), QFormLayout::AllNonFixedFieldsGrow);
|
||
|
QCOMPARE(layout.rowWrapPolicy(), QFormLayout::WrapLongRows);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::setFieldGrowthPolicy()
|
||
|
{
|
||
|
QWidget window;
|
||
|
QLineEdit fld1, fld2, fld3;
|
||
|
fld1.setSizePolicy(QSizePolicy::Fixed, QSizePolicy::Fixed);
|
||
|
fld2.setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Fixed);
|
||
|
fld3.setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Fixed);
|
||
|
|
||
|
QFormLayout layout;
|
||
|
layout.addRow("One:", &fld1);
|
||
|
layout.addRow("Two:", &fld2);
|
||
|
layout.addRow("Three:", &fld3);
|
||
|
window.setLayout(&layout);
|
||
|
window.resize(1000, 200);
|
||
|
|
||
|
for (int i = 0; i < 3; ++i) {
|
||
|
layout.setFieldGrowthPolicy(i == 0 ? QFormLayout::FieldsStayAtSizeHint :
|
||
|
i == 1 ? QFormLayout::ExpandingFieldsGrow :
|
||
|
QFormLayout::AllNonFixedFieldsGrow);
|
||
|
layout.activate();
|
||
|
|
||
|
if (i == 0) {
|
||
|
QCOMPARE(fld1.width(), fld2.width());
|
||
|
QCOMPARE(fld2.width(), fld3.width());
|
||
|
} else if (i == 1) {
|
||
|
QCOMPARE(fld1.width(), fld2.width());
|
||
|
QVERIFY(fld2.width() < fld3.width());
|
||
|
} else {
|
||
|
QVERIFY(fld1.width() < fld2.width());
|
||
|
QCOMPARE(fld2.width(), fld3.width());
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::setRowWrapPolicy()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::setLabelAlignment()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::setFormAlignment()
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::addRow()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
QWidget *w1 = new QWidget(&topLevel);
|
||
|
QWidget *w2 = new QWidget(&topLevel);
|
||
|
QWidget *w3 = new QWidget(&topLevel);
|
||
|
QHBoxLayout *l1 = new QHBoxLayout;
|
||
|
QHBoxLayout *l2 = new QHBoxLayout;
|
||
|
QHBoxLayout *l3 = new QHBoxLayout;
|
||
|
QLabel *lbl1 = new QLabel(&topLevel);
|
||
|
QLabel *lbl2 = new QLabel(&topLevel);
|
||
|
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
layout->addRow(lbl1, w1);
|
||
|
layout->addRow(lbl2, l1);
|
||
|
layout->addRow("Foo:", w2);
|
||
|
layout->addRow("Bar:", l2);
|
||
|
layout->addRow(w3);
|
||
|
layout->addRow(l3);
|
||
|
|
||
|
QCOMPARE(layout->rowCount(), 6);
|
||
|
|
||
|
QVERIFY(layout->itemAt(0, QFormLayout::LabelRole)->widget() == lbl1);
|
||
|
QVERIFY(layout->itemAt(1, QFormLayout::LabelRole)->widget() == lbl2);
|
||
|
QCOMPARE(layout->itemAt(2, QFormLayout::LabelRole)->widget()->property("text").toString(), QLatin1String("Foo:"));
|
||
|
QCOMPARE(layout->itemAt(3, QFormLayout::LabelRole)->widget()->property("text").toString(), QLatin1String("Bar:"));
|
||
|
QVERIFY(layout->itemAt(4, QFormLayout::LabelRole) == 0);
|
||
|
QVERIFY(layout->itemAt(5, QFormLayout::LabelRole) == 0);
|
||
|
|
||
|
QVERIFY(layout->itemAt(0, QFormLayout::FieldRole)->widget() == w1);
|
||
|
QVERIFY(layout->itemAt(1, QFormLayout::FieldRole)->layout() == l1);
|
||
|
QVERIFY(layout->itemAt(2, QFormLayout::FieldRole)->widget() == w2);
|
||
|
QVERIFY(layout->itemAt(3, QFormLayout::FieldRole)->layout() == l2);
|
||
|
// ### should have a third role, FullRowRole?
|
||
|
// QVERIFY(layout.itemAt(4, QFormLayout::FieldRole) == 0);
|
||
|
// QVERIFY(layout.itemAt(5, QFormLayout::FieldRole) == 0);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::insertRow_QWidget_QWidget()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
QLabel *lbl1 = new QLabel(&topLevel);
|
||
|
QLabel *lbl2 = new QLabel(&topLevel);
|
||
|
QLabel *lbl3 = new QLabel(&topLevel);
|
||
|
QLabel *lbl4 = new QLabel(&topLevel);
|
||
|
QLineEdit *fld1 = new QLineEdit(&topLevel);
|
||
|
QLineEdit *fld2 = new QLineEdit(&topLevel);
|
||
|
QLineEdit *fld3 = new QLineEdit(&topLevel);
|
||
|
QLineEdit *fld4 = new QLineEdit(&topLevel);
|
||
|
|
||
|
layout->insertRow(0, lbl1, fld1);
|
||
|
QCOMPARE(layout->rowCount(), 1);
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout->getWidgetPosition(lbl1, &row, &role);
|
||
|
QCOMPARE(row, 0);
|
||
|
QCOMPARE(int(role), int(QFormLayout::LabelRole));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout->getWidgetPosition(fld1, &row, &role);
|
||
|
QCOMPARE(row, 0);
|
||
|
QCOMPARE(int(role), int(QFormLayout::FieldRole));
|
||
|
}
|
||
|
|
||
|
// check that negative values append
|
||
|
layout->insertRow(-2, lbl2, fld2);
|
||
|
QCOMPARE(layout->rowCount(), 2);
|
||
|
|
||
|
QVERIFY(layout->itemAt(0, QFormLayout::LabelRole)->widget() == lbl1);
|
||
|
QVERIFY(layout->itemAt(1, QFormLayout::LabelRole)->widget() == lbl2);
|
||
|
|
||
|
// check that too large values append
|
||
|
layout->insertRow(100, lbl3, fld3);
|
||
|
QCOMPARE(layout->rowCount(), 3);
|
||
|
QCOMPARE(layout->count(), 6);
|
||
|
|
||
|
layout->insertRow(3, (QWidget *)0, (QWidget *)0);
|
||
|
QCOMPARE(layout->rowCount(), 4);
|
||
|
QCOMPARE(layout->count(), 6);
|
||
|
|
||
|
layout->insertRow(4, (QWidget *)0, fld4);
|
||
|
QCOMPARE(layout->rowCount(), 5);
|
||
|
QCOMPARE(layout->count(), 7);
|
||
|
|
||
|
layout->insertRow(5, lbl4, (QWidget *)0);
|
||
|
QCOMPARE(layout->rowCount(), 6);
|
||
|
QCOMPARE(layout->count(), 8);
|
||
|
|
||
|
QVERIFY(layout->itemAt(0, QFormLayout::LabelRole)->widget() == lbl1);
|
||
|
QVERIFY(layout->itemAt(1, QFormLayout::LabelRole)->widget() == lbl2);
|
||
|
QVERIFY(layout->itemAt(2, QFormLayout::LabelRole)->widget() == lbl3);
|
||
|
QVERIFY(layout->itemAt(3, QFormLayout::LabelRole) == 0);
|
||
|
QVERIFY(layout->itemAt(4, QFormLayout::LabelRole) == 0);
|
||
|
QVERIFY(layout->itemAt(5, QFormLayout::LabelRole)->widget() == lbl4);
|
||
|
|
||
|
QVERIFY(layout->itemAt(0, QFormLayout::FieldRole)->widget() == fld1);
|
||
|
QVERIFY(layout->itemAt(1, QFormLayout::FieldRole)->widget() == fld2);
|
||
|
QVERIFY(layout->itemAt(2, QFormLayout::FieldRole)->widget() == fld3);
|
||
|
QVERIFY(layout->itemAt(3, QFormLayout::FieldRole) == 0);
|
||
|
QVERIFY(layout->itemAt(4, QFormLayout::FieldRole)->widget() == fld4);
|
||
|
QVERIFY(layout->itemAt(5, QFormLayout::FieldRole) == 0);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::insertRow_QWidget_QLayout()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
QLabel *lbl1 = new QLabel(&topLevel);
|
||
|
QLabel *lbl2 = new QLabel(&topLevel);
|
||
|
QLabel *lbl3 = new QLabel(&topLevel);
|
||
|
QHBoxLayout *fld1 = new QHBoxLayout;
|
||
|
QHBoxLayout *fld2 = new QHBoxLayout;
|
||
|
QHBoxLayout *fld3 = new QHBoxLayout;
|
||
|
|
||
|
layout->insertRow(0, lbl1, fld1);
|
||
|
QCOMPARE(layout->rowCount(), 1);
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout->getWidgetPosition(lbl1, &row, &role);
|
||
|
QCOMPARE(row, 0);
|
||
|
QCOMPARE(int(role), int(QFormLayout::LabelRole));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout->getLayoutPosition(fld1, &row, &role);
|
||
|
QCOMPARE(row, 0);
|
||
|
QCOMPARE(int(role), int(QFormLayout::FieldRole));
|
||
|
}
|
||
|
|
||
|
// check that negative values append
|
||
|
layout->insertRow(-2, lbl2, fld2);
|
||
|
QCOMPARE(layout->rowCount(), 2);
|
||
|
|
||
|
QVERIFY(layout->itemAt(0, QFormLayout::LabelRole)->widget() == lbl1);
|
||
|
QVERIFY(layout->itemAt(1, QFormLayout::LabelRole)->widget() == lbl2);
|
||
|
|
||
|
// check that too large values append
|
||
|
layout->insertRow(100, lbl3, fld3);
|
||
|
QCOMPARE(layout->rowCount(), 3);
|
||
|
|
||
|
QVERIFY(layout->itemAt(0, QFormLayout::LabelRole)->widget() == lbl1);
|
||
|
QVERIFY(layout->itemAt(1, QFormLayout::LabelRole)->widget() == lbl2);
|
||
|
QVERIFY(layout->itemAt(2, QFormLayout::LabelRole)->widget() == lbl3);
|
||
|
|
||
|
QVERIFY(layout->itemAt(0, QFormLayout::FieldRole)->layout() == fld1);
|
||
|
QVERIFY(layout->itemAt(1, QFormLayout::FieldRole)->layout() == fld2);
|
||
|
QVERIFY(layout->itemAt(2, QFormLayout::FieldRole)->layout() == fld3);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::insertRow_QString_QWidget()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
QLineEdit *fld1 = new QLineEdit(&topLevel);
|
||
|
QLineEdit *fld2 = new QLineEdit(&topLevel);
|
||
|
QLineEdit *fld3 = new QLineEdit(&topLevel);
|
||
|
|
||
|
layout->insertRow(-5, "&Name:", fld1);
|
||
|
QLabel *label1 = qobject_cast<QLabel *>(layout->itemAt(0, QFormLayout::LabelRole)->widget());
|
||
|
QVERIFY(label1 != 0);
|
||
|
#if QT_CONFIG(shortcut)
|
||
|
QCOMPARE(label1->buddy(), fld1);
|
||
|
#endif
|
||
|
layout->insertRow(0, "&Email:", fld2);
|
||
|
QLabel *label2 = qobject_cast<QLabel *>(layout->itemAt(0, QFormLayout::LabelRole)->widget());
|
||
|
QVERIFY(label2 != 0);
|
||
|
#if QT_CONFIG(shortcut)
|
||
|
QCOMPARE(label2->buddy(), fld2);
|
||
|
#endif
|
||
|
layout->insertRow(5, "&Age:", fld3);
|
||
|
QLabel *label3 = qobject_cast<QLabel *>(layout->itemAt(2, QFormLayout::LabelRole)->widget());
|
||
|
QVERIFY(label3 != 0);
|
||
|
#if QT_CONFIG(shortcut)
|
||
|
QCOMPARE(label3->buddy(), fld3);
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::insertRow_QString_QLayout()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
QHBoxLayout *fld1 = new QHBoxLayout;
|
||
|
QHBoxLayout *fld2 = new QHBoxLayout;
|
||
|
QHBoxLayout *fld3 = new QHBoxLayout;
|
||
|
|
||
|
layout->insertRow(-5, "&Name:", fld1);
|
||
|
QLabel *label1 = qobject_cast<QLabel *>(layout->itemAt(0, QFormLayout::LabelRole)->widget());
|
||
|
QVERIFY(label1 != 0);
|
||
|
#if QT_CONFIG(shortcut)
|
||
|
QVERIFY(!label1->buddy());
|
||
|
#endif
|
||
|
|
||
|
QCOMPARE(layout->rowCount(), 1);
|
||
|
|
||
|
layout->insertRow(0, "&Email:", fld2);
|
||
|
QLabel *label2 = qobject_cast<QLabel *>(layout->itemAt(0, QFormLayout::LabelRole)->widget());
|
||
|
QVERIFY(label2 != 0);
|
||
|
#if QT_CONFIG(shortcut)
|
||
|
QVERIFY(!label2->buddy());
|
||
|
#endif
|
||
|
|
||
|
QCOMPARE(layout->rowCount(), 2);
|
||
|
|
||
|
layout->insertRow(5, "&Age:", fld3);
|
||
|
QLabel *label3 = qobject_cast<QLabel *>(layout->itemAt(2, QFormLayout::LabelRole)->widget());
|
||
|
QVERIFY(label3 != 0);
|
||
|
#if QT_CONFIG(shortcut)
|
||
|
QVERIFY(!label3->buddy());
|
||
|
#endif
|
||
|
|
||
|
QCOMPARE(layout->rowCount(), 3);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::insertRow_QWidget()
|
||
|
{
|
||
|
// ### come back to this later
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::insertRow_QLayout()
|
||
|
{
|
||
|
// ### come back to this later
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::removeRow()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
QPointer<QWidget> w1 = new QWidget;
|
||
|
QPointer<QWidget> w2 = new QWidget;
|
||
|
|
||
|
layout->addRow("test1", w1);
|
||
|
layout->addRow(w2);
|
||
|
|
||
|
QCOMPARE(layout->count(), 3);
|
||
|
QCOMPARE(layout->rowCount(), 2);
|
||
|
|
||
|
layout->removeRow(1);
|
||
|
|
||
|
QVERIFY(w1);
|
||
|
QVERIFY(!w2);
|
||
|
QCOMPARE(layout->count(), 2);
|
||
|
QCOMPARE(layout->rowCount(), 1);
|
||
|
|
||
|
layout->removeRow(0);
|
||
|
|
||
|
QVERIFY(!w1);
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::removeRow_QWidget()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
QPointer<QWidget> w1 = new QWidget;
|
||
|
QPointer<QWidget> w2 = new QWidget;
|
||
|
|
||
|
layout->addRow("test1", w1);
|
||
|
layout->addRow(w2);
|
||
|
|
||
|
QCOMPARE(layout->count(), 3);
|
||
|
QCOMPARE(layout->rowCount(), 2);
|
||
|
|
||
|
layout->removeRow(w1);
|
||
|
|
||
|
QVERIFY(!w1);
|
||
|
QCOMPARE(layout->count(), 1);
|
||
|
QCOMPARE(layout->rowCount(), 1);
|
||
|
|
||
|
layout->removeRow(w2);
|
||
|
|
||
|
QVERIFY(!w2);
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
QWidget *w3 = new QWidget;
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QFormLayout::takeRow: Invalid widget");
|
||
|
layout->removeRow(w3);
|
||
|
delete w3;
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::removeRow_QLayout()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
QPointer<QHBoxLayout> l1 = new QHBoxLayout;
|
||
|
QPointer<QWidget> w1 = new QWidget;
|
||
|
l1->addWidget(w1);
|
||
|
QPointer<QHBoxLayout> l2 = new QHBoxLayout;
|
||
|
QPointer<QWidget> w2 = new QWidget;
|
||
|
l2->addWidget(w2);
|
||
|
|
||
|
layout->addRow("test1", l1);
|
||
|
layout->addRow(l2);
|
||
|
|
||
|
QCOMPARE(layout->count(), 3);
|
||
|
QCOMPARE(layout->rowCount(), 2);
|
||
|
|
||
|
layout->removeRow(l1);
|
||
|
|
||
|
QVERIFY(!l1);
|
||
|
QVERIFY(!w1);
|
||
|
QCOMPARE(layout->count(), 1);
|
||
|
QCOMPARE(layout->rowCount(), 1);
|
||
|
|
||
|
layout->removeRow(l2);
|
||
|
|
||
|
QVERIFY(!l2);
|
||
|
QVERIFY(!w2);
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
QHBoxLayout *l3 = new QHBoxLayout;
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QFormLayout::takeRow: Invalid layout");
|
||
|
layout->removeRow(l3);
|
||
|
delete l3;
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::takeRow()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
QPointer<QWidget> w1 = new QWidget;
|
||
|
QPointer<QWidget> w2 = new QWidget;
|
||
|
|
||
|
layout->addRow("test1", w1);
|
||
|
layout->addRow(w2);
|
||
|
|
||
|
QCOMPARE(layout->count(), 3);
|
||
|
QCOMPARE(layout->rowCount(), 2);
|
||
|
|
||
|
QFormLayoutTakeRowResultHolder result = layout->takeRow(1);
|
||
|
|
||
|
QVERIFY(w2);
|
||
|
QVERIFY(result.fieldItem);
|
||
|
QVERIFY(!result.labelItem);
|
||
|
QCOMPARE(layout->count(), 2);
|
||
|
QCOMPARE(layout->rowCount(), 1);
|
||
|
QCOMPARE(result.fieldItem->widget(), w2.data());
|
||
|
|
||
|
result = layout->takeRow(0);
|
||
|
|
||
|
QVERIFY(w1);
|
||
|
QVERIFY(result.fieldItem);
|
||
|
QVERIFY(result.labelItem);
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
QCOMPARE(result.fieldItem->widget(), w1.data());
|
||
|
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QFormLayout::takeRow: Invalid row 0");
|
||
|
result = layout->takeRow(0);
|
||
|
|
||
|
QVERIFY(!result.fieldItem);
|
||
|
QVERIFY(!result.labelItem);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::takeRow_QWidget()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
QPointer<QWidget> w1 = new QWidget;
|
||
|
QPointer<QWidget> w2 = new QWidget;
|
||
|
|
||
|
layout->addRow("test1", w1);
|
||
|
layout->addRow(w2);
|
||
|
|
||
|
QCOMPARE(layout->count(), 3);
|
||
|
QCOMPARE(layout->rowCount(), 2);
|
||
|
|
||
|
QFormLayoutTakeRowResultHolder result = layout->takeRow(w1);
|
||
|
|
||
|
QVERIFY(w1);
|
||
|
QVERIFY(result.fieldItem);
|
||
|
QVERIFY(result.labelItem);
|
||
|
QCOMPARE(layout->count(), 1);
|
||
|
QCOMPARE(layout->rowCount(), 1);
|
||
|
|
||
|
result = layout->takeRow(w2);
|
||
|
|
||
|
QVERIFY(w2);
|
||
|
QVERIFY(result.fieldItem);
|
||
|
QVERIFY(!result.labelItem);
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
QWidget *w3 = new QWidget;
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QFormLayout::takeRow: Invalid widget");
|
||
|
result = layout->takeRow(w3);
|
||
|
delete w3;
|
||
|
|
||
|
QVERIFY(!result.fieldItem);
|
||
|
QVERIFY(!result.labelItem);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::takeRow_QLayout()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
QPointer<QHBoxLayout> l1 = new QHBoxLayout;
|
||
|
QPointer<QWidget> w1 = new QWidget;
|
||
|
l1->addWidget(w1);
|
||
|
QPointer<QHBoxLayout> l2 = new QHBoxLayout;
|
||
|
QPointer<QWidget> w2 = new QWidget;
|
||
|
l2->addWidget(w2);
|
||
|
|
||
|
layout->addRow("test1", l1);
|
||
|
layout->addRow(l2);
|
||
|
|
||
|
QCOMPARE(layout->count(), 3);
|
||
|
QCOMPARE(layout->rowCount(), 2);
|
||
|
|
||
|
QFormLayoutTakeRowResultHolder result = layout->takeRow(l1);
|
||
|
|
||
|
QVERIFY(l1);
|
||
|
QVERIFY(w1);
|
||
|
QVERIFY(result.fieldItem);
|
||
|
QVERIFY(result.labelItem);
|
||
|
QCOMPARE(layout->count(), 1);
|
||
|
QCOMPARE(layout->rowCount(), 1);
|
||
|
|
||
|
result = layout->takeRow(l2);
|
||
|
|
||
|
QVERIFY(l2);
|
||
|
QVERIFY(w2);
|
||
|
QVERIFY(result.fieldItem);
|
||
|
QVERIFY(!result.labelItem);
|
||
|
QCOMPARE(layout->count(), 0);
|
||
|
QCOMPARE(layout->rowCount(), 0);
|
||
|
|
||
|
QHBoxLayout *l3 = new QHBoxLayout;
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QFormLayout::takeRow: Invalid layout");
|
||
|
result = layout->takeRow(l3);
|
||
|
delete l3;
|
||
|
|
||
|
QVERIFY(!result.fieldItem);
|
||
|
QVERIFY(!result.labelItem);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::setWidget()
|
||
|
{
|
||
|
QFormLayout layout;
|
||
|
|
||
|
QWidget w1;
|
||
|
QWidget w2;
|
||
|
QWidget w3;
|
||
|
QWidget w4;
|
||
|
|
||
|
QCOMPARE(layout.count(), 0);
|
||
|
QCOMPARE(layout.rowCount(), 0);
|
||
|
|
||
|
layout.setWidget(5, QFormLayout::LabelRole, &w1);
|
||
|
QCOMPARE(layout.count(), 1);
|
||
|
QCOMPARE(layout.rowCount(), 6);
|
||
|
|
||
|
layout.setWidget(3, QFormLayout::FieldRole, &w2);
|
||
|
layout.setWidget(3, QFormLayout::LabelRole, &w3);
|
||
|
QCOMPARE(layout.count(), 3);
|
||
|
QCOMPARE(layout.rowCount(), 6);
|
||
|
|
||
|
// should be ignored and generate warnings
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QFormLayoutPrivate::setItem: Cell (3, 1) already occupied");
|
||
|
layout.setWidget(3, QFormLayout::FieldRole, &w4);
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QFormLayoutPrivate::setItem: Invalid cell (-1, 1)");
|
||
|
layout.setWidget(-1, QFormLayout::FieldRole, &w4);
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout.getWidgetPosition(&w1, &row, &role);
|
||
|
QCOMPARE(row, 5);
|
||
|
QCOMPARE(int(role), int(QFormLayout::LabelRole));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout.getWidgetPosition(&w2, &row, &role);
|
||
|
QCOMPARE(row, 3);
|
||
|
QCOMPARE(int(role), int(QFormLayout::FieldRole));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout.getWidgetPosition(&w3, &row, &role);
|
||
|
QCOMPARE(row, 3);
|
||
|
QCOMPARE(int(role), int(QFormLayout::LabelRole));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout.getWidgetPosition(&w4, &row, &role);
|
||
|
// not found
|
||
|
QCOMPARE(row, -1);
|
||
|
QCOMPARE(int(role), int(invalidRole));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout.getWidgetPosition(0, &row, &role);
|
||
|
// not found
|
||
|
QCOMPARE(row, -1);
|
||
|
QCOMPARE(int(role), int(invalidRole));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::setLayout()
|
||
|
{
|
||
|
QFormLayout layout;
|
||
|
|
||
|
QHBoxLayout l1;
|
||
|
QHBoxLayout l2;
|
||
|
QHBoxLayout l3;
|
||
|
QHBoxLayout l4;
|
||
|
|
||
|
QCOMPARE(layout.count(), 0);
|
||
|
QCOMPARE(layout.rowCount(), 0);
|
||
|
|
||
|
layout.setLayout(5, QFormLayout::LabelRole, &l1);
|
||
|
QCOMPARE(layout.count(), 1);
|
||
|
QCOMPARE(layout.rowCount(), 6);
|
||
|
|
||
|
layout.setLayout(3, QFormLayout::FieldRole, &l2);
|
||
|
layout.setLayout(3, QFormLayout::LabelRole, &l3);
|
||
|
QCOMPARE(layout.count(), 3);
|
||
|
QCOMPARE(layout.rowCount(), 6);
|
||
|
|
||
|
// should be ignored and generate warnings
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QFormLayoutPrivate::setItem: Cell (3, 1) already occupied");
|
||
|
layout.setLayout(3, QFormLayout::FieldRole, &l4);
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QLayout::addChildLayout: layout QHBoxLayout \"\" already has a parent");
|
||
|
layout.setLayout(-1, QFormLayout::FieldRole, &l4);
|
||
|
QCOMPARE(layout.count(), 3);
|
||
|
QCOMPARE(layout.rowCount(), 6);
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout.getLayoutPosition(&l1, &row, &role);
|
||
|
QCOMPARE(row, 5);
|
||
|
QCOMPARE(int(role), int(QFormLayout::LabelRole));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout.getLayoutPosition(&l2, &row, &role);
|
||
|
QCOMPARE(row, 3);
|
||
|
QCOMPARE(int(role), int(QFormLayout::FieldRole));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout.getLayoutPosition(&l3, &row, &role);
|
||
|
QCOMPARE(row, 3);
|
||
|
QCOMPARE(int(role), int(QFormLayout::LabelRole));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout.getLayoutPosition(&l4, &row, &role);
|
||
|
QCOMPARE(row, -1);
|
||
|
QCOMPARE(int(role), int(invalidRole));
|
||
|
}
|
||
|
|
||
|
{
|
||
|
int row = -1;
|
||
|
QFormLayout::ItemRole role = invalidRole;
|
||
|
layout.getLayoutPosition(0, &row, &role);
|
||
|
QCOMPARE(row, -1);
|
||
|
QCOMPARE(int(role), int(invalidRole));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::hideShowRow()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout layout;
|
||
|
|
||
|
const auto makeComplex = []{
|
||
|
QHBoxLayout *hboxField = new QHBoxLayout;
|
||
|
hboxField->addWidget(new QLineEdit("Left"));
|
||
|
hboxField->addWidget(new QLineEdit("Right"));
|
||
|
return hboxField;
|
||
|
};
|
||
|
|
||
|
layout.addRow("Label", new QLineEdit("one"));
|
||
|
layout.addRow("Label", new QLineEdit("two"));
|
||
|
layout.addRow("Label", new QLineEdit("three"));
|
||
|
layout.addRow("Label", makeComplex());
|
||
|
layout.addRow(new QLineEdit("five")); // spanning widget
|
||
|
layout.addRow(makeComplex()); // spanning layout
|
||
|
|
||
|
topLevel.setLayout(&layout);
|
||
|
topLevel.show();
|
||
|
QVERIFY(QTest::qWaitForWindowExposed(&topLevel));
|
||
|
|
||
|
// returns the top-left position of the items in a row
|
||
|
const auto rowPosition = [&layout](int row) {
|
||
|
QRect rect;
|
||
|
if (QLayoutItem *spanningItem = layout.itemAt(row, QFormLayout::SpanningRole)) {
|
||
|
rect = spanningItem->geometry();
|
||
|
} else {
|
||
|
if (QLayoutItem *labelItem = layout.itemAt(row, QFormLayout::LabelRole)) {
|
||
|
rect = labelItem->geometry();
|
||
|
}
|
||
|
if (QLayoutItem *fieldItem = layout.itemAt(row, QFormLayout::FieldRole)) {
|
||
|
rect |= fieldItem->geometry();
|
||
|
}
|
||
|
}
|
||
|
return rect.topLeft();
|
||
|
};
|
||
|
|
||
|
// returns the first widget in a row, even if that row is taken by a layout
|
||
|
const auto rowInputWidget = [&layout](int row) -> QWidget* {
|
||
|
auto fieldItem = layout.itemAt(row, QFormLayout::FieldRole);
|
||
|
if (!fieldItem)
|
||
|
return nullptr;
|
||
|
QWidget *fieldWidget = fieldItem->widget();
|
||
|
// we happen to know our layout structure
|
||
|
if (!fieldWidget)
|
||
|
fieldWidget = fieldItem->layout()->itemAt(0)->widget();
|
||
|
return fieldWidget;
|
||
|
};
|
||
|
|
||
|
// record the reference positions for all rows
|
||
|
QList<QPoint> rowPositions(layout.rowCount());
|
||
|
for (int row = 0; row < layout.rowCount(); ++row)
|
||
|
rowPositions[row] = rowPosition(row);
|
||
|
|
||
|
// hide each row in turn, the next row should take the space of the hidden row
|
||
|
for (int row = 0; row < layout.rowCount(); ++ row) {
|
||
|
layout.setRowVisible(row, false);
|
||
|
QVERIFY(!layout.isRowVisible(row));
|
||
|
if (row < layout.rowCount() - 1)
|
||
|
QTRY_COMPARE(rowPosition(row + 1), rowPositions[row]);
|
||
|
layout.setRowVisible(row, true);
|
||
|
QVERIFY(layout.isRowVisible(row));
|
||
|
}
|
||
|
|
||
|
// Hiding only the label or only the field doesn't hide the row.
|
||
|
for (int row = 0; row < layout.rowCount() - 1; ++row) {
|
||
|
const auto labelItem = layout.itemAt(0, QFormLayout::LabelRole);
|
||
|
if (labelItem) {
|
||
|
labelItem->widget()->hide();
|
||
|
QVERIFY(layout.isRowVisible(row));
|
||
|
QCOMPARE(rowPosition(row), rowPositions[row]);
|
||
|
layout.itemAt(0, QFormLayout::LabelRole)->widget()->show();
|
||
|
}
|
||
|
const auto fieldItem = layout.itemAt(0, QFormLayout::FieldRole);
|
||
|
if (fieldItem) {
|
||
|
fieldItem->widget()->hide();
|
||
|
QVERIFY(layout.isRowVisible(row));
|
||
|
QCOMPARE(rowPosition(row), rowPositions[row]);
|
||
|
layout.itemAt(0, QFormLayout::FieldRole)->widget()->show();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// If we hide both label and field, then the row should be considered hidden and the
|
||
|
// following row should move up into the space of the hidden row. We can only test
|
||
|
// this if both label and field are widgets, or if there is a spanning widget.
|
||
|
for (int row = 0; row < layout.rowCount() - 1; ++row) {
|
||
|
QWidget *labelWidget = nullptr;
|
||
|
if (auto labelItem = layout.itemAt(row, QFormLayout::LabelRole))
|
||
|
labelWidget = labelItem->widget();
|
||
|
QWidget *fieldWidget = nullptr;
|
||
|
if (auto fieldItem = layout.itemAt(row, QFormLayout::FieldRole))
|
||
|
fieldWidget = fieldItem->widget();
|
||
|
|
||
|
if (!fieldWidget)
|
||
|
continue;
|
||
|
if (labelWidget)
|
||
|
labelWidget->hide();
|
||
|
fieldWidget->hide();
|
||
|
QVERIFY(!layout.isRowVisible(row));
|
||
|
QVERIFY(!layout.isRowVisible(fieldWidget));
|
||
|
if (labelWidget)
|
||
|
QVERIFY(!layout.isRowVisible(labelWidget));
|
||
|
QTRY_COMPARE(rowPosition(row + 1), rowPositions[row]);
|
||
|
if (labelWidget)
|
||
|
labelWidget->show();
|
||
|
fieldWidget->show();
|
||
|
}
|
||
|
|
||
|
// hiding a row where a widget has focus must move focus to a widget in the next row
|
||
|
for (int row = 0; row < layout.rowCount(); ++row) {
|
||
|
QWidget *inputWidget = rowInputWidget(row);
|
||
|
QVERIFY(inputWidget);
|
||
|
inputWidget->setFocus();
|
||
|
layout.setRowVisible(row, false);
|
||
|
QVERIFY(!inputWidget->hasFocus());
|
||
|
}
|
||
|
|
||
|
// Now hide all rows, hide the toplevel widget, and show the toplevel widget again.
|
||
|
// None of the widgets inside must be visible.
|
||
|
for (int row = 0; row < layout.rowCount(); ++row)
|
||
|
layout.setRowVisible(row, false);
|
||
|
topLevel.hide();
|
||
|
topLevel.show();
|
||
|
for (int row = 0; row < layout.rowCount(); ++row)
|
||
|
QVERIFY(rowInputWidget(row)->isHidden());
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::showWithHiddenRow()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout layout;
|
||
|
|
||
|
for (int row = 0; row < 3; ++row)
|
||
|
layout.addRow(QString("Label %1").arg(row), new QLineEdit);
|
||
|
layout.setRowVisible(1, false);
|
||
|
|
||
|
topLevel.setLayout(&layout);
|
||
|
topLevel.show();
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Test that hiding rows does not leave outdated layout data behind
|
||
|
in hidden items that results in out-of-bounds array access. See
|
||
|
QTBUG-109237.
|
||
|
*/
|
||
|
void tst_QFormLayout::hiddenRowAndStretch()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout layout;
|
||
|
layout.setRowWrapPolicy(QFormLayout::WrapAllRows);
|
||
|
|
||
|
// We need our own stretcher item so that QFormLayout doesn't insert
|
||
|
// it's own, as that would grow the size of the layout data array again.
|
||
|
QSpacerItem *stretch = new QSpacerItem(100, 100, QSizePolicy::Expanding, QSizePolicy::Expanding);
|
||
|
layout.setItem(0, QFormLayout::FieldRole, stretch);
|
||
|
|
||
|
QLabel *lastLabel = nullptr;
|
||
|
QLineEdit *lastField = nullptr;
|
||
|
for (int row = 1; row < 4; ++row) {
|
||
|
QLabel *label = new QLabel(QString("Label %1").arg(row));
|
||
|
label->setWordWrap(true);
|
||
|
QLineEdit *field = new QLineEdit;
|
||
|
layout.setWidget(row, QFormLayout::LabelRole, label);
|
||
|
layout.setWidget(row, QFormLayout::FieldRole, field);
|
||
|
if (row == 3) {
|
||
|
lastLabel = label;
|
||
|
lastField = field;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Q_ASSERT(lastLabel);
|
||
|
Q_ASSERT(lastField);
|
||
|
|
||
|
topLevel.setLayout(&layout);
|
||
|
topLevel.sizeHint();
|
||
|
|
||
|
lastLabel->setVisible(false);
|
||
|
lastField->setVisible(false);
|
||
|
|
||
|
// should not assert here
|
||
|
topLevel.show();
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::itemAt()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
|
||
|
QWidget *w1 = new QWidget(&topLevel);
|
||
|
QWidget *w2 = new QWidget(&topLevel);
|
||
|
QWidget *w3 = new QWidget(&topLevel);
|
||
|
QWidget *w4 = new QWidget(&topLevel);
|
||
|
QWidget *w5 = new QWidget(&topLevel);
|
||
|
QHBoxLayout *l6 = new QHBoxLayout;
|
||
|
|
||
|
layout->setWidget(5, QFormLayout::LabelRole, w1);
|
||
|
layout->setWidget(3, QFormLayout::FieldRole, w2);
|
||
|
layout->setWidget(3, QFormLayout::LabelRole, w3);
|
||
|
layout->addRow(w4, w5);
|
||
|
layout->addRow("Foo:", l6);
|
||
|
|
||
|
QCOMPARE(layout->count(), 7);
|
||
|
|
||
|
QBitArray scoreBoard(7);
|
||
|
for (int i = 0; i < 7; ++i) {
|
||
|
QLayoutItem *item = layout->itemAt(i);
|
||
|
QVERIFY(item != 0);
|
||
|
|
||
|
if (item->widget() == w1) {
|
||
|
scoreBoard[0] = true;
|
||
|
} else if (item->widget() == w2) {
|
||
|
scoreBoard[1] = true;
|
||
|
} else if (item->widget() == w3) {
|
||
|
scoreBoard[2] = true;
|
||
|
} else if (item->widget() == w4) {
|
||
|
scoreBoard[3] = true;
|
||
|
} else if (item->widget() == w5) {
|
||
|
scoreBoard[4] = true;
|
||
|
} else if (item->layout() == l6) {
|
||
|
scoreBoard[5] = true;
|
||
|
} else if (qobject_cast<QLabel *>(item->widget())) {
|
||
|
scoreBoard[6] = true;
|
||
|
}
|
||
|
}
|
||
|
QCOMPARE(scoreBoard.count(false), 0);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::takeAt()
|
||
|
{
|
||
|
QWidget topLevel;
|
||
|
QFormLayout *layout = new QFormLayout(&topLevel);
|
||
|
|
||
|
QWidget *w1 = new QWidget(&topLevel);
|
||
|
QWidget *w2 = new QWidget(&topLevel);
|
||
|
QWidget *w3 = new QWidget(&topLevel);
|
||
|
QWidget *w4 = new QWidget(&topLevel);
|
||
|
QWidget *w5 = new QWidget(&topLevel);
|
||
|
QHBoxLayout *l6 = new QHBoxLayout;
|
||
|
|
||
|
layout->setWidget(5, QFormLayout::LabelRole, w1);
|
||
|
layout->setWidget(3, QFormLayout::FieldRole, w2);
|
||
|
layout->setWidget(3, QFormLayout::LabelRole, w3);
|
||
|
layout->addRow(w4, w5);
|
||
|
layout->addRow("Foo:", l6);
|
||
|
|
||
|
QCOMPARE(layout->count(), 7);
|
||
|
|
||
|
for (int i = 6; i >= 0; --i) {
|
||
|
delete layout->takeAt(0);
|
||
|
QCOMPARE(layout->count(), i);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::layoutAlone()
|
||
|
{
|
||
|
QWidget w;
|
||
|
QFormLayout layout;
|
||
|
layout.setFieldGrowthPolicy(QFormLayout::AllNonFixedFieldsGrow);
|
||
|
w.setLayout(&layout);
|
||
|
QLabel label("Here is a strange test case");
|
||
|
layout.setWidget(0, QFormLayout::LabelRole, &label);
|
||
|
QHBoxLayout hlay;
|
||
|
layout.setLayout(1, QFormLayout::LabelRole, &hlay);
|
||
|
QCOMPARE(layout.count(), 2);
|
||
|
w.setWindowTitle(QTest::currentTestFunction());
|
||
|
w.show();
|
||
|
layout.activate();
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::taskQTBUG_27420_takeAtShouldUnparentLayout()
|
||
|
{
|
||
|
QSharedPointer<QFormLayout> outer(new QFormLayout);
|
||
|
QAutoPointer<QFormLayout> holder{new QFormLayout};
|
||
|
auto inner = holder.get();
|
||
|
|
||
|
outer->addRow(inner);
|
||
|
QCOMPARE(outer->count(), 1);
|
||
|
QCOMPARE(inner->parent(), outer.data());
|
||
|
|
||
|
QLayoutItem *item = outer->takeAt(0);
|
||
|
QCOMPARE(item->layout(), inner);
|
||
|
QVERIFY(!item->layout()->parent());
|
||
|
|
||
|
outer.reset();
|
||
|
|
||
|
QVERIFY(holder); // a taken item/layout should not be deleted when the old parent is deleted
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::taskQTBUG_40609_addingWidgetToItsOwnLayout(){
|
||
|
QWidget widget;
|
||
|
widget.setObjectName("6435cbada60548b4522cbb6");
|
||
|
QFormLayout layout(&widget);
|
||
|
layout.setObjectName("c03c0e22c0b6d019a93a248");
|
||
|
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QLayout: Cannot add parent widget QWidget/6435cbada60548b4522cbb6 to its child layout QFormLayout/c03c0e22c0b6d019a93a248");
|
||
|
layout.addRow(QLatin1String("48c81f39b7320082f8"), &widget);
|
||
|
QCOMPARE(layout.count(), 0);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::taskQTBUG_40609_addingLayoutToItself(){
|
||
|
QWidget widget;
|
||
|
widget.setObjectName("2bc425637d084c07ce65956");
|
||
|
QFormLayout layout(&widget);
|
||
|
layout.setObjectName("60e31de0c8800eaba713a4f2");
|
||
|
|
||
|
QTest::ignoreMessage(QtWarningMsg, "QLayout: Cannot add layout QFormLayout/60e31de0c8800eaba713a4f2 to itself");
|
||
|
layout.addRow(QLatin1String("9a2cd4f40c06b489f889"), &layout);
|
||
|
QCOMPARE(layout.count(), 0);
|
||
|
}
|
||
|
|
||
|
void tst_QFormLayout::replaceWidget()
|
||
|
{
|
||
|
QWidget w;
|
||
|
QFormLayout *layout = new QFormLayout();
|
||
|
w.setLayout(layout);
|
||
|
QLineEdit *edit1 = new QLineEdit();
|
||
|
QLineEdit *edit2 = new QLineEdit();
|
||
|
QLineEdit *edit3 = new QLineEdit();
|
||
|
QLabel *label1 = new QLabel();
|
||
|
QLabel *label2 = new QLabel();
|
||
|
|
||
|
layout->addRow("Label", edit1);
|
||
|
layout->addRow(label1, edit2);
|
||
|
|
||
|
// Verify controls not in layout
|
||
|
QCOMPARE(layout->indexOf(edit3), -1);
|
||
|
QCOMPARE(layout->indexOf(label2), -1);
|
||
|
|
||
|
// Verify controls in layout
|
||
|
int editIndex = layout->indexOf(edit1);
|
||
|
int labelIndex = layout->indexOf(label1);
|
||
|
QVERIFY(editIndex > 0);
|
||
|
QVERIFY(labelIndex > 0);
|
||
|
int rownum;
|
||
|
QFormLayout::ItemRole role;
|
||
|
|
||
|
// replace editor
|
||
|
delete layout->replaceWidget(edit1, edit3);
|
||
|
edit1->hide(); // Not strictly needed for the test, but for normal usage it is.
|
||
|
QCOMPARE(layout->indexOf(edit1), -1);
|
||
|
QCOMPARE(layout->indexOf(edit3), editIndex);
|
||
|
QCOMPARE(layout->indexOf(label1), labelIndex);
|
||
|
rownum = -1;
|
||
|
role = QFormLayout::SpanningRole;
|
||
|
layout->getWidgetPosition(edit3, &rownum, &role);
|
||
|
QCOMPARE(rownum, 0);
|
||
|
QCOMPARE(role, QFormLayout::FieldRole);
|
||
|
|
||
|
delete layout->replaceWidget(label1, label2);
|
||
|
label1->hide();
|
||
|
QCOMPARE(layout->indexOf(label1), -1);
|
||
|
QCOMPARE(layout->indexOf(label2), labelIndex);
|
||
|
layout->getWidgetPosition(label2, &rownum, &role);
|
||
|
QCOMPARE(rownum, 1);
|
||
|
QCOMPARE(role, QFormLayout::LabelRole);
|
||
|
|
||
|
}
|
||
|
|
||
|
QTEST_MAIN(tst_QFormLayout)
|
||
|
|
||
|
#include "tst_qformlayout.moc"
|