File indexing completed on 2024-09-15 12:04:21
0001 #include "kactioncollectiontest.h" 0002 0003 #include <QAction> 0004 #include <QPointer> 0005 #include <QSignalSpy> 0006 #include <QtTestWidgets> 0007 0008 #include <KSharedConfig> 0009 #include <KStandardAction> 0010 0011 void tst_KActionCollection::init() 0012 { 0013 collection = new KActionCollection(static_cast<QObject *>(nullptr)); 0014 } 0015 0016 void tst_KActionCollection::cleanup() 0017 { 0018 delete collection; 0019 collection = nullptr; 0020 } 0021 0022 void tst_KActionCollection::clear() 0023 { 0024 QPointer<QAction> action1 = collection->add<QAction>(QStringLiteral("test1")); 0025 QPointer<QAction> action2 = collection->add<QAction>(QStringLiteral("test2")); 0026 QPointer<QAction> action3 = collection->add<QAction>(QStringLiteral("test3")); 0027 QPointer<QAction> action4 = collection->add<QAction>(QStringLiteral("test4")); 0028 QPointer<QAction> action5 = collection->add<QAction>(QStringLiteral("test5")); 0029 QPointer<QAction> action6 = collection->add<QAction>(QStringLiteral("test6")); 0030 QPointer<QAction> action7 = collection->add<QAction>(QStringLiteral("test7")); 0031 0032 collection->clear(); 0033 QVERIFY(collection->isEmpty()); 0034 0035 QVERIFY(action1.isNull()); 0036 QVERIFY(action2.isNull()); 0037 QVERIFY(action3.isNull()); 0038 QVERIFY(action4.isNull()); 0039 QVERIFY(action5.isNull()); 0040 QVERIFY(action6.isNull()); 0041 QVERIFY(action7.isNull()); 0042 } 0043 0044 void tst_KActionCollection::addStandardActionFunctorSignal() 0045 { 0046 bool received = false; 0047 QAction *a = collection->addAction(KStandardAction::New, QStringLiteral("test"), this, [&]() { 0048 received = true; 0049 }); 0050 a->trigger(); 0051 QVERIFY(received); 0052 delete a; 0053 QVERIFY(collection->isEmpty()); 0054 } 0055 0056 void tst_KActionCollection::deleted() 0057 { 0058 // Delete action -> automatically removed from collection 0059 QAction *a = collection->add<QAction>(QStringLiteral("test")); 0060 delete a; 0061 QVERIFY(collection->isEmpty()); 0062 0063 // Delete action's parent -> automatically removed from collection 0064 QWidget *myWidget = new QWidget(nullptr); 0065 QPointer<QAction> action = new QAction(/*i18n()*/ QStringLiteral("Foo"), myWidget); 0066 collection->addAction(QStringLiteral("foo"), action); 0067 delete myWidget; 0068 QVERIFY(collection->isEmpty()); 0069 QVERIFY(action.isNull()); 0070 0071 // Delete action's parent, but the action was added to another widget with setAssociatedWidget 0072 // and that widget gets deleted first. 0073 myWidget = new QWidget(nullptr); 0074 QWidget *myAssociatedWidget = new QWidget(myWidget); // child widget 0075 action = new QAction(/*i18n()*/ QStringLiteral("Foo"), myWidget); // child action 0076 collection->addAction(QStringLiteral("foo"), action); 0077 collection->addAssociatedWidget(myAssociatedWidget); 0078 QVERIFY(myAssociatedWidget->actions().contains(action)); 0079 delete myAssociatedWidget; // would be done by the line below, but let's make sure it happens first 0080 delete myWidget; 0081 QVERIFY(collection->isEmpty()); 0082 QVERIFY(action.isNull()); 0083 } 0084 0085 void tst_KActionCollection::take() 0086 { 0087 QAction *a = collection->add<QAction>(QStringLiteral("test")); 0088 collection->takeAction(a); 0089 QVERIFY(collection->isEmpty()); 0090 delete a; 0091 } 0092 0093 void tst_KActionCollection::writeSettings() 0094 { 0095 KConfigGroup cfg = clearConfig(); 0096 0097 QList<QKeySequence> defaultShortcut; 0098 defaultShortcut << Qt::Key_A << Qt::Key_B; 0099 0100 QList<QKeySequence> temporaryShortcut; 0101 temporaryShortcut << Qt::Key_C << Qt::Key_D; 0102 0103 QAction *actionWithDifferentShortcut = new QAction(this); 0104 collection->setDefaultShortcuts(actionWithDifferentShortcut, defaultShortcut); 0105 actionWithDifferentShortcut->setShortcuts(temporaryShortcut); 0106 collection->addAction(QStringLiteral("actionWithDifferentShortcut"), actionWithDifferentShortcut); 0107 0108 QAction *immutableAction = new QAction(this); 0109 collection->setDefaultShortcuts(immutableAction, defaultShortcut); 0110 immutableAction->setShortcuts(temporaryShortcut); 0111 collection->setShortcutsConfigurable(immutableAction, false); 0112 collection->addAction(QStringLiteral("immutableAction"), immutableAction); 0113 0114 QAction *actionWithSameShortcut = new QAction(this); 0115 collection->setDefaultShortcuts(actionWithSameShortcut, defaultShortcut); 0116 collection->addAction(QStringLiteral("actionWithSameShortcut"), actionWithSameShortcut); 0117 0118 cfg.writeEntry("actionToDelete", QStringLiteral("Foobar")); 0119 QAction *actionToDelete = new QAction(this); 0120 collection->setDefaultShortcuts(actionToDelete, defaultShortcut); 0121 collection->addAction(QStringLiteral("actionToDelete"), actionToDelete); 0122 0123 collection->writeSettings(&cfg); 0124 0125 QCOMPARE(cfg.readEntry("actionWithDifferentShortcut", QString()), QKeySequence::listToString(actionWithDifferentShortcut->shortcuts())); 0126 QCOMPARE(cfg.readEntry("immutableAction", QString()), QString()); 0127 QCOMPARE(cfg.readEntry("actionWithSameShortcut", QString()), QString()); 0128 QCOMPARE(cfg.readEntry("actionToDelete", QString()), QString()); 0129 0130 qDeleteAll(collection->actions()); 0131 } 0132 0133 void tst_KActionCollection::readSettings() 0134 { 0135 KConfigGroup cfg = clearConfig(); 0136 0137 QList<QKeySequence> defaultShortcut; 0138 defaultShortcut << Qt::Key_A << Qt::Key_B; 0139 0140 QList<QKeySequence> temporaryShortcut; 0141 temporaryShortcut << Qt::Key_C << Qt::Key_D; 0142 0143 cfg.writeEntry("normalAction", QKeySequence::listToString(defaultShortcut)); 0144 cfg.writeEntry("immutable", QKeySequence::listToString(defaultShortcut)); 0145 cfg.writeEntry("empty", QString()); 0146 0147 QAction *normal = new QAction(this); 0148 collection->addAction(QStringLiteral("normalAction"), normal); 0149 0150 QAction *immutable = new QAction(this); 0151 immutable->setShortcuts(temporaryShortcut); 0152 collection->setDefaultShortcuts(immutable, temporaryShortcut); 0153 collection->setShortcutsConfigurable(immutable, false); 0154 collection->addAction(QStringLiteral("immutable"), immutable); 0155 0156 QAction *empty = new QAction(this); 0157 collection->addAction(QStringLiteral("empty"), empty); 0158 collection->setDefaultShortcuts(empty, defaultShortcut); 0159 empty->setShortcuts(temporaryShortcut); 0160 QCOMPARE(QKeySequence::listToString(empty->shortcuts()), QKeySequence::listToString(temporaryShortcut)); 0161 0162 collection->readSettings(&cfg); 0163 0164 QCOMPARE(QKeySequence::listToString(normal->shortcuts()), QKeySequence::listToString(defaultShortcut)); 0165 QCOMPARE(QKeySequence::listToString(empty->shortcuts()), QKeySequence::listToString(defaultShortcut)); 0166 0167 QCOMPARE(QKeySequence::listToString(immutable->shortcuts()), QKeySequence::listToString(temporaryShortcut)); 0168 0169 qDeleteAll(collection->actions()); 0170 } 0171 0172 void tst_KActionCollection::insertReplaces1() 0173 { 0174 QAction *a = new QAction(nullptr); 0175 QAction *b = new QAction(nullptr); 0176 0177 collection->addAction(QStringLiteral("a"), a); 0178 QVERIFY(collection->actions().contains(a)); 0179 QVERIFY(collection->action(QStringLiteral("a")) == a); 0180 0181 collection->addAction(QStringLiteral("a"), b); 0182 QVERIFY(!collection->actions().contains(a)); 0183 QVERIFY(collection->actions().contains(b)); 0184 QVERIFY(collection->action(QStringLiteral("a")) == b); 0185 0186 delete a; 0187 delete b; 0188 } 0189 0190 /** 0191 * Check that a action added twice under different names only ends up once in 0192 * the collection 0193 */ 0194 void tst_KActionCollection::insertReplaces2() 0195 { 0196 QAction *a = new QAction(nullptr); 0197 0198 collection->addAction(QStringLiteral("a"), a); 0199 QVERIFY(collection->actions().contains(a)); 0200 QVERIFY(collection->action(QStringLiteral("a")) == a); 0201 0202 // Simple test: Just add it twice 0203 collection->addAction(QStringLiteral("b"), a); 0204 QVERIFY(collection->actions().contains(a)); 0205 QVERIFY(!collection->action(QStringLiteral("a"))); 0206 QVERIFY(collection->action(QStringLiteral("b")) == a); 0207 0208 // Complex text: Mesh with the objectname 0209 a->setObjectName(QStringLiteral("c")); 0210 collection->addAction(QStringLiteral("d"), a); 0211 QVERIFY(collection->actions().contains(a)); 0212 QVERIFY(!collection->action(QStringLiteral("b"))); 0213 QVERIFY(!collection->action(QStringLiteral("c"))); 0214 QVERIFY(collection->action(QStringLiteral("d")) == a); 0215 0216 delete a; 0217 } 0218 0219 KConfigGroup tst_KActionCollection::clearConfig() 0220 { 0221 KSharedConfig::Ptr cfg = KSharedConfig::openConfig(); 0222 cfg->deleteGroup(collection->configGroup()); 0223 return KConfigGroup(cfg, collection->configGroup()); 0224 } 0225 0226 void tst_KActionCollection::testSetShortcuts() 0227 { 0228 QAction *action = new QAction(/*i18n*/ (QStringLiteral("Next Unread &Folder")), this); 0229 collection->addAction(QStringLiteral("go_next_unread_folder"), action); 0230 collection->setDefaultShortcut(action, QKeySequence(Qt::ALT | Qt::Key_Plus)); 0231 QList<QKeySequence> shortcut = action->shortcuts(); 0232 shortcut << QKeySequence(Qt::CTRL | Qt::Key_Plus); 0233 action->setShortcuts(shortcut); 0234 QCOMPARE(QKeySequence::listToString(action->shortcuts()), QStringLiteral("Alt++; Ctrl++")); 0235 0236 // Simpler way: 0237 QList<QKeySequence> shortcut2; 0238 shortcut2 << QKeySequence(Qt::ALT | Qt::Key_Plus) << QKeySequence(Qt::CTRL | Qt::Key_Plus); 0239 QCOMPARE(QKeySequence::listToString(shortcut2), QStringLiteral("Alt++; Ctrl++")); 0240 } 0241 0242 void tst_KActionCollection::implicitStandardActionInsertionUsingCreate() 0243 { 0244 KActionCollection collection(static_cast<QObject *>(nullptr)); 0245 QAction *a = KStandardAction::create(KStandardAction::Undo, qApp, &QCoreApplication::quit, &collection); 0246 QVERIFY(a); 0247 0248 QVERIFY(a->parent() == &collection); 0249 QVERIFY(collection.action(QString::fromLatin1(KStandardAction::name(KStandardAction::Undo))) == a); 0250 } 0251 0252 void tst_KActionCollection::implicitStandardActionInsertionUsingCut() 0253 { 0254 KActionCollection collection(static_cast<QObject *>(nullptr)); 0255 QAction *cut = KStandardAction::cut(&collection); 0256 QAction *a = collection.action(QString::fromLatin1(KStandardAction::name(KStandardAction::Cut))); 0257 QVERIFY(a); 0258 QVERIFY(a == cut); 0259 } 0260 0261 void tst_KActionCollection::shouldEmitSignals() 0262 { 0263 QAction *a = new QAction(nullptr); 0264 QAction *b = new QAction(nullptr); 0265 0266 QSignalSpy insertedSpy(collection, &KActionCollection::inserted); 0267 QSignalSpy changedSpy(collection, &KActionCollection::changed); 0268 0269 // Insert "first" 0270 collection->addAction(QStringLiteral("first"), a); 0271 QVERIFY(collection->actions().contains(a)); 0272 QCOMPARE(insertedSpy.count(), 1); 0273 QCOMPARE(insertedSpy.at(0).at(0).value<QAction *>(), a); 0274 QCOMPARE(changedSpy.count(), 1); 0275 insertedSpy.clear(); 0276 changedSpy.clear(); 0277 0278 // Replace "first" 0279 collection->addAction(QStringLiteral("first"), b); 0280 QVERIFY(!collection->actions().contains(a)); 0281 QVERIFY(collection->actions().contains(b)); 0282 QCOMPARE(insertedSpy.count(), 1); 0283 QCOMPARE(insertedSpy.at(0).at(0).value<QAction *>(), b); 0284 QCOMPARE(changedSpy.count(), 2); // once for removing a, once for inserting b 0285 insertedSpy.clear(); 0286 changedSpy.clear(); 0287 0288 // Insert "second" 0289 collection->addAction(QStringLiteral("second"), a); 0290 QCOMPARE(insertedSpy.count(), 1); 0291 QCOMPARE(insertedSpy.at(0).at(0).value<QAction *>(), a); 0292 QCOMPARE(changedSpy.count(), 1); 0293 insertedSpy.clear(); 0294 changedSpy.clear(); 0295 0296 // Remove and delete "second" (which is a) 0297 collection->removeAction(a); 0298 QCOMPARE(changedSpy.count(), 1); 0299 changedSpy.clear(); 0300 0301 // Delete b directly, should automatically remove it from the collection and emit changed 0302 delete b; 0303 QCOMPARE(changedSpy.count(), 1); 0304 changedSpy.clear(); 0305 } 0306 0307 QTEST_MAIN(tst_KActionCollection) 0308 0309 #include "moc_kactioncollectiontest.cpp"