1c6bf3e09e
This is a semantic patch using ClangTidyTransformator as in qtbase/df9d882d41b741fef7c5beeddb0abe9d904443d8, but extended to handle typedefs and accesses through pointers, too: const std::string o = "object"; auto hasTypeIgnoringPointer = [](auto type) { return anyOf(hasType(type), hasType(pointsTo(type))); }; auto derivedFromAnyOfClasses = [&](ArrayRef<StringRef> classes) { auto exprOfDeclaredType = [&](auto decl) { return expr(hasTypeIgnoringPointer(hasUnqualifiedDesugaredType(recordType(hasDeclaration(decl))))).bind(o); }; return exprOfDeclaredType(cxxRecordDecl(isSameOrDerivedFrom(hasAnyName(classes)))); }; auto renameMethod = [&] (ArrayRef<StringRef> classes, StringRef from, StringRef to) { return makeRule(cxxMemberCallExpr(on(derivedFromAnyOfClasses(classes)), callee(cxxMethodDecl(hasName(from), parameterCountIs(0)))), changeTo(cat(access(o, cat(to)), "()")), cat("use '", to, "' instead of '", from, "'")); }; renameMethod(<classes>, "count", "size"); renameMethod(<classes>, "length", "size"); except that the on() matcher has been replaced by one that doesn't ignoreParens(). a.k.a qt-port-to-std-compatible-api V5 with config Scope: 'Container'. Added two NOLINTNEXTLINEs in tst_qbitarray and tst_qcontiguouscache, to avoid porting calls that explicitly test count(). Change-Id: Icfb8808c2ff4a30187e9935a51cad26987451c22 Reviewed-by: Ivan Solovev <ivan.solovev@qt.io> Reviewed-by: Qt CI Bot <qt_ci_bot@qt-project.org>
372 lines
11 KiB
C++
372 lines
11 KiB
C++
// Copyright (C) 2019 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
|
|
#include <QTest>
|
|
#include <QSignalSpy>
|
|
#include <QFont>
|
|
|
|
#include <qguiapplication.h>
|
|
#include <qevent.h>
|
|
#include <qaction.h>
|
|
#include <qactiongroup.h>
|
|
#include <qpa/qplatformtheme.h>
|
|
|
|
#include <private/qguiapplication_p.h>
|
|
|
|
class tst_QAction : public QObject
|
|
{
|
|
Q_OBJECT
|
|
|
|
public:
|
|
tst_QAction();
|
|
|
|
private slots:
|
|
void cleanup();
|
|
void getSetCheck();
|
|
void setText_data();
|
|
void setText();
|
|
void setIconText_data() { setText_data(); }
|
|
void setIconText();
|
|
void setToolTip_data();
|
|
void setToolTip();
|
|
#if QT_CONFIG(shortcut)
|
|
void setStandardKeys();
|
|
void task200823_tooltip();
|
|
#endif
|
|
void task229128TriggeredSignalWithoutActiongroup();
|
|
void setData();
|
|
void setEnabledSetVisible();
|
|
void setCheckabledSetChecked();
|
|
|
|
private:
|
|
const int m_keyboardScheme;
|
|
};
|
|
|
|
tst_QAction::tst_QAction()
|
|
: m_keyboardScheme(QGuiApplicationPrivate::platformTheme()->themeHint(QPlatformTheme::KeyboardScheme).toInt())
|
|
{
|
|
}
|
|
|
|
void tst_QAction::cleanup()
|
|
{
|
|
QVERIFY(QGuiApplication::topLevelWindows().isEmpty());
|
|
}
|
|
|
|
// Testing get/set functions
|
|
void tst_QAction::getSetCheck()
|
|
{
|
|
QAction obj1(nullptr);
|
|
auto var1 = new QActionGroup(nullptr);
|
|
obj1.setActionGroup(var1);
|
|
QCOMPARE(obj1.actionGroup(), var1);
|
|
obj1.setActionGroup(nullptr);
|
|
QCOMPARE(obj1.actionGroup(), nullptr);
|
|
delete var1;
|
|
|
|
QCOMPARE(obj1.isCheckable(), false);
|
|
QCOMPARE(obj1.isChecked(), false);
|
|
|
|
obj1.setCheckable(true);
|
|
QCOMPARE(obj1.isCheckable(), true);
|
|
obj1.setChecked(true);
|
|
QCOMPARE(obj1.isChecked(), true);
|
|
|
|
obj1.setCheckable(false);
|
|
QCOMPARE(obj1.isCheckable(), false);
|
|
QCOMPARE(obj1.isChecked(), false);
|
|
|
|
QCOMPARE(obj1.isEnabled(), true);
|
|
obj1.setEnabled(false);
|
|
QCOMPARE(obj1.isEnabled(), false);
|
|
|
|
QCOMPARE(obj1.icon(), QIcon());
|
|
QIcon themeIcon = QIcon::fromTheme("messagebox_info");
|
|
if (!themeIcon.isNull()) {
|
|
obj1.setIcon(themeIcon);
|
|
QCOMPARE(obj1.icon(), themeIcon);
|
|
}
|
|
|
|
QString statusTip("StatusTip");
|
|
obj1.setStatusTip(statusTip);
|
|
QCOMPARE(obj1.statusTip(), statusTip);
|
|
|
|
QString whatsThis("WhatsThis");
|
|
obj1.setWhatsThis(whatsThis);
|
|
QCOMPARE(obj1.whatsThis(), whatsThis);
|
|
|
|
QFont font;
|
|
font.setBold(true);
|
|
obj1.setFont(font);
|
|
QCOMPARE(obj1.font(), font);
|
|
|
|
#if QT_CONFIG(shortcut)
|
|
QCOMPARE(obj1.shortcut(), QKeySequence());
|
|
QKeySequence quit(Qt::CTRL | Qt::Key_Q);
|
|
obj1.setShortcut(quit);
|
|
QCOMPARE(obj1.shortcut(), quit);
|
|
|
|
QCOMPARE(obj1.shortcutContext(), Qt::WindowShortcut);
|
|
obj1.setShortcutContext(Qt::ApplicationShortcut);
|
|
QCOMPARE(obj1.shortcutContext(), Qt::ApplicationShortcut);
|
|
|
|
QCOMPARE(obj1.autoRepeat(), true);
|
|
obj1.setAutoRepeat(false);
|
|
QCOMPARE(obj1.autoRepeat(), false);
|
|
#endif
|
|
|
|
QCOMPARE(obj1.isVisible(), true);
|
|
obj1.setVisible(false);
|
|
QCOMPARE(obj1.isChecked(), false);
|
|
|
|
QCOMPARE(obj1.menuRole(), QAction::TextHeuristicRole);
|
|
obj1.setMenuRole(QAction::PreferencesRole);
|
|
QCOMPARE(obj1.menuRole(), QAction::PreferencesRole);
|
|
|
|
// default value for those depends on application attributes and/or style
|
|
obj1.setIconVisibleInMenu(true);
|
|
QCOMPARE(obj1.isIconVisibleInMenu(), true);
|
|
obj1.setIconVisibleInMenu(false);
|
|
QCOMPARE(obj1.isIconVisibleInMenu(), false);
|
|
|
|
obj1.setShortcutVisibleInContextMenu(true);
|
|
QCOMPARE(obj1.isShortcutVisibleInContextMenu(), true);
|
|
obj1.setShortcutVisibleInContextMenu(false);
|
|
QCOMPARE(obj1.isShortcutVisibleInContextMenu(), false);
|
|
|
|
QCOMPARE(obj1.priority(), QAction::NormalPriority);
|
|
obj1.setPriority(QAction::LowPriority);
|
|
QCOMPARE(obj1.priority(), QAction::LowPriority);
|
|
}
|
|
|
|
void tst_QAction::setText_data()
|
|
{
|
|
QTest::addColumn<QString>("text");
|
|
QTest::addColumn<QString>("iconText");
|
|
QTest::addColumn<QString>("textFromIconText");
|
|
|
|
//next we fill it with data
|
|
QTest::newRow("Normal") << "Action" << "Action" << "Action";
|
|
QTest::newRow("Ampersand") << "Search && Destroy" << "Search & Destroy" << "Search && Destroy";
|
|
QTest::newRow("Mnemonic and ellipsis") << "O&pen File ..." << "Open File" << "Open File";
|
|
}
|
|
|
|
void tst_QAction::setText()
|
|
{
|
|
QFETCH(QString, text);
|
|
|
|
QAction action(nullptr);
|
|
action.setText(text);
|
|
|
|
QCOMPARE(action.text(), text);
|
|
|
|
QFETCH(QString, iconText);
|
|
QCOMPARE(action.iconText(), iconText);
|
|
}
|
|
|
|
void tst_QAction::setIconText()
|
|
{
|
|
QFETCH(QString, iconText);
|
|
|
|
QAction action(nullptr);
|
|
action.setIconText(iconText);
|
|
QCOMPARE(action.iconText(), iconText);
|
|
|
|
QFETCH(QString, textFromIconText);
|
|
QCOMPARE(action.text(), textFromIconText);
|
|
}
|
|
|
|
void tst_QAction::setToolTip_data()
|
|
{
|
|
QTest::addColumn<QByteArrayList>("properties");
|
|
QTest::addColumn<QStringList>("values");
|
|
QTest::addColumn<QStringList>("expectedToolTips");
|
|
|
|
QTest::newRow("Tooltip")
|
|
<< QByteArrayList{"toolTip", "toolTip", "toolTip"}
|
|
<< QStringList{"ToolTip", "Tool&Tip", "Tool && Tip"}
|
|
<< QStringList{"ToolTip", "Tool&Tip", "Tool && Tip"};
|
|
QTest::newRow("Only text")
|
|
<< QByteArrayList{"text", "text", "text", "text", "toolTip", "toolTip"}
|
|
<< QStringList {"Text", "Te&xt2", "Find && Replace", "O&pen File...", "ToolTip", QString()}
|
|
<< QStringList{"Text", "Text2", "Find & Replace", "Open File", "ToolTip", "Open File"};
|
|
QTest::newRow("Only iconText")
|
|
<< QByteArrayList{"iconText", "iconText", "iconText", "toolTip", "toolTip"}
|
|
<< QStringList{"Icon Text", "Find && Replace", "Icon&Text", "ToolTip", QString()}
|
|
<< QStringList{"Icon Text", "Find & Replace", "IconText", "ToolTip", "IconText"};
|
|
QTest::newRow("Text and iconText")
|
|
<< QByteArrayList{"iconText", "text", "iconText", "text", "iconText"}
|
|
<< QStringList{"Icon", "Text", "Icon2", QString(), "Icon3"}
|
|
<< QStringList{"Icon", "Text", "Text", "Icon2", "Icon3"};
|
|
QTest::newRow("All and nothing")
|
|
<< QByteArrayList{"text", "iconText", "toolTip", "text", "iconText", "toolTip", "text", "text", "iconText", "iconText", "text"}
|
|
<< QStringList {"Te&xt", "I&&con", "ToolTip", "Text", "Icon", QString(), "Te&&xt2", QString(), "I&&con2", QString(), "Text"}
|
|
<< QStringList {"Text", "Text", "ToolTip", "ToolTip", "ToolTip", "Text", "Te&xt2", "Icon", "I&con2", QString(), "Text"};
|
|
}
|
|
|
|
void tst_QAction::setToolTip()
|
|
{
|
|
QFETCH(QByteArrayList, properties);
|
|
QFETCH(QStringList, values);
|
|
QFETCH(QStringList, expectedToolTips);
|
|
|
|
QCOMPARE(properties.size(), values.size());
|
|
QCOMPARE(properties.size(), expectedToolTips.size());
|
|
|
|
QAction action(nullptr);
|
|
for (int i = 0; i < properties.size(); ++i) {
|
|
const auto property = properties.at(i);
|
|
const auto value = values.at(i);
|
|
const auto expectedToolTip = expectedToolTips.at(i);
|
|
action.setProperty(property, value);
|
|
QCOMPARE(action.toolTip(), expectedToolTip);
|
|
}
|
|
}
|
|
|
|
#if QT_CONFIG(shortcut)
|
|
|
|
//basic testing of standard keys
|
|
void tst_QAction::setStandardKeys()
|
|
{
|
|
QAction act(nullptr);
|
|
act.setShortcut(QKeySequence("CTRL+L"));
|
|
QList<QKeySequence> list;
|
|
act.setShortcuts(list);
|
|
act.setShortcuts(QKeySequence::Copy);
|
|
QCOMPARE(act.shortcut(), act.shortcuts().constFirst());
|
|
|
|
QList<QKeySequence> expected;
|
|
const QKeySequence ctrlC = QKeySequence(QStringLiteral("CTRL+C"));
|
|
const QKeySequence ctrlInsert = QKeySequence(QStringLiteral("CTRL+INSERT"));
|
|
switch (m_keyboardScheme) {
|
|
case QPlatformTheme::MacKeyboardScheme:
|
|
expected << ctrlC;
|
|
break;
|
|
case QPlatformTheme::WindowsKeyboardScheme:
|
|
expected << ctrlC << ctrlInsert;
|
|
break;
|
|
default: // X11
|
|
expected << ctrlC << ctrlInsert << QKeySequence(QStringLiteral("F16"));
|
|
break;
|
|
}
|
|
|
|
QCOMPARE(act.shortcuts(), expected);
|
|
}
|
|
|
|
void tst_QAction::task200823_tooltip()
|
|
{
|
|
const QScopedPointer<QAction> action(new QAction("foo", nullptr));
|
|
QString shortcut("ctrl+o");
|
|
action->setShortcut(shortcut);
|
|
|
|
// we want a non-standard tooltip that shows the shortcut
|
|
action->setToolTip(action->text() + QLatin1String(" (") + action->shortcut().toString() + QLatin1Char(')'));
|
|
|
|
QString ref = QLatin1String("foo (") + QKeySequence(shortcut).toString() + QLatin1Char(')');
|
|
QCOMPARE(action->toolTip(), ref);
|
|
}
|
|
|
|
#endif // QT_CONFIG(shortcut)
|
|
|
|
void tst_QAction::task229128TriggeredSignalWithoutActiongroup()
|
|
{
|
|
// test without a group
|
|
const QScopedPointer<QAction> actionWithoutGroup(new QAction("Test", nullptr));
|
|
QSignalSpy spyWithoutGroup(actionWithoutGroup.data(), QOverload<bool>::of(&QAction::triggered));
|
|
QCOMPARE(spyWithoutGroup.size(), 0);
|
|
actionWithoutGroup->trigger();
|
|
// signal should be emitted
|
|
QCOMPARE(spyWithoutGroup.size(), 1);
|
|
|
|
// it is now a checkable checked action
|
|
actionWithoutGroup->setCheckable(true);
|
|
actionWithoutGroup->setChecked(true);
|
|
spyWithoutGroup.clear();
|
|
QCOMPARE(spyWithoutGroup.size(), 0);
|
|
actionWithoutGroup->trigger();
|
|
// signal should be emitted
|
|
QCOMPARE(spyWithoutGroup.size(), 1);
|
|
}
|
|
|
|
void tst_QAction::setData() // QTBUG-62006
|
|
{
|
|
QAction act(nullptr);
|
|
QSignalSpy spy(&act, &QAction::changed);
|
|
QCOMPARE(act.data(), QVariant());
|
|
QCOMPARE(spy.size(), 0);
|
|
act.setData(QVariant());
|
|
QCOMPARE(spy.size(), 0);
|
|
|
|
act.setData(-1);
|
|
QCOMPARE(spy.size(), 1);
|
|
act.setData(-1);
|
|
QCOMPARE(spy.size(), 1);
|
|
}
|
|
|
|
void tst_QAction::setEnabledSetVisible()
|
|
{
|
|
QAction action(nullptr);
|
|
QSignalSpy spy(&action, &QAction::enabledChanged);
|
|
QVERIFY(action.isEnabled());
|
|
QVERIFY(action.isVisible());
|
|
QCOMPARE(spy.size(), 0);
|
|
action.setVisible(false);
|
|
QVERIFY(!action.isEnabled());
|
|
QVERIFY(!action.isVisible());
|
|
QCOMPARE(spy.size(), 1);
|
|
action.setEnabled(false);
|
|
QVERIFY(!action.isEnabled());
|
|
QVERIFY(!action.isVisible());
|
|
QCOMPARE(spy.size(), 1);
|
|
action.setVisible(true);
|
|
QVERIFY(!action.isEnabled());
|
|
QVERIFY(action.isVisible());
|
|
QCOMPARE(spy.size(), 1);
|
|
action.resetEnabled();
|
|
QVERIFY(action.isEnabled());
|
|
QCOMPARE(spy.size(), 2);
|
|
}
|
|
|
|
void tst_QAction::setCheckabledSetChecked()
|
|
{
|
|
QAction action(nullptr);
|
|
QSignalSpy changedSpy(&action, &QAction::changed);
|
|
QSignalSpy checkedSpy(&action, &QAction::toggled);
|
|
QSignalSpy checkableSpy(&action, &QAction::checkableChanged);
|
|
QVERIFY(!action.isCheckable());
|
|
QVERIFY(!action.isChecked());
|
|
QCOMPARE(changedSpy.size(), 0);
|
|
QCOMPARE(checkedSpy.size(), 0);
|
|
QCOMPARE(checkableSpy.size(), 0);
|
|
|
|
action.setCheckable(true);
|
|
QVERIFY(action.isCheckable());
|
|
QVERIFY(!action.isChecked());
|
|
QCOMPARE(changedSpy.size(), 1);
|
|
QCOMPARE(checkedSpy.size(), 0);
|
|
QCOMPARE(checkableSpy.size(), 1);
|
|
|
|
action.setChecked(true);
|
|
QVERIFY(action.isCheckable());
|
|
QVERIFY(action.isChecked());
|
|
QCOMPARE(changedSpy.size(), 2);
|
|
QCOMPARE(checkedSpy.size(), 1);
|
|
QCOMPARE(checkableSpy.size(), 1);
|
|
|
|
action.setCheckable(false);
|
|
QVERIFY(!action.isCheckable());
|
|
QVERIFY(!action.isChecked());
|
|
QCOMPARE(changedSpy.size(), 3);
|
|
QCOMPARE(checkedSpy.size(), 2);
|
|
QCOMPARE(checkableSpy.size(), 2);
|
|
|
|
action.setCheckable(true);
|
|
QVERIFY(action.isCheckable());
|
|
QVERIFY(action.isChecked());
|
|
QCOMPARE(changedSpy.size(), 4);
|
|
QCOMPARE(checkedSpy.size(), 3);
|
|
QCOMPARE(checkableSpy.size(), 3);
|
|
}
|
|
|
|
QTEST_MAIN(tst_QAction)
|
|
#include "tst_qaction.moc"
|