2020-08-02 22:22:19 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
2013-07-10 09:41:16 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-FileCopyrightText: 2013 Martin Gräßlin <mgraesslin@kde.org>
|
2013-07-10 09:41:16 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
2013-07-10 09:41:16 +00:00
|
|
|
// own
|
|
|
|
#include "globalshortcuts.h"
|
2022-03-23 10:13:38 +00:00
|
|
|
// config
|
|
|
|
#include <config-kwin.h>
|
2013-07-10 09:41:16 +00:00
|
|
|
// kwin
|
2021-03-30 13:32:32 +00:00
|
|
|
#include "gestures.h"
|
2021-03-29 23:51:15 +00:00
|
|
|
#include "kwinglobals.h"
|
2015-06-26 08:51:11 +00:00
|
|
|
#include "main.h"
|
2022-01-18 08:35:52 +00:00
|
|
|
#include "utils/common.h"
|
2013-07-10 09:41:16 +00:00
|
|
|
// KDE
|
2015-06-26 11:47:08 +00:00
|
|
|
#include <KGlobalAccel/private/kglobalaccel_interface.h>
|
2021-03-30 13:32:32 +00:00
|
|
|
#include <KGlobalAccel/private/kglobalacceld.h>
|
2013-07-10 09:41:16 +00:00
|
|
|
// Qt
|
|
|
|
#include <QAction>
|
2022-03-23 10:13:38 +00:00
|
|
|
// system
|
2021-04-19 04:42:46 +00:00
|
|
|
#include <signal.h>
|
2022-03-23 10:13:38 +00:00
|
|
|
#include <variant>
|
2013-07-10 09:41:16 +00:00
|
|
|
|
|
|
|
namespace KWin
|
|
|
|
{
|
2021-03-30 13:32:32 +00:00
|
|
|
GlobalShortcut::GlobalShortcut(Shortcut &&sc, QAction *action)
|
2022-07-04 00:09:23 +00:00
|
|
|
: m_shortcut(std::move(sc))
|
2021-03-30 13:32:32 +00:00
|
|
|
, m_action(action)
|
2021-03-29 23:51:15 +00:00
|
|
|
{
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
2013-07-10 09:41:16 +00:00
|
|
|
GlobalShortcut::~GlobalShortcut()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2021-03-30 13:32:32 +00:00
|
|
|
QAction *GlobalShortcut::action() const
|
2013-07-10 09:41:16 +00:00
|
|
|
{
|
2021-03-29 23:51:15 +00:00
|
|
|
return m_action;
|
2013-07-10 09:41:16 +00:00
|
|
|
}
|
|
|
|
|
2021-03-29 23:51:15 +00:00
|
|
|
void GlobalShortcut::invoke() const
|
2013-07-14 20:52:58 +00:00
|
|
|
{
|
2021-03-29 23:51:15 +00:00
|
|
|
QMetaObject::invokeMethod(m_action, "trigger", Qt::QueuedConnection);
|
2013-07-14 20:52:58 +00:00
|
|
|
}
|
|
|
|
|
2021-03-30 13:32:32 +00:00
|
|
|
const Shortcut &GlobalShortcut::shortcut() const
|
2013-07-10 09:41:16 +00:00
|
|
|
{
|
2021-03-29 23:51:15 +00:00
|
|
|
return m_shortcut;
|
2013-07-10 09:41:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
GlobalShortcutsManager::GlobalShortcutsManager(QObject *parent)
|
|
|
|
: QObject(parent)
|
2022-03-16 18:00:24 +00:00
|
|
|
, m_touchpadGestureRecognizer(new GestureRecognizer(this))
|
|
|
|
, m_touchscreenGestureRecognizer(new GestureRecognizer(this))
|
2013-07-10 09:41:16 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2013-07-14 20:52:58 +00:00
|
|
|
GlobalShortcutsManager::~GlobalShortcutsManager()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-06-26 11:47:08 +00:00
|
|
|
void GlobalShortcutsManager::init()
|
|
|
|
{
|
|
|
|
if (kwinApp()->shouldUseWaylandForCompositing()) {
|
2015-07-06 02:50:20 +00:00
|
|
|
qputenv("KGLOBALACCELD_PLATFORM", QByteArrayLiteral("org.kde.kwin"));
|
2022-07-05 11:43:25 +00:00
|
|
|
m_kglobalAccel = std::make_unique<KGlobalAccelD>();
|
2015-06-26 11:47:08 +00:00
|
|
|
if (!m_kglobalAccel->init()) {
|
|
|
|
qCDebug(KWIN_CORE) << "Init of kglobalaccel failed";
|
2022-07-05 11:43:25 +00:00
|
|
|
m_kglobalAccel.reset();
|
2015-06-26 11:47:08 +00:00
|
|
|
} else {
|
|
|
|
qCDebug(KWIN_CORE) << "KGlobalAcceld inited";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-29 23:51:15 +00:00
|
|
|
void GlobalShortcutsManager::objectDeleted(QObject *object)
|
2013-07-10 09:41:16 +00:00
|
|
|
{
|
2021-03-29 23:51:15 +00:00
|
|
|
auto it = m_shortcuts.begin();
|
|
|
|
while (it != m_shortcuts.end()) {
|
|
|
|
if (it->action() == object) {
|
|
|
|
it = m_shortcuts.erase(it);
|
|
|
|
} else {
|
|
|
|
++it;
|
2013-07-10 09:41:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-04 00:09:23 +00:00
|
|
|
bool GlobalShortcutsManager::addIfNotExists(GlobalShortcut sc, GestureDeviceType device)
|
2013-07-14 20:52:58 +00:00
|
|
|
{
|
2021-05-13 16:41:39 +00:00
|
|
|
for (const auto &cs : qAsConst(m_shortcuts)) {
|
2021-03-29 23:51:15 +00:00
|
|
|
if (sc.shortcut() == cs.shortcut()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2013-07-14 20:52:58 +00:00
|
|
|
|
2022-07-04 00:09:23 +00:00
|
|
|
// Register if GestureSHortcut
|
|
|
|
const auto &recognizer = device == GestureDeviceType::Touchpad ? m_touchpadGestureRecognizer : m_touchscreenGestureRecognizer;
|
|
|
|
if (std::holds_alternative<GestureShortcut>(sc.shortcut())) {
|
|
|
|
const GestureShortcut *shortcut = &std::get<GestureShortcut>(sc.shortcut());
|
|
|
|
if (shortcut->swipeGesture) {
|
|
|
|
recognizer->registerSwipeGesture(shortcut->swipeGesture.get());
|
|
|
|
} else {
|
|
|
|
recognizer->registerPinchGesture(shortcut->pinchGesture.get());
|
|
|
|
}
|
2013-07-16 16:50:10 +00:00
|
|
|
}
|
2021-03-29 23:51:15 +00:00
|
|
|
connect(sc.action(), &QAction::destroyed, this, &GlobalShortcutsManager::objectDeleted);
|
|
|
|
m_shortcuts.push_back(std::move(sc));
|
|
|
|
return true;
|
2013-07-16 16:50:10 +00:00
|
|
|
}
|
|
|
|
|
2013-07-14 20:52:58 +00:00
|
|
|
void GlobalShortcutsManager::registerPointerShortcut(QAction *action, Qt::KeyboardModifiers modifiers, Qt::MouseButtons pointerButtons)
|
|
|
|
{
|
2021-03-30 13:32:32 +00:00
|
|
|
addIfNotExists(GlobalShortcut(PointerButtonShortcut{modifiers, pointerButtons}, action));
|
2013-07-14 20:52:58 +00:00
|
|
|
}
|
|
|
|
|
2013-07-15 09:26:56 +00:00
|
|
|
void GlobalShortcutsManager::registerAxisShortcut(QAction *action, Qt::KeyboardModifiers modifiers, PointerAxisDirection axis)
|
|
|
|
{
|
2021-03-30 13:32:32 +00:00
|
|
|
addIfNotExists(GlobalShortcut(PointerAxisShortcut{modifiers, axis}, action));
|
2013-07-15 09:26:56 +00:00
|
|
|
}
|
|
|
|
|
2022-09-11 10:14:08 +00:00
|
|
|
void GlobalShortcutsManager::registerGesture(GestureDeviceType device, GestureDirection direction, uint fingerCount, QAction *onUp, std::function<void(qreal)> progressCallback)
|
2022-07-04 00:09:23 +00:00
|
|
|
{
|
|
|
|
// Create and setup the GestureShortcut
|
|
|
|
GestureShortcut shortcut{device, direction};
|
|
|
|
if (isSwipeDirection(direction)) {
|
|
|
|
std::unique_ptr<SwipeGesture> gesture = std::make_unique<SwipeGesture>();
|
|
|
|
gesture->addFingerCount(fingerCount);
|
2022-09-11 10:14:10 +00:00
|
|
|
gesture->setDirection(direction);
|
2022-09-11 10:14:08 +00:00
|
|
|
gesture->setMinimumDelta(QSizeF(200, 200));
|
|
|
|
connect(gesture.get(), &SwipeGesture::progress, progressCallback);
|
2022-07-04 00:09:23 +00:00
|
|
|
connect(gesture.get(), &Gesture::triggered, onUp, &QAction::trigger, Qt::QueuedConnection);
|
|
|
|
connect(gesture.get(), &Gesture::cancelled, onUp, &QAction::trigger, Qt::QueuedConnection);
|
|
|
|
shortcut.swipeGesture = std::move(gesture);
|
|
|
|
} else if (isPinchDirection(direction)) {
|
|
|
|
std::unique_ptr<PinchGesture> gesture = std::make_unique<PinchGesture>();
|
|
|
|
gesture->addFingerCount(fingerCount);
|
2022-09-11 10:14:10 +00:00
|
|
|
gesture->setDirection(direction);
|
2022-09-11 10:14:08 +00:00
|
|
|
connect(gesture.get(), &PinchGesture::progress, progressCallback);
|
2022-07-04 00:09:23 +00:00
|
|
|
connect(gesture.get(), &Gesture::triggered, onUp, &QAction::trigger, Qt::QueuedConnection);
|
|
|
|
connect(gesture.get(), &Gesture::cancelled, onUp, &QAction::trigger, Qt::QueuedConnection);
|
|
|
|
shortcut.pinchGesture = std::move(gesture);
|
|
|
|
}
|
2013-07-10 09:41:16 +00:00
|
|
|
|
2022-07-04 00:09:23 +00:00
|
|
|
addIfNotExists(GlobalShortcut(std::move(shortcut), onUp), device);
|
2022-03-16 18:00:24 +00:00
|
|
|
}
|
|
|
|
|
2022-09-11 10:13:37 +00:00
|
|
|
void GlobalShortcutsManager::forceRegisterTouchscreenSwipe(QAction *onUp, std::function<void(qreal)> progressCallback, GestureDirection direction, uint fingerCount)
|
2022-03-16 18:00:24 +00:00
|
|
|
{
|
2022-07-04 00:09:23 +00:00
|
|
|
std::unique_ptr<SwipeGesture> gesture = std::make_unique<SwipeGesture>();
|
|
|
|
gesture->addFingerCount(fingerCount);
|
|
|
|
gesture->setDirection(direction);
|
2022-09-11 10:14:08 +00:00
|
|
|
gesture->setMinimumDelta(QSizeF(200, 200));
|
|
|
|
connect(gesture.get(), &SwipeGesture::progress, progressCallback);
|
2022-07-04 00:09:23 +00:00
|
|
|
connect(gesture.get(), &Gesture::triggered, onUp, &QAction::trigger, Qt::QueuedConnection);
|
|
|
|
connect(gesture.get(), &Gesture::cancelled, onUp, &QAction::trigger, Qt::QueuedConnection);
|
2021-04-19 04:42:46 +00:00
|
|
|
|
2022-09-11 10:14:08 +00:00
|
|
|
GlobalShortcut shortcut{GestureShortcut{GestureDeviceType::Touchscreen, direction, std::move(gesture)}, onUp};
|
2022-05-27 15:26:24 +00:00
|
|
|
const auto it = std::find_if(m_shortcuts.begin(), m_shortcuts.end(), [&shortcut](const auto &s) {
|
|
|
|
return shortcut.shortcut() == s.shortcut();
|
|
|
|
});
|
|
|
|
if (it != m_shortcuts.end()) {
|
|
|
|
m_shortcuts.erase(it);
|
|
|
|
}
|
2022-09-11 10:14:08 +00:00
|
|
|
m_touchscreenGestureRecognizer->registerSwipeGesture(gesture.get());
|
2022-05-27 15:26:24 +00:00
|
|
|
connect(shortcut.action(), &QAction::destroyed, this, &GlobalShortcutsManager::objectDeleted);
|
|
|
|
m_shortcuts.push_back(std::move(shortcut));
|
|
|
|
}
|
|
|
|
|
2017-07-28 19:31:09 +00:00
|
|
|
bool GlobalShortcutsManager::processKey(Qt::KeyboardModifiers mods, int keyQt)
|
2013-07-14 20:52:58 +00:00
|
|
|
{
|
2015-06-26 11:47:08 +00:00
|
|
|
if (m_kglobalAccelInterface) {
|
2018-04-21 16:35:22 +00:00
|
|
|
if (!keyQt && !mods) {
|
|
|
|
return false;
|
|
|
|
}
|
2021-03-30 13:32:32 +00:00
|
|
|
auto check = [this](Qt::KeyboardModifiers mods, int keyQt) {
|
2016-09-15 07:02:38 +00:00
|
|
|
bool retVal = false;
|
|
|
|
QMetaObject::invokeMethod(m_kglobalAccelInterface,
|
2021-03-30 13:32:32 +00:00
|
|
|
"checkKeyPressed",
|
|
|
|
Qt::DirectConnection,
|
|
|
|
Q_RETURN_ARG(bool, retVal),
|
|
|
|
Q_ARG(int, int(mods) | keyQt));
|
2016-09-15 07:02:38 +00:00
|
|
|
return retVal;
|
|
|
|
};
|
|
|
|
if (check(mods, keyQt)) {
|
|
|
|
return true;
|
|
|
|
} else if (keyQt == Qt::Key_Backtab) {
|
|
|
|
// KGlobalAccel on X11 has some workaround for Backtab
|
|
|
|
// see kglobalaccel/src/runtime/plugins/xcb/kglobalccel_x11.cpp method x11KeyPress
|
|
|
|
// Apparently KKeySequenceWidget captures Shift+Tab instead of Backtab
|
|
|
|
// thus if the key is backtab we should adjust to add shift again and use tab
|
|
|
|
// in addition KWin registers the shortcut incorrectly as Alt+Shift+Backtab
|
|
|
|
// this should be changed to either Alt+Backtab or Alt+Shift+Tab to match KKeySequenceWidget
|
|
|
|
// trying the variants
|
|
|
|
if (check(mods | Qt::ShiftModifier, keyQt)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (check(mods | Qt::ShiftModifier, Qt::Key_Tab)) {
|
2015-06-26 11:47:08 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
2013-07-14 20:52:58 +00:00
|
|
|
}
|
|
|
|
|
2022-03-27 22:43:28 +00:00
|
|
|
bool GlobalShortcutsManager::processKeyRelease(Qt::KeyboardModifiers mods, int keyQt)
|
|
|
|
{
|
|
|
|
if (m_kglobalAccelInterface) {
|
|
|
|
QMetaObject::invokeMethod(m_kglobalAccelInterface,
|
|
|
|
"checkKeyReleased",
|
|
|
|
Qt::DirectConnection,
|
|
|
|
Q_ARG(int, int(mods) | keyQt));
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-03-29 23:51:15 +00:00
|
|
|
template<typename ShortcutKind, typename... Args>
|
2022-07-04 00:09:23 +00:00
|
|
|
bool match(std::vector<GlobalShortcut> &shortcuts, Args... args)
|
2021-03-29 23:51:15 +00:00
|
|
|
{
|
2021-03-30 13:32:32 +00:00
|
|
|
for (auto &sc : shortcuts) {
|
2021-03-29 23:51:15 +00:00
|
|
|
if (std::holds_alternative<ShortcutKind>(sc.shortcut())) {
|
|
|
|
if (std::get<ShortcutKind>(sc.shortcut()) == ShortcutKind{args...}) {
|
|
|
|
sc.invoke();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-03-30 13:32:32 +00:00
|
|
|
// TODO(C++20): use ranges for a nicer way of filtering by shortcut type
|
2013-07-14 20:52:58 +00:00
|
|
|
bool GlobalShortcutsManager::processPointerPressed(Qt::KeyboardModifiers mods, Qt::MouseButtons pointerButtons)
|
|
|
|
{
|
2021-03-29 23:51:15 +00:00
|
|
|
return match<PointerButtonShortcut>(m_shortcuts, mods, pointerButtons);
|
2013-07-14 20:52:58 +00:00
|
|
|
}
|
|
|
|
|
2013-07-15 09:26:56 +00:00
|
|
|
bool GlobalShortcutsManager::processAxis(Qt::KeyboardModifiers mods, PointerAxisDirection axis)
|
|
|
|
{
|
2021-03-29 23:51:15 +00:00
|
|
|
return match<PointerAxisShortcut>(m_shortcuts, mods, axis);
|
2013-07-15 09:26:56 +00:00
|
|
|
}
|
|
|
|
|
2022-07-04 00:09:23 +00:00
|
|
|
void GlobalShortcutsManager::processSwipeStart(GestureDeviceType device, uint fingerCount)
|
2017-03-18 10:00:30 +00:00
|
|
|
{
|
2022-07-04 00:09:23 +00:00
|
|
|
if (device == GestureDeviceType::Touchpad) {
|
2022-03-16 18:00:24 +00:00
|
|
|
m_touchpadGestureRecognizer->startSwipeGesture(fingerCount);
|
|
|
|
} else {
|
|
|
|
m_touchscreenGestureRecognizer->startSwipeGesture(fingerCount);
|
|
|
|
}
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
2022-07-04 00:09:23 +00:00
|
|
|
void GlobalShortcutsManager::processSwipeUpdate(GestureDeviceType device, const QSizeF &delta)
|
2017-03-18 10:00:30 +00:00
|
|
|
{
|
2022-07-04 00:09:23 +00:00
|
|
|
if (device == GestureDeviceType::Touchpad) {
|
2022-03-16 18:00:24 +00:00
|
|
|
m_touchpadGestureRecognizer->updateSwipeGesture(delta);
|
|
|
|
} else {
|
|
|
|
m_touchscreenGestureRecognizer->updateSwipeGesture(delta);
|
|
|
|
}
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
2022-07-04 00:09:23 +00:00
|
|
|
void GlobalShortcutsManager::processSwipeCancel(GestureDeviceType device)
|
2017-03-18 10:00:30 +00:00
|
|
|
{
|
2022-07-04 00:09:23 +00:00
|
|
|
if (device == GestureDeviceType::Touchpad) {
|
2022-03-16 18:00:24 +00:00
|
|
|
m_touchpadGestureRecognizer->cancelSwipeGesture();
|
|
|
|
} else {
|
|
|
|
m_touchscreenGestureRecognizer->cancelSwipeGesture();
|
|
|
|
}
|
2017-03-18 10:00:30 +00:00
|
|
|
}
|
|
|
|
|
2022-07-04 00:09:23 +00:00
|
|
|
void GlobalShortcutsManager::processSwipeEnd(GestureDeviceType device)
|
2017-03-18 10:00:30 +00:00
|
|
|
{
|
2022-07-04 00:09:23 +00:00
|
|
|
if (device == GestureDeviceType::Touchpad) {
|
2022-03-16 18:00:24 +00:00
|
|
|
m_touchpadGestureRecognizer->endSwipeGesture();
|
|
|
|
} else {
|
|
|
|
m_touchscreenGestureRecognizer->endSwipeGesture();
|
|
|
|
}
|
2017-03-18 10:00:30 +00:00
|
|
|
// TODO: cancel on Wayland Seat if one triggered
|
|
|
|
}
|
|
|
|
|
2022-02-02 21:51:39 +00:00
|
|
|
void GlobalShortcutsManager::processPinchStart(uint fingerCount)
|
|
|
|
{
|
2022-03-16 18:00:24 +00:00
|
|
|
m_touchpadGestureRecognizer->startPinchGesture(fingerCount);
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GlobalShortcutsManager::processPinchUpdate(qreal scale, qreal angleDelta, const QSizeF &delta)
|
|
|
|
{
|
2022-03-16 18:00:24 +00:00
|
|
|
m_touchpadGestureRecognizer->updatePinchGesture(scale, angleDelta, delta);
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GlobalShortcutsManager::processPinchCancel()
|
|
|
|
{
|
2022-03-16 18:00:24 +00:00
|
|
|
m_touchpadGestureRecognizer->cancelPinchGesture();
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void GlobalShortcutsManager::processPinchEnd()
|
|
|
|
{
|
2022-03-16 18:00:24 +00:00
|
|
|
m_touchpadGestureRecognizer->endPinchGesture();
|
2022-02-02 21:51:39 +00:00
|
|
|
}
|
|
|
|
|
2022-07-04 00:09:23 +00:00
|
|
|
Gesture *GestureShortcut::gesture() const
|
|
|
|
{
|
|
|
|
if (swipeGesture != nullptr) {
|
|
|
|
return swipeGesture.get();
|
|
|
|
} else {
|
|
|
|
return pinchGesture.get();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-10 09:41:16 +00:00
|
|
|
} // namespace
|