qt5base-lts/tests/baseline/widgets/tst_baseline_widgets.cpp
Volker Hilsheimer 94039c9a6a Baseline test: add coverage for treeview with alternate rows
After f4976f86cd, alternate rows are no
longer painted all the way, as the code is never called for styles that
don't include the tree item decoration in the selection, such as the
Windows styles.

Add a baseline test to record the appearance of such trees.

Task-number: QTBUG-106227
Pick-to: 6.4 6.2
Change-Id: If21076393fdf65205cab91299f8e557dbe9c4ea9
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
Reviewed-by: Axel Spoerl <axel.spoerl@qt.io>
2022-10-24 14:20:28 +02:00

1111 lines
39 KiB
C++

// Copyright (C) 2021 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
#include <qbaselinetest.h>
#include <qwidgetbaselinetest.h>
#include <QtWidgets>
#include <QStyleOptionSlider>
class tst_Widgets : public QWidgetBaselineTest
{
Q_OBJECT
public:
tst_Widgets() = default;
private slots:
void tst_QSlider_data();
void tst_QSlider();
void tst_QPushButton_data();
void tst_QPushButton();
void tst_QPushButtonSquare();
void tst_QProgressBar_data();
void tst_QProgressBar();
void tst_QSpinBox_data();
void tst_QSpinBox();
void tst_QDoubleSpinBox_data();
void tst_QDoubleSpinBox();
void tst_QDateTimeEdit_data();
void tst_QDateTimeEdit();
void tst_QTimeEdit_data();
void tst_QTimeEdit();
void tst_QDateEdit_data();
void tst_QDateEdit();
void tst_QDial_data();
void tst_QDial();
void tst_QCheckbox_data();
void tst_QCheckbox();
void tst_QRadioButton_data();
void tst_QRadioButton();
void tst_QScrollBar_data();
void tst_QScrollBar();
void tst_QTabBar_data();
void tst_QTabBar();
void tst_QTabWidget_data();
void tst_QTabWidget();
void tst_QListView_data();
void tst_QListView();
void tst_QTableView_data();
void tst_QTableView();
void tst_QTreeView_data();
void tst_QTreeView();
void tst_QLineEdit_data();
void tst_QLineEdit();
private:
// Abstract SpinBox test for QSpinBox, QDoubleSpinBox, QDateTimeEdit, QDateEdit, QTimeEdit
void tst_SpinBox_data();
void tst_SpinBox(QAbstractSpinBox* spinBox);
// 78 standard icons from 6.3
const int numberStandardIcons = 78;
// recursive methods for QTreeView population
void tst_QTreeView_populateTree(QStandardItem* node, int height, int itemsPerNode, bool hasIcon);
QStandardItem* tst_QTreeView_populateItem(int height, int number, bool hasIcon);
};
void tst_Widgets::tst_QSlider_data()
{
QTest::addColumn<Qt::Orientation>("orientation");
QTest::addColumn<QSlider::TickPosition>("tickPosition");
QBaselineTest::newRow("horizontal") << Qt::Horizontal << QSlider::NoTicks;
QBaselineTest::newRow("horizontal ticks above") << Qt::Horizontal << QSlider::TicksAbove;
QBaselineTest::newRow("horizontal ticks below") << Qt::Horizontal << QSlider::TicksBelow;
QBaselineTest::newRow("horizontal ticks both") << Qt::Horizontal << QSlider::TicksBothSides;
QBaselineTest::newRow("vertical") << Qt::Vertical << QSlider::NoTicks;
QBaselineTest::newRow("vertical ticks left") << Qt::Vertical << QSlider::TicksLeft;
QBaselineTest::newRow("vertical ticks right") << Qt::Vertical << QSlider::TicksRight;
QBaselineTest::newRow("vertical ticks both") << Qt::Vertical << QSlider::TicksBothSides;
}
void tst_Widgets::tst_QSlider()
{
struct PublicSlider : QSlider { friend tst_Widgets; };
QFETCH(Qt::Orientation, orientation);
QFETCH(QSlider::TickPosition, tickPosition);
QBoxLayout *box = new QBoxLayout(orientation == Qt::Horizontal ? QBoxLayout::TopToBottom
: QBoxLayout::LeftToRight);
QList<QSlider*> _sliders;
for (int i = 0; i < 3; ++i) {
QSlider *slider = new QSlider;
slider->setOrientation(orientation);
slider->setTickPosition(tickPosition);
_sliders << slider;
box->addWidget(slider);
}
const auto sliders = _sliders;
testWindow()->setLayout(box);
// we want to see sliders with different values
int value = 0;
for (const auto &slider : sliders)
slider->setValue(value += 33);
takeStandardSnapshots();
PublicSlider *slider = static_cast<PublicSlider*>(sliders.first());
QStyleOptionSlider sliderOptions;
slider->initStyleOption(&sliderOptions);
const QRect handleRect = slider->style()->subControlRect(QStyle::CC_Slider, &sliderOptions,
QStyle::SubControl::SC_SliderHandle, slider);
QTest::mousePress(slider, Qt::LeftButton, {}, handleRect.center());
QBASELINE_CHECK(takeSnapshot(), "pressed");
QTest::mouseRelease(slider, Qt::LeftButton, {}, handleRect.center());
QBASELINE_CHECK(takeSnapshot(), "released");
slider->setSliderDown(true);
QBASELINE_CHECK(takeSnapshot(), "down");
sliders.first()->setSliderDown(false);
QBASELINE_CHECK(takeSnapshot(), "notdown");
}
void tst_Widgets::tst_QPushButton_data()
{
QTest::addColumn<bool>("isFlat");
QBaselineTest::newRow("normal") << false;
QBaselineTest::newRow("flat") << true;
}
void tst_Widgets::tst_QPushButton()
{
QFETCH(bool, isFlat);
QVBoxLayout *vbox = new QVBoxLayout;
QPushButton *testButton = new QPushButton("Ok");
testButton->setFlat(isFlat);
vbox->addWidget(testButton);
testWindow()->setLayout(vbox);
takeStandardSnapshots();
testButton->setDown(true);
QBASELINE_CHECK(takeSnapshot(), "down");
testButton->setDown(false);
QBASELINE_CHECK(takeSnapshot(), "up");
testButton->setDefault(true);
QBASELINE_CHECK(takeSnapshot(), "default_up");
testButton->setDown(true);
QBASELINE_CHECK(takeSnapshot(), "default_down");
testButton->setDown(false);
}
void tst_Widgets::tst_QPushButtonSquare()
{
QVBoxLayout layout;
QPushButton button(testWindow());
button.setText(QLatin1String("Square"));
const auto sizeHint = button.sizeHint().width();
// Depending on the current QStyle, this may result in
// a different button look - on macOS it will look as
// a toolbutton:
button.setFixedSize(sizeHint, sizeHint);
layout.addWidget(&button);
testWindow()->setLayout(&layout);
takeStandardSnapshots();
button.setCheckable(true);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "square_unchecked");
button.setChecked(true);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "square_checked");
}
void tst_Widgets::tst_QProgressBar_data()
{
QTest::addColumn<Qt::Orientation>("orientation");
QTest::addColumn<bool>("invertedAppearance");
QTest::addColumn<bool>("textVisible");
QTest::newRow("vertical_normalAppearance_textVisible") << Qt::Vertical << false << true;
QTest::newRow("vertical_invertedAppearance_textVisible") << Qt::Vertical << true << true;
QTest::newRow("horizontal_normalAppearance_textVisible") << Qt::Horizontal << false << true;
QTest::newRow("horizontal_invertedAppearance_textVisible") << Qt::Horizontal << true << true;
QTest::newRow("vertical_normalAppearance_textNotVisible") << Qt::Vertical << false << false;
QTest::newRow("vertical_invertedAppearance_textNotVisible") << Qt::Vertical << true << false;
QTest::newRow("horizontal_normalAppearance_textNotVisible") << Qt::Horizontal << false << false;
QTest::newRow("horizontal_invertedAppearance_textNotVisible") << Qt::Horizontal << true << false;
}
void tst_Widgets::tst_QProgressBar()
{
QFETCH(Qt::Orientation, orientation);
QFETCH(bool, invertedAppearance);
QFETCH(bool, textVisible);
QBoxLayout box((orientation == Qt::Vertical) ? QBoxLayout::LeftToRight : QBoxLayout::TopToBottom);
for (int i = 0; i < 4; ++i) {
QProgressBar *bar = new QProgressBar(testWindow());
bar->setOrientation(orientation);
bar->setInvertedAppearance(invertedAppearance);
bar->setTextVisible(textVisible);
bar->setValue(i * 33);
box.addWidget(bar);
}
testWindow()->setLayout(&box);
takeStandardSnapshots();
}
void tst_Widgets::tst_SpinBox_data()
{
QTest::addColumn<QAbstractSpinBox::ButtonSymbols>("buttons");
QTest::addRow("NoButtons") << QAbstractSpinBox::NoButtons;
QTest::addRow("UpDownArrows") << QAbstractSpinBox::UpDownArrows;
QTest::addRow("PlusMinus") << QAbstractSpinBox::PlusMinus;
}
void tst_Widgets::tst_SpinBox(QAbstractSpinBox *spinBox)
{
QFETCH(const QAbstractSpinBox::ButtonSymbols, buttons);
spinBox->setButtonSymbols(buttons);
spinBox->setMinimumWidth(200);
QVBoxLayout layout;
layout.addWidget(spinBox);
testWindow()->setLayout(&layout);
takeStandardSnapshots();
spinBox->setAlignment(Qt::AlignHCenter);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "alignCenter");
spinBox->setAlignment(Qt::AlignRight);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "alignRight");
// Press / release up button
QStyleOptionSpinBox styleOption;
styleOption.initFrom(spinBox);
QPoint clickTarget = spinBox->style()->subControlRect(QStyle::CC_SpinBox,&styleOption,
QStyle::SC_SpinBoxUp,spinBox).center();
QTest::mousePress(spinBox, Qt::LeftButton, Qt::KeyboardModifiers(), clickTarget);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "upPressed");
QTest::mouseRelease(spinBox, Qt::LeftButton, Qt::KeyboardModifiers(), clickTarget);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "upReleased");
// Press / release down button
clickTarget = spinBox->style()->subControlRect(QStyle::CC_SpinBox,&styleOption,
QStyle::SC_SpinBoxDown,spinBox).center();
QTest::mousePress(spinBox, Qt::LeftButton, Qt::KeyboardModifiers(), clickTarget);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "downPressed");
QTest::mouseRelease(spinBox, Qt::LeftButton, Qt::KeyboardModifiers(), clickTarget);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "downReleased");
}
void tst_Widgets::tst_QSpinBox_data()
{
tst_SpinBox_data();
}
void tst_Widgets::tst_QSpinBox()
{
QSpinBox spinBox;
tst_SpinBox(&spinBox);
}
void tst_Widgets::tst_QDoubleSpinBox_data()
{
tst_SpinBox_data();
}
void tst_Widgets::tst_QDoubleSpinBox()
{
QDoubleSpinBox spinBox;
tst_SpinBox(&spinBox);
}
void tst_Widgets::tst_QDateTimeEdit_data()
{
tst_SpinBox_data();
}
void tst_Widgets::tst_QDateTimeEdit()
{
QDateTimeEdit edit;
tst_SpinBox(&edit);
// show calendar popup
QStyleOptionSpinBox styleOption;
styleOption.initFrom(&edit);
const QRect buttonUp = edit.style()->subControlRect(QStyle::CC_SpinBox,&styleOption,
QStyle::SC_SpinBoxUp,&edit);
// no rect for popup button => use bottom center of up-button
QPoint clickTarget = buttonUp.center();
clickTarget.setY(buttonUp.bottomLeft().y());
edit.setCalendarPopup(true);
QTest::mouseClick(&edit, Qt::LeftButton, Qt::KeyboardModifiers(), clickTarget);
QCalendarWidget* calendar = edit.calendarWidget();
QVERIFY(calendar);
QVBoxLayout layout;
layout.addWidget(calendar);
testWindow()->setLayout(&layout);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "showCalendar");
}
void tst_Widgets::tst_QTimeEdit_data()
{
tst_SpinBox_data();
}
void tst_Widgets::tst_QTimeEdit()
{
QTimeEdit edit;
tst_SpinBox(&edit);
}
void tst_Widgets::tst_QDateEdit_data()
{
tst_SpinBox_data();
}
void tst_Widgets::tst_QDateEdit()
{
QDateEdit edit;
tst_SpinBox(&edit);
}
void tst_Widgets::tst_QDial_data()
{
QTest::addColumn<int>("minimum");
QTest::addColumn<int>("maximum");
QTest::addColumn<bool>("notchesVisible");
QTest::addColumn<qreal>("notchTarget");
QTest::newRow("0..99_notches") << 0 << 99 << true << 3.7;
QTest::newRow("0..99_noNotches") << 0 << 99 << false << 3.7;
QTest::newRow("1..100_notches") << 1 << 100 << true << 5.7;
QTest::newRow("1..100_noNotches") << 1 << 100 << false << 3.7;
QTest::newRow("1..5_notches") << 1 << 5 << true << 8.7;
QTest::newRow("1..5_noNotches") << 1 << 5 << false << 3.7;
}
void tst_Widgets::tst_QDial()
{
QFETCH(int, minimum);
QFETCH(int, maximum);
QFETCH(bool, notchesVisible);
QFETCH(qreal, notchTarget);
QVERIFY(maximum > minimum);
const int steps = maximum - minimum;
QDial dial(testWindow());
dial.setMinimum(minimum);
dial.setMaximum(maximum);
dial.setNotchTarget(notchTarget);
dial.setSliderPosition(minimum + (steps / 2));
dial.setNotchesVisible(notchesVisible);
QBoxLayout box(QBoxLayout::LeftToRight);
box.addWidget(&dial);
testWindow()->setLayout(&box);
takeStandardSnapshots();
}
void tst_Widgets::tst_QCheckbox_data()
{
QTest::addColumn<QString>("text");
QTest::addColumn<bool>("hasIcon");
QTest::addColumn<bool>("isTriState");
QTest::newRow("SimpleCheckbox") << "" << false << false;
QTest::newRow("SimpleCheckboxWithIcon") << "" << true << false;
QTest::newRow("SimpleCheckboxWithText") << "checkBox" << false << false;
QTest::newRow("SimpleCheckboxWithTextAndIcon") << "checkBox with icon" << true << false;
QTest::newRow("SimpleTristate") << "" << false << true;
QTest::newRow("SimpleTristateWithText") << "tristateBox" << false << true;
}
void tst_Widgets::tst_QCheckbox()
{
QFETCH(QString, text);
QFETCH(bool, hasIcon);
QFETCH(bool, isTriState);
class CheckBox : public QCheckBox
{
public:
using QCheckBox::initStyleOption;
};
QBoxLayout layout(QBoxLayout::TopToBottom);
CheckBox box;
box.setTristate(isTriState);
if (!text.isEmpty())
box.setText(text);
if (hasIcon)
box.setIcon(QApplication::style()->standardIcon(QStyle::SP_ComputerIcon));
layout.addWidget(&box);
testWindow()->setLayout(&layout);
takeStandardSnapshots();
do {
const Qt::CheckState checkState = box.checkState();
QStyleOptionButton styleOption;
box.initStyleOption(&styleOption);
const QPoint clickTarget = box.style()->subElementRect(QStyle::SE_CheckBoxClickRect, &styleOption, &box).center();
const std::array titles = {"unChecked", "partiallyChecked", "checked"};
const QString snapShotTitle = titles[checkState];
QTest::mousePress(&box,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QVERIFY(box.isDown());
QBASELINE_CHECK_DEFERRED(takeSnapshot(), (snapShotTitle + "_pressed").toLocal8Bit().constData());
QTest::mouseRelease(&box,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QVERIFY(!box.isDown());
QVERIFY(checkState != box.checkState());
QBASELINE_CHECK_DEFERRED(takeSnapshot(), (snapShotTitle + "_released").toLocal8Bit().constData());
} while (box.checkState() != Qt::Unchecked);
}
void tst_Widgets::tst_QRadioButton_data()
{
QTest::addColumn<QString>("text");
QTest::addColumn<bool>("hasIcon");
QTest::newRow("SimpleRadioButton") << "" << false;
QTest::newRow("RadioButtonWithText") << "RadioButton" << false;
QTest::newRow("SimpleRadioButtonWithIcon") << "" << true;
QTest::newRow("RadioButtonWithTextAndIcon") << "RadioButton" << true;
}
void tst_Widgets::tst_QRadioButton()
{
QFETCH(QString,text);
QFETCH(bool,hasIcon);
class RadioButton : public QRadioButton
{
public:
using QRadioButton::QRadioButton;
using QRadioButton::initStyleOption;
};
RadioButton button1(testWindow());
if (!text.isEmpty())
button1.setText(text);
if (hasIcon)
button1.setIcon(QApplication::style()->standardIcon(QStyle::SP_ComputerIcon));
button1.setChecked(false);
RadioButton button2(testWindow());
if (!text.isEmpty())
button2.setText(text);
if (hasIcon)
button2.setIcon(QApplication::style()->standardIcon(QStyle::SP_ComputerIcon));
// button2 has to start checked for the following tests to work
button2.setChecked(true);
QBoxLayout box(QBoxLayout::TopToBottom);
box.addWidget(&button1);
box.addWidget(&button2);
testWindow()->setLayout(&box);
takeStandardSnapshots();
QStyleOptionButton styleOption;
button1.initStyleOption(&styleOption);
const QPoint clickTarget = button1.style()->subElementRect(QStyle::SE_RadioButtonClickRect, &styleOption, &button1).center();
QTest::mousePress(&button1,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QVERIFY(button1.isDown());
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressUnchecked");
QTest::mouseRelease(&button1,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QVERIFY(!button1.isDown());
// button1 has grabbed the check from button2
QVERIFY(button1.isChecked());
QVERIFY(!button2.isChecked());
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "releaseUnchecked");
// press and release checked button1 again
QTest::mousePress(&button1,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QVERIFY(button1.isDown());
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressChecked");
QTest::mouseRelease(&button1,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QVERIFY(!button1.isDown());
// checkstate not supposed to change
QVERIFY(button1.isChecked());
QVERIFY(!button2.isChecked());
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "releaseChecked");
}
void tst_Widgets::tst_QScrollBar_data()
{
QTest::addColumn<Qt::Orientation>("orientation");
QTest::newRow("Horizontal") << Qt::Horizontal;
QTest::newRow("Vertical") << Qt::Vertical;
}
void tst_Widgets::tst_QScrollBar()
{
QFETCH(Qt::Orientation, orientation);
QBoxLayout box((orientation == Qt::Vertical) ? QBoxLayout::LeftToRight
: QBoxLayout::TopToBottom);
QList<QScrollBar*> bars;
for (int i = 0; i < 4; ++i) {
QScrollBar *bar = new QScrollBar(testWindow());
(orientation == Qt::Vertical) ? bar->setFixedHeight(100)
: bar->setFixedWidth(100);
bar->setOrientation(orientation);
bar->setValue(i * 33);
box.addWidget(bar);
bars.append(bar);
}
testWindow()->setLayout(&box);
takeStandardSnapshots();
// press left/up of first bar
QScrollBar *bar = bars.at(0);
QStyleOptionSlider styleOption = qt_qscrollbarStyleOption(bar);
QPoint clickTarget = bar->style()->subControlRect(QStyle::CC_ScrollBar, &styleOption,
QStyle::SC_ScrollBarSubLine, bar).center();
QTest::mousePress(bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressLeftUp");
QTest::mouseRelease(bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
// press slider of first bar
styleOption = qt_qscrollbarStyleOption(bar);
clickTarget = bar->style()->subControlRect(QStyle::CC_ScrollBar, &styleOption,
QStyle::SC_ScrollBarSlider, bar).center();
QTest::mousePress(bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QVERIFY(bar->isSliderDown());
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressSlider");
QTest::mouseRelease(bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
// Press AddPage up on first bar
clickTarget = bar->style()->subControlRect(QStyle::CC_ScrollBar, &styleOption,
QStyle::SC_ScrollBarAddPage, bar).center();
QTest::mousePress(bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressAddPage");
QTest::mouseRelease(bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
// press SubPage of last bar
bar = bars.at(3);
styleOption = qt_qscrollbarStyleOption(bar);
clickTarget = bar->style()->subControlRect(QStyle::CC_ScrollBar, &styleOption,
QStyle::SC_ScrollBarAddLine, bar).center();
QTest::mousePress(bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressRightDown");
QTest::mouseRelease(bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
}
void tst_Widgets::tst_QTabBar_data()
{
QTest::addColumn<QTabBar::Shape>("shape");
QTest::addColumn<int>("numberTabs");
QTest::addColumn<int>("fixedWidth");
QTest::addColumn<bool>("isClosable");
// fixedWidth <0 will be interpreted as variable width
QTest::newRow("RoundedNorth_3_variableWidth") << QTabBar::RoundedNorth << 3 << -1 << false;
QTest::newRow("RoundedEast_3_variableWidth") << QTabBar::RoundedEast << 3 << -1 << false;
QTest::newRow("RoundedWest_3_variableWidth") << QTabBar::RoundedWest << 3 << -1 << false;
QTest::newRow("RoundedSouth_3_variableWidth") << QTabBar::RoundedSouth << 3 << -1 << false;
QTest::newRow("RoundedNorth_20_fixedWidth") << QTabBar::RoundedNorth << 20 << 250 << true;
}
void tst_Widgets::tst_QTabBar()
{
QFETCH(QTabBar::Shape, shape);
QFETCH(int, numberTabs);
QFETCH(int, fixedWidth);
QFETCH(bool, isClosable);
QTabBar bar (testWindow());
bar.setShape(shape);
bar.setTabsClosable(isClosable);
if (fixedWidth > 0)
bar.setFixedWidth(fixedWidth);
for (int i = 0; i < numberTabs; ++i) {
bar.insertTab(i,"Tab_" + QString::number(i));
}
QBoxLayout box(QBoxLayout::LeftToRight, testWindow());
box.addWidget(&bar);
testWindow()->setLayout(&box);
takeStandardSnapshots();
// press/release first tab
bar.setCurrentIndex(0);
QPoint clickTarget = bar.tabRect(0).center();
QTest::mousePress(&bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressFirstTab");
QTest::mouseRelease(&bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QVERIFY(bar.currentIndex() == 0);
// press/release second tab if it exists
if (bar.count() > 1) {
clickTarget = bar.tabRect(1).center();
QTest::mousePress(&bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressSecondTab");
QTest::mouseRelease(&bar,Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QVERIFY(bar.currentIndex() == 1);
}
// test press/release on close button
if (isClosable) {
// CloseButton is either left or right
QWidget *leftButton = bar.tabButton(bar.currentIndex(),QTabBar::ButtonPosition::LeftSide);
QWidget *rightButton = bar.tabButton(bar.currentIndex(),QTabBar::ButtonPosition::RightSide);
QAbstractButton *button = qobject_cast<QAbstractButton*>(leftButton);
if (button == nullptr)
button = qobject_cast<QAbstractButton*>(rightButton);
if (button != nullptr) {
clickTarget = button->rect().center();
QTest::mousePress(button,Qt::MouseButton::LeftButton,
Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressCloseFirstTab");
QTest::mouseRelease(button,Qt::MouseButton::LeftButton,
Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "releaseCloseFirstTab");
}
}
}
void tst_Widgets::tst_QTabWidget_data()
{
QTest::addColumn<QTabWidget::TabPosition>("tabPosition");
QTest::addColumn<int>("numberTabs");
QTest::addColumn<QString>("tabText");
QTest::addColumn<int>("fixedWidth");
QTest::addColumn<bool>("isClosable");
QTest::addColumn<bool>("isDocumentMode");
// fixedWidth <0 will be interpreted as variable width
QTest::newRow("North_3_variableWidthDocMode") << QTabWidget::North << 3 << "This is a tab text" << -1 << false << true;
QTest::newRow("East_3_variableWidth") << QTabWidget::East << 3 << "This is a tab text" << -1 << false << false;
QTest::newRow("West_3_variableWidthDocMode") << QTabWidget::West << 3 << "This is a tab text" << -1 << false << true;
QTest::newRow("South_3_variableWidth") << QTabWidget::South << 3 << "This is a tab text" << -1 << true << false;
QTest::newRow("North_20_fixedWidthDocMode") << QTabWidget::North << 20
<< "This is a very long text to actually force wrapping!" << 100 << true << true;
QTest::newRow("South_20_variableWidth") << QTabWidget::South << 20
<< "This is a very long text to actually force wrapping!" << -1 << false << false;
}
void tst_Widgets::tst_QTabWidget()
{
QFETCH(QTabWidget::TabPosition, tabPosition);
QFETCH(int, numberTabs);
QFETCH(QString, tabText);
QFETCH(int, fixedWidth);
QFETCH(bool, isClosable);
QFETCH(bool, isDocumentMode);
QTabWidget tabWidget (testWindow());
if (fixedWidth > 0)
tabWidget.setFixedWidth(fixedWidth);
tabWidget.setTabPosition(tabPosition);
tabWidget.setTabsClosable(isClosable);
tabWidget.setDocumentMode(isDocumentMode);
for (int i = 0; i < numberTabs; ++i) {
QLabel *tabLabel = new QLabel("Tab number " + QString::number(i) + "\n" + tabText, &tabWidget);
QBoxLayout *tabBox = new QBoxLayout(QBoxLayout::TopToBottom,&tabWidget);
tabBox->addWidget(tabLabel);
tabWidget.insertTab(i,tabLabel,"Tab_" + QString::number(i));
tabWidget.setCurrentIndex(i);
tabWidget.currentWidget()->setLayout(tabBox);
}
tabWidget.setCurrentIndex(0);
QBoxLayout box(QBoxLayout::LeftToRight, testWindow());
box.addWidget(&tabWidget);
testWindow()->setLayout(&box);
takeStandardSnapshots();
// press/release on second tab if it exists
if (numberTabs > 1) {
const QPoint clickTarget = tabWidget.tabBar()->tabRect(1).center();
QTest::mousePress(tabWidget.tabBar(),Qt::MouseButton::LeftButton,Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressSecondTab");
QTest::mouseRelease(tabWidget.tabBar(),Qt::MouseButton::LeftButton, Qt::KeyboardModifiers(), clickTarget,0);
QVERIFY(tabWidget.currentIndex() == 1);
}
// test press/release on close button
if (isClosable) {
// CloseButton is either left or right
QWidget *leftButton = tabWidget.tabBar()->tabButton(tabWidget.currentIndex(),QTabBar::ButtonPosition::LeftSide);
QWidget *rightButton = tabWidget.tabBar()->tabButton(tabWidget.currentIndex(),QTabBar::ButtonPosition::RightSide);
QAbstractButton *button = qobject_cast<QAbstractButton*>(leftButton);
if (button == nullptr)
button = qobject_cast<QAbstractButton*>(rightButton);
if (button != nullptr) {
const QPoint clickTarget = button->rect().center();
QTest::mousePress(button,Qt::MouseButton::LeftButton,
Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "pressCloseTab");
QTest::mouseRelease(button,Qt::MouseButton::LeftButton,
Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "releaseCloseTab");
}
}
}
void tst_Widgets::tst_QListView_data()
{
QTest::addColumn<QListView::ViewMode>("viewMode");
QTest::addColumn<bool>("isWrapping");
QTest::addColumn<bool>("hasWordWrap");
QTest::addColumn<int>("numberItems");
QTest::addColumn<QSize>("fixedSize");
// QSize() will be interpreted as variable size
QTest::newRow("ListModeWrappingNoWordWrapFixed_10") <<
QListView::ListMode << true << false << 10 << QSize(100, 500);
QTest::newRow("ListModeNoWrappingNoWordWrapVariable_20") <<
QListView::ListMode << false << true << 20 << QSize();
QTest::newRow("ListModeNoWrappingWordWrapVariable_30") <<
QListView::ListMode << false << true << 30 << QSize();
QTest::newRow("IconModeNoWrappingNoWordWrapFixed_10") <<
QListView::IconMode << false << false << 10 << QSize(100, 500);
QTest::newRow("IconModeWrappingNoWordWrapVariable_20") <<
QListView::IconMode << true << false << 20 << QSize();
QTest::newRow("IconModeWrappingWordWrapVariable_30") <<
QListView::IconMode << true << true << 30 << QSize(100, 500);
}
void tst_Widgets::tst_QListView()
{
QFETCH(QListView::ViewMode,viewMode);
QFETCH(bool,isWrapping);
QFETCH(bool,hasWordWrap);
QFETCH(int,numberItems);
QFETCH(QSize,fixedSize);
QListView listView;
listView.setViewMode(viewMode);
listView.setWrapping(isWrapping);
listView.setWordWrap(hasWordWrap);
if (fixedSize.isValid())
listView.setFixedSize(fixedSize);
QStandardItemModel model(0,1,testWindow());
// Populate model, add standard icons if required
const QString itemText = hasWordWrap ? "This is a long text for word wrapping Item_"
: "ListItem_";
int icon = 0;
for (int i = 0; i < numberItems; ++i) {
QStandardItem *item;
if (viewMode == QListView::IconMode) {
item = new QStandardItem(QApplication::style()->standardIcon
(static_cast<QStyle::StandardPixmap>(icon)), itemText + QString::number(i));
icon = (icon + 1) % numberStandardIcons;
} else {
item = new QStandardItem(itemText + QString::number(i));
}
model.appendRow(item);
}
listView.setModel(&model);
QBoxLayout layout(QBoxLayout::LeftToRight, testWindow());
layout.addWidget(&listView);
testWindow()->setLayout(&layout);
takeStandardSnapshots();
// click on first item
QPoint clickTarget = listView.visualRect(model.index(0,0)).center();
QTest::mouseClick(listView.viewport(),Qt::MouseButton::LeftButton,
Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "clickFirstItem");
// click on scond item
if (numberItems > 1) {
clickTarget = listView.visualRect(model.index(1,0)).center();
QTest::mouseClick(listView.viewport(),Qt::MouseButton::LeftButton,
Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "clickSecondItem");
}
// Hide first row
listView.setRowHidden(0,true);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "hideFirstItem");
}
void tst_Widgets::tst_QTableView_data()
{
QTest::addColumn<bool>("hasHeader");
QTest::addColumn<bool>("hasRowNumbers");
QTest::addColumn<bool>("hasWordWrap");
QTest::addColumn<int>("numberRows");
QTest::addColumn<int>("numberColumns");
QTest::addColumn<int>("iconColumn");
QTest::addColumn<QSize>("fixedSize");
// QSize() => variable size; iconColumn -1 => no icon
QTest::newRow("HeaderRowNumWordWrapFixed_10") << true << true << true << 10 << 3 << -1 << QSize(500, 100);
QTest::newRow("HeaderVariable_20") << true << false << false << 20 << 4 << 1 << QSize();
QTest::newRow("HeaderFixed_20") << true << false << false << 20 << 4 << 1 << QSize(500, 700);
}
void tst_Widgets::tst_QTableView()
{
QFETCH(bool, hasHeader);
QFETCH(bool, hasRowNumbers);
QFETCH(bool, hasWordWrap);
QFETCH(int, numberRows);
QFETCH(int, numberColumns);
QFETCH(int, iconColumn);
QFETCH(QSize, fixedSize);
// Populate model
int icon = 0;
QStandardItemModel model(numberRows, numberColumns, testWindow());
if (hasHeader) {
for (int i = 0; i < numberColumns; ++i)
model.setHorizontalHeaderItem(i, new QStandardItem("Header_" + QString::number(i)));
}
const QString wrap = hasWordWrap ? "\n long text to wrap words" : "" ;
for (int row = 0; row < numberRows; ++row) {
for (int column = 0; column < numberColumns; ++column) {
QStandardItem *item;
const QString itemText = QString::number(row) + "/" + QString::number(column) + wrap;
if (iconColumn == column) {
item = new QStandardItem(QApplication::style()->standardIcon
(static_cast<QStyle::StandardPixmap>(icon)),itemText);
icon = (icon + 1) % numberStandardIcons;
} else {
item = new QStandardItem(itemText);
}
model.setItem(row,column,item);
}
if (hasRowNumbers)
model.setVerticalHeaderItem(row, new QStandardItem(QString::number(row)));
}
QTableView tableView(testWindow());
tableView.setWordWrap(hasWordWrap);
if (fixedSize.isValid())
tableView.setFixedSize(fixedSize);
QBoxLayout layout(QBoxLayout::LeftToRight, testWindow());
tableView.setModel(&model);
layout.addWidget(&tableView);
takeStandardSnapshots();
// Hide grid
tableView.setShowGrid(false);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "hideGrid");
tableView.setShowGrid(true);
// click item 0,0
QPoint clickTarget = tableView.visualRect(model.index(0,0)).center();
QTest::mouseClick(tableView.viewport(),Qt::MouseButton::LeftButton,
Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "clickFirstItem");
// click item 0,1 if it exists
if (numberColumns > 1) {
clickTarget = tableView.visualRect(model.index(0,1)).center();
QTest::mouseClick(tableView.viewport(),Qt::MouseButton::LeftButton,
Qt::KeyboardModifiers(), clickTarget,0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "clickSecondItem");
}
tableView.clearSelection();
// Hide first row and column
tableView.setRowHidden(0, true);
tableView.setColumnHidden(0, true);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "hideFirstRowColumn");
tableView.setRowHidden(0, false);
tableView.setColumnHidden(0, false);
// Select first row
tableView.selectRow(0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "selectFirstRow");
// Select first column
tableView.selectColumn(0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "selectFirstColumn");
}
void tst_Widgets::tst_QTreeView_data()
{
QTest::addColumn<bool>("showHeader");
QTest::addColumn<bool>("hasIcons");
QTest::addColumn<bool>("alternatingRowColors");
QTest::addColumn<QSize>("fixedSize");
QTest::addColumn<int>("treeHeight");
QTest::addColumn<int>("itemsPerNode");
// QSize() => variable size
QTest::newRow("HeaderIcons_4_3") << true << true << false << QSize() << 3 << 2;
QTest::newRow("NoHeaderNoIcons_4_4") << false << false << false << QSize(100, 350) << 3 << 2;
QTest::newRow("AlternatingRows") << true << true << true << QSize() << 3 << 2;
}
void tst_Widgets::tst_QTreeView()
{
QFETCH(bool, showHeader);
QFETCH(bool, hasIcons);
QFETCH(bool, alternatingRowColors);
QFETCH(QSize, fixedSize);
QFETCH(int, treeHeight);
QFETCH(int, itemsPerNode);
QVERIFY(treeHeight > 0 && itemsPerNode > 0);
QTreeView treeView(testWindow());
fixedSize.isValid() ? treeView.setFixedSize(fixedSize)
: treeView.setSizeAdjustPolicy(QAbstractScrollArea::AdjustToContents);
QStandardItemModel model(&treeView);
showHeader ? model.setHorizontalHeaderItem(0, new QStandardItem("TreeHeader"))
: treeView.setHeaderHidden(true);
treeView.setAlternatingRowColors(alternatingRowColors);
// Populate tree model
for (int i = 0; i < itemsPerNode; ++i) {
QStandardItem* root = tst_QTreeView_populateItem(treeHeight, i, hasIcons);
tst_QTreeView_populateTree(root,treeHeight - 1,itemsPerNode, hasIcons);
model.appendRow(root);
}
treeView.setModel(&model);
QBoxLayout layout(QBoxLayout::LeftToRight, testWindow());
layout.addWidget(&treeView);
testWindow()->setLayout(&layout);
treeView.expandAll();
treeView.resizeColumnToContents(0);
takeStandardSnapshots();
// Partly expand if possible
if (treeHeight > 1) {
treeView.expandToDepth(1);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "partlyExpanded");
}
// Click on first node
QPoint clickTarget = treeView.visualRect(model.index(0, 0)).center();
QTest::mouseClick(treeView.viewport(),Qt::MouseButton::LeftButton,
Qt::KeyboardModifiers(), clickTarget, 0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "clickFirstNode");
// Hide first row
treeView.setRowHidden(0, model.index(0, 0), true);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "hideFirstRow");
treeView.setRowHidden(0, model.index(0, 0), false);
// Click on second row if it exists
if (itemsPerNode > 1) {
clickTarget = treeView.visualRect(model.index(1, 0)).center();
QTest::mouseClick(treeView.viewport(), Qt::MouseButton::LeftButton,
Qt::KeyboardModifiers(), clickTarget, 0);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "clickSecondNode");
}
}
void tst_Widgets::tst_QTreeView_populateTree(QStandardItem* node, int height, int itemsPerNode, bool hasIcon)
{
QList<QStandardItem*> items;
for (int i = 0; i < itemsPerNode; ++i) {
if (height == 0) {
items.append(tst_QTreeView_populateItem(height, i, hasIcon));
} else {
QStandardItem* item = tst_QTreeView_populateItem(height, i, hasIcon);
tst_QTreeView_populateTree(item, height - 1, itemsPerNode, hasIcon);
items.append(item);
}
}
return node->appendColumn(items);
}
QStandardItem* tst_Widgets::tst_QTreeView_populateItem(int height, int number, bool hasIcon)
{
static int icon = 0;
static int itemCount = 0;
QStandardItem* item;
const QString itemText = QString("%1/%2/%3").arg(height).arg(number).arg(itemCount);
++itemCount;
if (hasIcon) {
item = new QStandardItem(QApplication::style()->standardIcon
(static_cast<QStyle::StandardPixmap>(icon)), itemText);
icon = (icon + 1) % numberStandardIcons;
} else {
item = new QStandardItem(itemText);
}
return item;
}
void tst_Widgets::tst_QLineEdit_data()
{
QTest::addColumn<bool>("hasFrame");
QTest::addColumn<QLineEdit::EchoMode>("echoMode");
QTest::addColumn<QString>("placeHolderText");
QTest::addColumn<QString>("text");
QTest::newRow("framePassword") << true << QLineEdit::Password << "password" << "secret";
QTest::newRow("noFrameCleartext") << false << QLineEdit::Normal << "text" << "this is a text";
}
void tst_Widgets::tst_QLineEdit()
{
QFETCH(const bool, hasFrame);
QFETCH(const QLineEdit::EchoMode, echoMode);
QFETCH(const QString, placeHolderText);
QFETCH(const QString, text);
QLineEdit lineEdit(testWindow());
lineEdit.setFrame(hasFrame);
lineEdit.setEchoMode(echoMode);
lineEdit.setPlaceholderText(placeHolderText);
QHBoxLayout layout;
layout.addWidget(&lineEdit);
testWindow()->setLayout(&layout);
takeStandardSnapshots();
lineEdit.setText(text);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "setText");
lineEdit.setAlignment(Qt::AlignRight);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "alignedRight");
lineEdit.setAlignment(Qt::AlignCenter);
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "alignedCenter");
lineEdit.setSelection(0,text.size());
QBASELINE_CHECK_DEFERRED(takeSnapshot(), "textSelected");
}
#define main _realmain
QTEST_MAIN(tst_Widgets)
#undef main
int main(int argc, char *argv[])
{
// Avoid rendering variations caused by QHash randomization
QHashSeed::setDeterministicGlobalSeed();
QBaselineTest::handleCmdLineArgs(&argc, &argv);
return _realmain(argc, argv);
}
#include "tst_baseline_widgets.moc"