WingTool/QHotkey/qhotkey.cpp

327 lines
9.9 KiB
C++

#include "qhotkey.h"
#include "qhotkey_p.h"
#include <QAbstractEventDispatcher>
#include <QCoreApplication>
#include <QDebug>
#include <QMetaMethod>
#include <QThread>
Q_LOGGING_CATEGORY(logQHotkey, "QHotkey")
void QHotkey::addGlobalMapping(const QKeySequence &shortcut,
QHotkey::NativeShortcut nativeShortcut) {
QMetaObject::invokeMethod(
QHotkeyPrivate::instance(), "addMappingInvoked", Qt::QueuedConnection,
Q_ARG(Qt::Key, Qt::Key(uint(shortcut[0]) & ~Qt::KeyboardModifierMask)),
Q_ARG(
Qt::KeyboardModifiers,
Qt::KeyboardModifiers(uint(shortcut[0]) & Qt::KeyboardModifierMask)),
Q_ARG(QHotkey::NativeShortcut, nativeShortcut));
}
bool QHotkey::isPlatformSupported() {
return QHotkeyPrivate::isPlatformSupported();
}
QHotkey::QHotkey(QObject *parent)
: QObject(parent), _keyCode(Qt::Key_unknown), _modifiers(Qt::NoModifier),
_registered(false) {}
QHotkey::QHotkey(const QKeySequence &shortcut, bool autoRegister,
QObject *parent)
: QHotkey(parent) {
setShortcut(shortcut, autoRegister);
}
QHotkey::QHotkey(Qt::Key keyCode, Qt::KeyboardModifiers modifiers,
bool autoRegister, QObject *parent)
: QHotkey(parent) {
setShortcut(keyCode, modifiers, autoRegister);
}
QHotkey::QHotkey(QHotkey::NativeShortcut shortcut, bool autoRegister,
QObject *parent)
: QHotkey(parent) {
setNativeShortcut(shortcut, autoRegister);
}
QHotkey::~QHotkey() {
if (_registered)
QHotkeyPrivate::instance()->removeShortcut(this);
}
QKeySequence QHotkey::shortcut() const {
if (_keyCode == Qt::Key_unknown)
return QKeySequence();
return QKeySequence(static_cast<int>(_keyCode | _modifiers));
}
Qt::Key QHotkey::keyCode() const { return _keyCode; }
Qt::KeyboardModifiers QHotkey::modifiers() const { return _modifiers; }
QHotkey::NativeShortcut QHotkey::currentNativeShortcut() const {
return _nativeShortcut;
}
bool QHotkey::isRegistered() const { return _registered; }
bool QHotkey::setShortcut(const QKeySequence &shortcut, bool autoRegister) {
if (shortcut.isEmpty())
return resetShortcut();
if (shortcut.count() > 1) {
qCWarning(logQHotkey,
"Keysequences with multiple shortcuts are not allowed! "
"Only the first shortcut will be used!");
}
return setShortcut(
Qt::Key(uint(shortcut[0]) & ~Qt::KeyboardModifierMask),
Qt::KeyboardModifiers(uint(shortcut[0]) & Qt::KeyboardModifierMask),
autoRegister);
}
bool QHotkey::setShortcut(Qt::Key keyCode, Qt::KeyboardModifiers modifiers,
bool autoRegister) {
if (_registered) {
if (autoRegister) {
if (!QHotkeyPrivate::instance()->removeShortcut(this))
return false;
} else
return false;
}
if (keyCode == Qt::Key_unknown) {
_keyCode = Qt::Key_unknown;
_modifiers = Qt::NoModifier;
_nativeShortcut = NativeShortcut();
return true;
}
_keyCode = keyCode;
_modifiers = modifiers;
_nativeShortcut =
QHotkeyPrivate::instance()->nativeShortcut(keyCode, modifiers);
if (_nativeShortcut.isValid()) {
if (autoRegister)
return QHotkeyPrivate::instance()->addShortcut(this);
return true;
}
qCWarning(logQHotkey) << "Unable to map shortcut to native keys. Key:"
<< keyCode << "Modifiers:" << modifiers;
_keyCode = Qt::Key_unknown;
_modifiers = Qt::NoModifier;
_nativeShortcut = NativeShortcut();
return false;
}
bool QHotkey::resetShortcut() {
if (_registered && !QHotkeyPrivate::instance()->removeShortcut(this)) {
return false;
}
_keyCode = Qt::Key_unknown;
_modifiers = Qt::NoModifier;
_nativeShortcut = NativeShortcut();
return true;
}
bool QHotkey::setNativeShortcut(QHotkey::NativeShortcut nativeShortcut,
bool autoRegister) {
if (_registered) {
if (autoRegister) {
if (!QHotkeyPrivate::instance()->removeShortcut(this))
return false;
} else
return false;
}
if (nativeShortcut.isValid()) {
_keyCode = Qt::Key_unknown;
_modifiers = Qt::NoModifier;
_nativeShortcut = nativeShortcut;
if (autoRegister)
return QHotkeyPrivate::instance()->addShortcut(this);
return true;
}
_keyCode = Qt::Key_unknown;
_modifiers = Qt::NoModifier;
_nativeShortcut = NativeShortcut();
return true;
}
bool QHotkey::setRegistered(bool registered) {
if (_registered && !registered)
return QHotkeyPrivate::instance()->removeShortcut(this);
if (!_registered && registered) {
if (!_nativeShortcut.isValid())
return false;
return QHotkeyPrivate::instance()->addShortcut(this);
}
return true;
}
// ---------- QHotkeyPrivate implementation ----------
QHotkeyPrivate::QHotkeyPrivate() {
Q_ASSERT_X(qApp, Q_FUNC_INFO,
"QHotkey requires QCoreApplication to be instantiated");
qApp->eventDispatcher()->installNativeEventFilter(this);
}
QHotkeyPrivate::~QHotkeyPrivate() {
if (!shortcuts.isEmpty())
qCWarning(logQHotkey)
<< "QHotkeyPrivate destroyed with registered shortcuts!";
if (qApp && qApp->eventDispatcher())
qApp->eventDispatcher()->removeNativeEventFilter(this);
}
QHotkey::NativeShortcut
QHotkeyPrivate::nativeShortcut(Qt::Key keycode,
Qt::KeyboardModifiers modifiers) {
Qt::ConnectionType conType =
(QThread::currentThread() == thread() ? Qt::DirectConnection
: Qt::BlockingQueuedConnection);
QHotkey::NativeShortcut res;
if (!QMetaObject::invokeMethod(this, "nativeShortcutInvoked", conType,
Q_RETURN_ARG(QHotkey::NativeShortcut, res),
Q_ARG(Qt::Key, keycode),
Q_ARG(Qt::KeyboardModifiers, modifiers))) {
return QHotkey::NativeShortcut();
}
return res;
}
bool QHotkeyPrivate::addShortcut(QHotkey *hotkey) {
if (hotkey->_registered)
return false;
Qt::ConnectionType conType =
(QThread::currentThread() == thread() ? Qt::DirectConnection
: Qt::BlockingQueuedConnection);
bool res = false;
if (!QMetaObject::invokeMethod(this, "addShortcutInvoked", conType,
Q_RETURN_ARG(bool, res),
Q_ARG(QHotkey *, hotkey))) {
return false;
}
if (res)
emit hotkey->registeredChanged(true);
return res;
}
bool QHotkeyPrivate::removeShortcut(QHotkey *hotkey) {
if (!hotkey->_registered)
return false;
Qt::ConnectionType conType =
(QThread::currentThread() == thread() ? Qt::DirectConnection
: Qt::BlockingQueuedConnection);
bool res = false;
if (!QMetaObject::invokeMethod(this, "removeShortcutInvoked", conType,
Q_RETURN_ARG(bool, res),
Q_ARG(QHotkey *, hotkey))) {
return false;
}
if (res)
emit hotkey->registeredChanged(false);
return res;
}
void QHotkeyPrivate::activateShortcut(QHotkey::NativeShortcut shortcut) {
QMetaMethod signal = QMetaMethod::fromSignal(&QHotkey::activated);
for (QHotkey *hkey : shortcuts.values(shortcut))
signal.invoke(hkey, Qt::QueuedConnection);
}
void QHotkeyPrivate::releaseShortcut(QHotkey::NativeShortcut shortcut) {
QMetaMethod signal = QMetaMethod::fromSignal(&QHotkey::released);
for (QHotkey *hkey : shortcuts.values(shortcut))
signal.invoke(hkey, Qt::QueuedConnection);
}
void QHotkeyPrivate::addMappingInvoked(Qt::Key keycode,
Qt::KeyboardModifiers modifiers,
QHotkey::NativeShortcut nativeShortcut) {
mapping.insert({keycode, modifiers}, nativeShortcut);
}
bool QHotkeyPrivate::addShortcutInvoked(QHotkey *hotkey) {
QHotkey::NativeShortcut shortcut = hotkey->_nativeShortcut;
if (!shortcuts.contains(shortcut)) {
if (!registerShortcut(shortcut)) {
qCWarning(logQHotkey) << QHotkey::tr("Failed to register %1. Error: %2")
.arg(hotkey->shortcut().toString(), error);
return false;
}
}
shortcuts.insert(shortcut, hotkey);
hotkey->_registered = true;
return true;
}
bool QHotkeyPrivate::removeShortcutInvoked(QHotkey *hotkey) {
QHotkey::NativeShortcut shortcut = hotkey->_nativeShortcut;
if (shortcuts.remove(shortcut, hotkey) == 0)
return false;
hotkey->_registered = false;
emit hotkey->registeredChanged(true);
if (shortcuts.count(shortcut) == 0) {
if (!unregisterShortcut(shortcut)) {
qCWarning(logQHotkey) << QHotkey::tr("Failed to unregister %1. Error: %2")
.arg(hotkey->shortcut().toString(), error);
return false;
}
return true;
}
return true;
}
QHotkey::NativeShortcut
QHotkeyPrivate::nativeShortcutInvoked(Qt::Key keycode,
Qt::KeyboardModifiers modifiers) {
if (mapping.contains({keycode, modifiers}))
return mapping.value({keycode, modifiers});
bool ok1 = false;
auto k = nativeKeycode(keycode, ok1);
bool ok2 = false;
auto m = nativeModifiers(modifiers, ok2);
if (ok1 && ok2)
return {k, m};
return {};
}
QHotkey::NativeShortcut::NativeShortcut() : key(), modifier(), valid(false) {}
QHotkey::NativeShortcut::NativeShortcut(quint32 key, quint32 modifier)
: key(key), modifier(modifier), valid(true) {}
bool QHotkey::NativeShortcut::isValid() const { return valid; }
bool QHotkey::NativeShortcut::operator==(QHotkey::NativeShortcut other) const {
return (key == other.key) && (modifier == other.modifier) &&
valid == other.valid;
}
bool QHotkey::NativeShortcut::operator!=(QHotkey::NativeShortcut other) const {
return (key != other.key) || (modifier != other.modifier) ||
valid != other.valid;
}
uint qHash(QHotkey::NativeShortcut key) {
return qHash(key.key) ^ qHash(key.modifier);
}
uint qHash(QHotkey::NativeShortcut key, uint seed) {
return qHash(key.key, seed) ^ qHash(key.modifier, seed);
}