File indexing completed on 2024-09-08 08:00:06

0001 /*
0002     SPDX-FileCopyrightText: 2016 Smith AR <audoban@openmailbox.org>
0003     SPDX-FileCopyrightText: 2016 Michail Vourlakos <mvourlakos@gmail.com>
0004 
0005     SPDX-License-Identifier: GPL-2.0-or-later
0006 */
0007 
0008 #include "globalshortcuts.h"
0009 
0010 // local
0011 #include "modifiertracker.h"
0012 #include "shortcutstracker.h"
0013 #include "../lattecorona.h"
0014 #include "../layout/centrallayout.h"
0015 #include "../layouts/manager.h"
0016 #include "../layouts/synchronizer.h"
0017 #include "../settings/settingsdialog/settingsdialog.h"
0018 #include "../settings/universalsettings.h"
0019 #include "../view/containmentinterface.h"
0020 #include "../view/view.h"
0021 
0022 // C++
0023 #include <array>
0024 
0025 // Qt
0026 #include <QAction>
0027 #include <QDebug>
0028 #include <QQuickItem>
0029 #include <QMetaMethod>
0030 #include <QX11Info>
0031 
0032 // KDE
0033 #include <KActionCollection>
0034 #include <KGlobalAccel>
0035 #include <KLocalizedString>
0036 #include <KPluginMetaData>
0037 
0038 // Plasma
0039 #include <Plasma/Applet>
0040 #include <Plasma/Containment>
0041 
0042 
0043 namespace Latte {
0044 
0045 GlobalShortcuts::GlobalShortcuts(QObject *parent)
0046     : QObject(parent)
0047 {
0048     m_corona = qobject_cast<Latte::Corona *>(parent);
0049     m_modifierTracker = new ShortcutsPart::ModifierTracker(this);
0050     m_shortcutsTracker = new ShortcutsPart::ShortcutsTracker(this);
0051 
0052     if (m_corona) {
0053         init();
0054     }
0055 
0056     m_hideViewsTimer.setSingleShot(true);
0057 
0058     if (QX11Info::isPlatformX11()) {
0059         //in X11 the timer is a poller that checks to see if the modifier keys
0060         //from user global shortcut have been released
0061         m_hideViewsTimer.setInterval(300);
0062     } else {
0063         //on wayland in acting just as simple timer that hides the view afterwards
0064         m_hideViewsTimer.setInterval(2500);
0065     }
0066 
0067     connect(&m_hideViewsTimer, &QTimer::timeout, this, &GlobalShortcuts::hideViewsTimerSlot);
0068 }
0069 
0070 GlobalShortcuts::~GlobalShortcuts()
0071 {
0072     if (m_modifierTracker) {
0073         m_modifierTracker->deleteLater();
0074     }
0075 
0076     if (m_shortcutsTracker) {
0077         m_shortcutsTracker->deleteLater();
0078     }
0079 }
0080 
0081 void GlobalShortcuts::init()
0082 {
0083     KActionCollection *generalActions = new KActionCollection(m_corona);
0084 
0085     //show-hide the main view in the primary screen
0086     QAction *showAction = generalActions->addAction(QStringLiteral("show latte view"));
0087     showAction->setText(i18n("Show Latte Dock/Panel"));
0088     showAction->setShortcut(QKeySequence(Qt::META + '`'));
0089     KGlobalAccel::setGlobalShortcut(showAction, QKeySequence(Qt::META + '`'));
0090     connect(showAction, &QAction::triggered, this, [this]() {
0091         showViews();
0092     });
0093 
0094     //show-cycle between Latte settings windows
0095     QAction *settingsAction = generalActions->addAction(QStringLiteral("show view settings"));
0096     settingsAction->setText(i18n("Cycle Through Dock/Panel Settings Windows"));
0097     KGlobalAccel::setGlobalShortcut(settingsAction, QKeySequence(Qt::META + Qt::Key_A));
0098     connect(settingsAction, &QAction::triggered, this, [this] {
0099         m_modifierTracker->cancelMetaPressed();
0100         showSettings();
0101     });
0102 
0103     //show the layouts editor
0104     QAction *layoutsAction = generalActions->addAction(QStringLiteral("show latte global settings"));
0105     layoutsAction->setText(i18n("Show Latte Global Settings"));
0106     layoutsAction->setShortcut(QKeySequence());
0107     KGlobalAccel::setGlobalShortcut(layoutsAction, QKeySequence());
0108     connect(layoutsAction, &QAction::triggered, this, [this]() {
0109         m_modifierTracker->cancelMetaPressed();
0110         m_corona->layoutsManager()->showLatteSettingsDialog(Settings::Dialog::PreferencesPage, true);
0111     });
0112 
0113     KActionCollection *taskbarActions = new KActionCollection(m_corona);
0114 
0115     //activate actions [1-9]
0116     for (int i = 1; i < 10; ++i) {
0117         const int entryNumber = i;
0118         const Qt::Key key = static_cast<Qt::Key>(Qt::Key_0 + i);
0119 
0120         QAction *action = taskbarActions->addAction(QStringLiteral("activate entry %1").arg(QString::number(entryNumber)));
0121         action->setText(i18n("Activate Entry %1", entryNumber));
0122         action->setShortcut(QKeySequence(Qt::META + key));
0123         KGlobalAccel::setGlobalShortcut(action, QKeySequence(Qt::META + key));
0124         connect(action, &QAction::triggered, this, [this, i] {
0125             // qDebug() << "meta action...";
0126             m_modifierTracker->cancelMetaPressed();
0127             activateEntry(i, static_cast<Qt::Key>(Qt::META));
0128         });
0129     }
0130 
0131     //! Array that is used to register correctly actions for task index>=10 and <19
0132     std::array<Qt::Key, 10> keysAboveTen{  Qt::Key_0,  Qt::Key_Z,  Qt::Key_X, Qt::Key_C, Qt::Key_V, Qt::Key_B, Qt::Key_N, Qt::Key_M, Qt::Key_Comma, Qt::Key_Period };
0133 
0134     //activate actions [10-19]
0135     for (int i = 10; i < 20; ++i) {
0136         QAction *action = taskbarActions->addAction(QStringLiteral("activate entry %1").arg(QString::number(i)));
0137         action->setText(i18n("Activate Entry %1", i));
0138         action->setShortcut(QKeySequence(Qt::META + keysAboveTen[i - 10]));
0139         KGlobalAccel::setGlobalShortcut(action, QKeySequence(Qt::META + keysAboveTen[i - 10]));
0140         connect(action, &QAction::triggered, this, [this, i] {
0141             m_modifierTracker->cancelMetaPressed();
0142             activateEntry(i, static_cast<Qt::Key>(Qt::META));
0143         });
0144     }
0145 
0146     //new instance actions [1-9]
0147     for (int i = 1; i < 10; ++i) {
0148         const int entryNumber = i;
0149         const Qt::Key key = static_cast<Qt::Key>(Qt::Key_0 + i);
0150 
0151         QAction *action = taskbarActions->addAction(QStringLiteral("new instance for entry %1").arg(QString::number(entryNumber)));
0152         action->setText(i18n("New Instance for Entry %1", entryNumber));
0153         KGlobalAccel::setGlobalShortcut(action, QKeySequence(Qt::META + Qt::CTRL + key));
0154         connect(action, &QAction::triggered, this, [this, i] {
0155             // qDebug() << "meta + ctrl + action...";
0156             m_modifierTracker->cancelMetaPressed();
0157             activateEntry(i, static_cast<Qt::Key>(Qt::CTRL));
0158         });
0159     }
0160 
0161     //new instance actions [10-19]
0162     for (int i = 10; i < 20; ++i) {
0163         QAction *action = taskbarActions->addAction(QStringLiteral("new instance for entry %1").arg(QString::number(i)));
0164         action->setText(i18n("New Instance for Entry %1", i));
0165         KGlobalAccel::setGlobalShortcut(action, QKeySequence(Qt::META + Qt::CTRL + keysAboveTen[i - 10]));
0166         connect(action, &QAction::triggered, this, [this, i] {
0167             m_modifierTracker->cancelMetaPressed();
0168             activateEntry(i, static_cast<Qt::Key>(Qt::CTRL));
0169         });
0170     }
0171 
0172     m_singleMetaAction = new QAction(this);
0173     m_singleMetaAction->setShortcut(QKeySequence(Qt::META));
0174 
0175     connect(m_corona->universalSettings(), &UniversalSettings::metaPressAndHoldEnabledChanged , this, [&]() {
0176         if (!m_corona->universalSettings()->metaPressAndHoldEnabled()) {
0177             m_modifierTracker->blockModifierTracking(Qt::Key_Super_L);
0178             m_modifierTracker->blockModifierTracking(Qt::Key_Super_R);
0179         } else {
0180             m_modifierTracker->unblockModifierTracking(Qt::Key_Super_L);
0181             m_modifierTracker->unblockModifierTracking(Qt::Key_Super_R);
0182         }
0183     });
0184 
0185     //display shortcut badges while holding Meta
0186     connect(m_modifierTracker, &ShortcutsPart::ModifierTracker::metaModifierPressed, this, [&]() {
0187         m_metaShowedViews = true;
0188         showViews();
0189     });
0190 }
0191 
0192 ShortcutsPart::ShortcutsTracker *GlobalShortcuts::shortcutsTracker() const
0193 {
0194     return m_shortcutsTracker;
0195 }
0196 
0197 Latte::View *GlobalShortcuts::highestApplicationLauncherView(const QList<Latte::View *> &views) const
0198 {
0199     if (views.isEmpty()) {
0200         return nullptr;
0201     }
0202 
0203     Latte::View *highestPriorityView{nullptr};
0204 
0205     for (const auto view : views) {
0206         if (view->extendedInterface()->applicationLauncherHasGlobalShortcut()) {
0207             highestPriorityView = view;
0208             break;
0209         }
0210     }
0211 
0212     if (!highestPriorityView) {
0213         for (const auto view : views) {
0214             if (view->extendedInterface()->containsApplicationLauncher()) {
0215                 highestPriorityView = view;
0216                 break;
0217             }
0218         }
0219     }
0220 
0221     return highestPriorityView;
0222 }
0223 
0224 //! Activate launcher menu through dbus interface
0225 void GlobalShortcuts::activateLauncherMenu()
0226 {
0227     if (m_metaShowedViews) {
0228         return;
0229     }
0230 
0231     QList<Latte::View *> sortedViews = m_corona->layoutsManager()->synchronizer()->sortedCurrentViews();
0232 
0233     Latte::View *highestPriorityView = highestApplicationLauncherView(sortedViews);
0234 
0235     if (highestPriorityView) {
0236         if (!highestPriorityView->visibility()->isShownFully() && highestPriorityView->extendedInterface()->applicationLauncherInPopup()) {
0237             m_lastInvokedAction = m_singleMetaAction;
0238 
0239             //! wait for view to fully shown
0240             connect(highestPriorityView->visibility(), &Latte::ViewPart::VisibilityManager::isShownFullyChanged, this, [&, highestPriorityView](){
0241                 if (highestPriorityView->visibility()->isShownFully()) {
0242                     highestPriorityView->extendedInterface()->toggleAppletExpanded(highestPriorityView->extendedInterface()->applicationLauncherId());
0243                     //!remove that signal tracking
0244                     disconnect(highestPriorityView->visibility(), &Latte::ViewPart::VisibilityManager::isShownFullyChanged, this, nullptr);
0245                 }
0246             });
0247 
0248             //! That signal is removed from Latte::View only when the popup is really shown!
0249             highestPriorityView->visibility()->addBlockHidingEvent(SHORTCUTBLOCKHIDINGTYPE);
0250 
0251         } else {
0252             highestPriorityView->extendedInterface()->toggleAppletExpanded(highestPriorityView->extendedInterface()->applicationLauncherId());
0253         }
0254     }
0255 }
0256 
0257 bool GlobalShortcuts::activatePlasmaTaskManager(const Latte::View *view, int index, Qt::Key modifier, bool *delayedExecution)
0258 {
0259     bool activation{modifier == static_cast<Qt::Key>(Qt::META)};
0260     bool newInstance{!activation};
0261 
0262     if (!view->visibility()->isShownFully()) {
0263         //! wait for view to fully shown
0264         connect(view->visibility(), &Latte::ViewPart::VisibilityManager::isShownFullyChanged, this, [&, view, index, activation](){
0265             if (view->visibility()->isShownFully()) {
0266                 if (activation) {
0267                     view->extendedInterface()->activatePlasmaTask(index);
0268                 } else {
0269                     view->extendedInterface()->newInstanceForPlasmaTask(index);
0270                 }
0271                 //!remove that signal tracking
0272                 disconnect(view->visibility(), &Latte::ViewPart::VisibilityManager::isShownFullyChanged, this, nullptr);
0273             }
0274         });
0275 
0276         *delayedExecution = true;
0277 
0278         return true;
0279     } else {
0280         *delayedExecution = false;
0281 
0282         return (activation ? view->extendedInterface()->activatePlasmaTask(index) : view->extendedInterface()->newInstanceForPlasmaTask(index));
0283     }
0284 }
0285 
0286 bool GlobalShortcuts::activateLatteEntry(Latte::View *view, int index, Qt::Key modifier, bool *delayedExecution)
0287 {
0288     bool activation{modifier == static_cast<Qt::Key>(Qt::META)};
0289     bool newInstance{!activation};
0290 
0291     int appletId = view->extendedInterface()->appletIdForVisualIndex(index);
0292     bool hasPopUp {(appletId>-1 && view->extendedInterface()->appletIsExpandable(appletId))};
0293 
0294     if (!view->visibility()->isShownFully() && hasPopUp) {
0295         //! wait for view to fully shown
0296         connect(view->visibility(), &Latte::ViewPart::VisibilityManager::isShownFullyChanged, this, [&, view, index, activation](){
0297             if (view->visibility()->isShownFully()) {
0298                 if (activation) {
0299                     view->extendedInterface()->activateEntry(index);
0300                 } else {
0301                     view->extendedInterface()->newInstanceForEntry(index);
0302                 }
0303                 //!remove that signal tracking
0304                 disconnect(view->visibility(), &Latte::ViewPart::VisibilityManager::isShownFullyChanged, this, nullptr);
0305             }
0306         });
0307 
0308         *delayedExecution = true;
0309 
0310         return true;
0311     } else {
0312         *delayedExecution = false;
0313 
0314         return (activation ? view->extendedInterface()->activateEntry(index) : view->extendedInterface()->newInstanceForEntry(index));
0315     }
0316 }
0317 
0318 bool GlobalShortcuts::activateEntryForView(Latte::View *view, int index, Qt::Key modifier)
0319 {
0320     if (!view) {
0321         return false;
0322     }
0323 
0324     bool delayed{false};
0325 
0326     bool executed = ((!view->extendedInterface()->hasLatteTasks() && view->extendedInterface()->hasPlasmaTasks()
0327                       && activatePlasmaTaskManager(view, index, modifier, &delayed))
0328                      || activateLatteEntry(view, index, modifier, &delayed));
0329 
0330     if (executed) {
0331         if (!m_hideViews.contains(view)) {
0332             m_hideViews.append(view);
0333         }
0334 
0335         if (delayed) {
0336             view->visibility()->addBlockHidingEvent(SHORTCUTBLOCKHIDINGTYPE);
0337             m_hideViewsTimer.start();
0338         }
0339 
0340         return true;
0341     }
0342 
0343     return false;
0344 }
0345 
0346 //! Activate task manager entry
0347 void GlobalShortcuts::activateEntry(int index, Qt::Key modifier)
0348 {
0349     m_lastInvokedAction = dynamic_cast<QAction *>(sender());
0350 
0351     QList<Latte::View *> sortedViews = m_corona->layoutsManager()->synchronizer()->sortedCurrentViews();
0352 
0353     Latte::View *highest{nullptr};
0354 
0355     for (const auto view : sortedViews) {
0356         if (view->isPreferredForShortcuts()) {
0357             highest = view;
0358             break;
0359         }
0360     }
0361 
0362     if (highest) {
0363         activateEntryForView(highest, index, modifier);
0364     } else {
0365         for (const auto view : sortedViews) {
0366             if (activateEntryForView(view, index, modifier)) {
0367                 return;
0368             }
0369         }
0370     }
0371 }
0372 
0373 //! update badge for specific view item
0374 void GlobalShortcuts::updateViewItemBadge(QString identifier, QString value)
0375 {
0376     QList<Latte::View *> views = m_corona->layoutsManager()->synchronizer()->currentViews();
0377 
0378     // update badges in all Latte Tasks plasmoids
0379     for (const auto &view : views) {
0380         view->extendedInterface()->updateBadgeForLatteTask(identifier, value);
0381     }
0382 }
0383 
0384 void GlobalShortcuts::showViews()
0385 {
0386     m_lastInvokedAction = dynamic_cast<QAction *>(sender());
0387     if (!m_lastInvokedAction) {
0388         // when holding Meta
0389         m_lastInvokedAction = m_singleMetaAction;
0390     }
0391 
0392     QList<Latte::View *> sortedViews = m_corona->layoutsManager()->synchronizer()->sortedCurrentViews();
0393 
0394     Latte::View *viewWithTasks{nullptr};
0395     Latte::View *viewWithMeta{nullptr};
0396 
0397     bool hasPreferredForShortcutsView{false};
0398 
0399     for (const auto view : sortedViews) {
0400         if (view->isPreferredForShortcuts()) {
0401             hasPreferredForShortcutsView = true;
0402             break;
0403         }
0404     }
0405 
0406     for(const auto view : sortedViews) {
0407         if (!viewWithTasks
0408                 && (!hasPreferredForShortcutsView || view->isPreferredForShortcuts())) {
0409             viewWithTasks = view;
0410             break;
0411         }
0412     }
0413 
0414     if (m_corona->universalSettings()->kwin_metaForwardedToLatte()) {
0415         viewWithMeta = highestApplicationLauncherView(sortedViews);
0416     }
0417 
0418     bool viewFound{false};
0419 
0420     if (!m_hideViewsTimer.isActive()) {
0421         m_hideViews.clear();
0422     }
0423 
0424     //! show view that contains tasks plasmoid
0425     if (viewWithTasks) {
0426         viewFound = true;
0427 
0428         bool showMeta = (viewWithMeta && (viewWithMeta == viewWithTasks));
0429 
0430         viewWithTasks->extendedInterface()->showShortcutBadges(true, showMeta);
0431 
0432         if (!m_hideViewsTimer.isActive()) {
0433             m_hideViews.append(viewWithTasks);
0434             viewWithTasks->visibility()->addBlockHidingEvent(SHORTCUTBLOCKHIDINGTYPE);
0435         }
0436     }
0437 
0438     //! show view that contains only meta
0439     if (viewWithMeta && viewWithMeta != viewWithTasks && viewWithMeta->extendedInterface()->showOnlyMeta()) {
0440         viewFound = true;
0441 
0442         if (!m_hideViewsTimer.isActive()) {
0443             m_hideViews.append(viewWithMeta);
0444             viewWithMeta->visibility()->addBlockHidingEvent(SHORTCUTBLOCKHIDINGTYPE);
0445         }
0446     }
0447 
0448     //! show all the rest views that contain plasma shortcuts
0449     QList<Latte::View *> viewsWithShortcuts = m_corona->layoutsManager()->synchronizer()->currentViewsWithPlasmaShortcuts();
0450 
0451     if (viewsWithShortcuts.count() > 0) {
0452         viewFound = true;
0453 
0454         if (!m_hideViewsTimer.isActive()) {
0455             for(const auto view : viewsWithShortcuts) {
0456                 if (view != viewWithTasks && view != viewWithMeta) {
0457                     if (view->extendedInterface()->showShortcutBadges(false, false)) {
0458                         m_hideViews.append(view);
0459                         view->visibility()->addBlockHidingEvent(SHORTCUTBLOCKHIDINGTYPE);
0460                     }
0461                 }
0462             }
0463         }
0464     }
0465 
0466     if (viewFound) {
0467         if (!m_hideViewsTimer.isActive()) {
0468             m_hideViewsTimer.start();
0469         } else {
0470             m_hideViewsTimer.stop();
0471             hideViewsTimerSlot();
0472         }
0473     }
0474 }
0475 
0476 bool GlobalShortcuts::viewsToHideAreValid()
0477 {
0478     for(const auto view : m_hideViews) {
0479         if (!m_corona->layoutsManager()->synchronizer()->latteViewExists(view)) {
0480             return false;
0481         }
0482 
0483     }
0484 
0485     return true;
0486 }
0487 
0488 void GlobalShortcuts::showSettings()
0489 {
0490     QList<Latte::View *> sortedViews = m_corona->layoutsManager()->synchronizer()->sortedCurrentOriginalViews();
0491 
0492     //! find which is the next view to show its settings
0493     if (sortedViews.count() > 0) {
0494         int openSettings = -1;
0495 
0496         //! find last view that showed its config view
0497         for (int i = 0; i < sortedViews.size(); ++i) {
0498             for (auto currentLayout : m_corona->layoutsManager()->currentLayouts()) {
0499                 if (sortedViews[i] == currentLayout->lastConfigViewFor()) {
0500                     openSettings = i;
0501                     break;
0502                 }
0503             }
0504 
0505             if (openSettings >= 0) {
0506                 break;
0507             }
0508         }
0509 
0510         if (openSettings >= 0 && sortedViews.count() > 1) {
0511             //! check if there is a view with opened settings window
0512             openSettings = sortedViews[openSettings]->settingsWindowIsShown() ? openSettings + 1 : openSettings;
0513 
0514             if (openSettings >= sortedViews.size()) {
0515                 openSettings = 0;
0516             }
0517 
0518             sortedViews[openSettings]->showSettingsWindow();
0519         } else {
0520             sortedViews[0]->showSettingsWindow();
0521         }
0522     }
0523 }
0524 
0525 void GlobalShortcuts::hideViewsTimerSlot()
0526 {
0527     if (!m_lastInvokedAction || m_hideViews.count() == 0) {
0528         return;
0529     }
0530 
0531     auto initParameters = [this]() {
0532         m_lastInvokedAction = Q_NULLPTR;
0533 
0534         if (viewsToHideAreValid()) {
0535             for(const auto latteView : m_hideViews) {
0536                 latteView->visibility()->removeBlockHidingEvent(SHORTCUTBLOCKHIDINGTYPE);
0537                 latteView->extendedInterface()->hideShortcutBadges();
0538 
0539                 if (latteView->visibility()->isSidebar() && !latteView->visibility()->isHidden()) {
0540                     latteView->visibility()->toggleHiddenState();
0541                 }
0542             }
0543         }
0544 
0545         m_hideViews.clear();
0546         m_metaShowedViews = false;
0547     };
0548 
0549     // qDebug() << "MEMORY ::: " << m_hideViews.count() << " _ " << m_viewItemsCalled.count() << " _ " << m_showShortcutBadgesMethods.count();
0550 
0551     if (QX11Info::isPlatformX11()) {
0552         if (!m_modifierTracker->sequenceModifierPressed(m_lastInvokedAction->shortcut())) {
0553             initParameters();
0554 
0555             return;
0556         } else {
0557             m_hideViewsTimer.start();
0558         }
0559     } else {
0560         // TODO: This is needs to be fixed in wayland
0561         initParameters();
0562     }
0563 }
0564 
0565 }
0566 
0567 #include "moc_globalshortcuts.cpp"