diff options
Diffstat (limited to 'qxt-mini/qxtglobalshortcut_x11.cpp')
-rw-r--r-- | qxt-mini/qxtglobalshortcut_x11.cpp | 459 |
1 files changed, 283 insertions, 176 deletions
diff --git a/qxt-mini/qxtglobalshortcut_x11.cpp b/qxt-mini/qxtglobalshortcut_x11.cpp index 01dfe8df..b214a3a0 100644 --- a/qxt-mini/qxtglobalshortcut_x11.cpp +++ b/qxt-mini/qxtglobalshortcut_x11.cpp @@ -30,136 +30,154 @@ ** <http://libqxt.org> <foundation@libqxt.org> *****************************************************************************/ +// qt must go first or #error #include <QHash> #include <QMutex> #include <QDebug> #include <QPair> #include <QKeyEvent> #include <QApplication> +#include "qplatformnativeinterface.h" + +#include "x11-keymap.hpp" #include <X11/Xlib.h> #include <X11/XKBlib.h> #include <xcb/xcb.h> -#include "qplatformnativeinterface.h" #include "compat/util.hpp" #include <iterator> -#include "x11-keymap.hpp" +#include <vector> +#include <type_traits> +#include <utility> +#include <cinttypes> +#include <array> + +template<typename t, int M, typename size_type_ = std::uintptr_t> +struct powerset final +{ + static_assert(std::is_integral<size_type_>::value, ""); -static constexpr quint32 AllMods = ShiftMask|LockMask|ControlMask|Mod1Mask|Mod2Mask|Mod3Mask|Mod4Mask|Mod5Mask; + using size_type = size_type_; -static constexpr quint32 evil_mods[] = { -#if 0 - LockMask, // caps lock - Mod3Mask, // scroll lock -#endif - Mod2Mask, // num lock - Mod5Mask, // altgr + static_assert(M > 0, ""); + static_assert(M < sizeof(size_type[8]), ""); + static_assert(std::is_unsigned<size_type>::value || M < sizeof(size_type)*8 - 1, ""); - Mod2Mask | Mod5Mask, -}; + using N = std::integral_constant<size_type, (size_type(1) << size_type(M))-1>; + static_assert((N::value & (N::value + 1)) == 0, ""); -typedef int (*X11ErrorHandler)(Display *display, XErrorEvent *event); + using set_type = std::vector<t>; + using sets_type = std::array<set_type, N::value>; + using element_type = t; + using element_count = std::integral_constant<size_type, N::value>; + using self_type = powerset<t, M>; -using pair = QPair<quint32, quint32>; + operator const sets_type&() const { return sets_; } + operator sets_type&() { return sets_; } -struct keybinding final -{ - quint32 code, mods; - int refcnt; + const sets_type& sets() const { return sets_; } + sets_type& sets() { return sets_; } - static QHash<pair, keybinding> list; - static QMutex lock; + set_type& operator[](unsigned k) { return sets_[k]; } + const set_type& operator[](unsigned k) const { return sets_[k]; } - static bool incf(quint32 code, quint32 mods); - static bool decf(quint32 code, quint32 mods); + const set_type& elements() const { return elements_; } + set_type& elements() { return elements_; } - ~keybinding(); + template<typename = void> + operator QString() const + { + QString str; + unsigned k = 0; + for (const auto& set : sets_) + { + str.append(QStringLiteral("#%1: ").arg(++k)); + for (const auto& x : set) + str.append(QStringLiteral("%1 ").arg(x)); + str.append('\n'); + } + return str.mid(0, str.size() - 1); + } + + powerset() {} private: - keybinding(quint32 code, quint32 mods); + sets_type sets_; + set_type elements_; }; -bool operator==(const keybinding& k1, const keybinding& k2) +template<typename t, typename... xs> +static auto +make_powerset(const t& arg, const xs&... args) { - return k1.code == k2.code && k1.mods == k2.mods; -} + using cnt = std::integral_constant<std::uintptr_t, sizeof...(xs)+1>; + using p = powerset<t, cnt::value>; + using len = typename p::element_count; + using vec = typename p::set_type; + using size_type = typename p::size_type; -inline bool operator!=(const keybinding& k1, const keybinding& k2) -{ - return !(k1 == k2); -} - -uint qHash(const keybinding& k) -{ - return uint(k.code * 41) ^ qHash(k.mods); -} - -uint qHash(const keybinding& k, uint seed) -{ - return qHash(uint(k.code * 41) ^ qHash(k.mods), seed); -} - -keybinding::keybinding(quint32 code, quint32 mods) : - code(code), mods(mods), - refcnt(0) -{ -} + p ret; + vec v; + v.reserve(len()); -keybinding::~keybinding() -{ -} + const typename p::set_type ts {{arg, static_cast<t>(args)...}}; -bool keybinding::incf(quint32 code, quint32 mods) -{ - QMutexLocker l(&lock); - - keybinding k = list.value(pair(code, mods), keybinding(code, mods)); + ret.elements() = std::vector<t>(std::begin(ts), std::end(ts)); - const bool ret = k.refcnt == 0; - - if (ret) + // no nullary set + for (size_type i = 0; i < len(); i++) { - //qDebug() << "qxt-mini: registered keybinding" << code; - } - - k.refcnt++; - list.insert(pair(code, mods), k); + v.clear(); + size_type k = 1; + for (const t& x : ts) + { + if ((i+1) & k) + v.push_back(std::move(x)); + k <<= 1; + } - //qDebug() << "qxt-mini: incf: refcount for" << code << "now" << k.refcnt; + ret[i] = vec(std::begin(v), std::end(v)); + ret[i].shrink_to_fit(); + } return ret; } -bool keybinding::decf(quint32 code, quint32 mods) -{ - QMutexLocker l(&lock); +static auto evil_mods = make_powerset(LockMask, Mod3Mask, Mod5Mask); - auto it = list.find(pair(code, mods)); +static Qt::KeyboardModifiers evil_qt_mods = Qt::KeypadModifier; - if (it == list.end()) +static inline quint32 filter_evil_mods(quint32 mods) +{ + for (auto mod : evil_mods.elements()) { - qWarning() << "qxt-mini: spurious keybinding decf on" << code; - return false; + mods &= quint32(~mod); } + return mods; +} - keybinding& k = *it; - k.refcnt--; +using pair = QPair<quint32, quint32>; - if (k.refcnt == 0) - { - list.erase(it); - //qDebug() << "qxt-mini: removed keybinding" << code; - return true; - } +struct keybinding final +{ + quint32 code, mods; + int refcnt; - //qDebug() << "qxt-mini: decf: refcount for" << code << "now" << k.refcnt; + static QHash<pair, keybinding> list; + static QMutex lock; - return false; -} + static bool incf(quint32 code, quint32 mods); + static bool decf(quint32 code, quint32 mods); -QHash<pair, keybinding> keybinding::list; -QMutex keybinding::lock; + ~keybinding(); + +private: + keybinding(quint32 code, quint32 mods); +}; + +static std::vector<pair> native_key(Qt::Key key, Qt::KeyboardModifiers modifiers); +typedef int (*X11ErrorHandler)(Display *display, XErrorEvent *event); class QxtX11ErrorHandler { public: @@ -197,8 +215,6 @@ private: X11ErrorHandler m_previousErrorHandler; }; -bool QxtX11ErrorHandler::error = false; - class QxtX11Data { public: QxtX11Data() @@ -225,95 +241,203 @@ public: return DefaultRootWindow(display()); } - static constexpr quint32 filter_evil_mods(quint32 mods) + bool grabKey(quint32 code, quint32 mods) { - for (quint32 mod : evil_mods) - mods &= ~mod; - return mods; - } + const std::vector<pair> keycodes = native_key(Qt::Key(code), Qt::KeyboardModifiers(mods)); + bool ret = true; - bool grabKey(quint32 code, quint32 mods, Window window) - { - // qDebug() << "grabbing key" << code << mods; + for (pair x : keycodes) + { + int native_code = x.first, native_mods = x.second; - mods = filter_evil_mods(mods); + if (keybinding::incf(native_code, native_mods)) + { + QxtX11ErrorHandler errorHandler; - // qDebug() << "mods now" << mods; + XGrabKey(display(), native_code, native_mods, rootWindow(), True, GrabModeAsync, GrabModeAsync); - if (keybinding::incf(code, mods)) - { - QxtX11ErrorHandler errorHandler; - - XGrabKey(display(), code, mods, window, True, GrabModeAsync, GrabModeAsync); + for (const auto& set : evil_mods.sets()) + { + quint32 m = native_mods; - for (quint32 evil : evil_mods) - { - quint32 m = mods | evil; + for (auto value : set) + m |= value; - XGrabKey(display(), code, m, window, True, GrabModeAsync, GrabModeAsync); - } + XGrabKey(display(), native_code, m, rootWindow(), True, GrabModeAsync, GrabModeAsync); + } - if (errorHandler.error) - { - qDebug() << "qxt-mini: error while binding to" << code << mods; - ungrabKey(code, mods, window); - return false; + if (errorHandler.error) + { + qDebug() << "qxt-mini: error while binding to" << code << mods; + ungrabKey(code, mods); + ret = false; + } } } - return true; + return ret; } - bool ungrabKey(quint32 code, quint32 mods, Window window) + bool ungrabKey(quint32 code, quint32 mods) { - mods = filter_evil_mods(mods); + const std::vector<pair> keycodes = native_key(Qt::Key(code), Qt::KeyboardModifiers(mods)); + bool ret = true; - if (keybinding::decf(code, mods)) + for (pair x : keycodes) { - QxtX11ErrorHandler errorHandler; - XUngrabKey(display(), code, mods, window); + int native_code = x.first, native_mods = x.second; - for (quint32 evil : evil_mods) + if (keybinding::decf(native_code, native_mods)) { - quint32 m = mods | evil; - XUngrabKey(display(), code, m, window); - } + QxtX11ErrorHandler errorHandler; + XUngrabKey(display(), native_code, native_mods, rootWindow()); - if (errorHandler.error) - { - qDebug() << "qxt-mini: error while unbinding" << code << mods; - return false; + for (const auto& set : evil_mods.sets()) + { + quint32 m = mods; + + for (auto value : set) + m |= value; + + XUngrabKey(display(), code, m, rootWindow()); + } + + if (errorHandler.error) + { + qDebug() << "qxt-mini: error while unbinding" << code << mods; + ret = false; + } } } - return true; + return ret; } private: Display *m_display; }; +static std::vector<pair> native_key(Qt::Key key, Qt::KeyboardModifiers modifiers) +{ + std::vector<pair> ret; + + QxtX11Data x11; + if (!x11.isValid()) + return ret; + + std::vector<quint32> keycodes = qt_key_to_x11(x11.display(), key, modifiers); + unsigned mods = qt_mods_to_x11(modifiers); + mods = filter_evil_mods(mods); + + for (quint32 code : keycodes) + ret.push_back(pair(code, mods)); + + return ret; +} + +bool operator==(const keybinding& k1, const keybinding& k2) +{ + return k1.code == k2.code && k1.mods == k2.mods; +} + +inline bool operator!=(const keybinding& k1, const keybinding& k2) +{ + return !(k1 == k2); +} + +uint qHash(const keybinding& k) +{ + return uint(k.code * 41) ^ qHash(k.mods); +} + +uint qHash(const keybinding& k, uint seed) +{ + return qHash(uint(k.code * 41) ^ qHash(k.mods), seed); +} + +keybinding::keybinding(quint32 code, quint32 mods) : + code(code), mods(mods), + refcnt(0) +{ +} + +keybinding::~keybinding() +{ +} + +bool keybinding::incf(quint32 code, quint32 mods) +{ + QMutexLocker l(&lock); + + keybinding k = list.value(pair(code, mods), keybinding(code, mods)); + + const bool ret = k.refcnt == 0; + + if (ret) + { + //qDebug() << "qxt-mini: registered keybinding" << code; + } + + k.refcnt++; + list.insert(pair(code, mods), k); + + //qDebug() << "qxt-mini: incf: refcount for" << code << "now" << k.refcnt; + + return ret; +} + +bool keybinding::decf(quint32 code, quint32 mods) +{ + QMutexLocker l(&lock); + + auto it = list.find(pair(code, mods)); + + if (it == list.end()) + { + qWarning() << "qxt-mini: spurious keybinding decf on" << code; + return false; + } + + keybinding& k = *it; + k.refcnt--; + + if (k.refcnt == 0) + { + list.erase(it); + //qDebug() << "qxt-mini: removed keybinding" << code; + return true; + } + + //qDebug() << "qxt-mini: decf: refcount for" << code << "now" << k.refcnt; + + return false; +} + +QHash<pair, keybinding> keybinding::list; +QMutex keybinding::lock; + +bool QxtX11ErrorHandler::error = false; + bool QxtGlobalShortcutPrivate::nativeEventFilter(const QByteArray & eventType, - void *message, long *result) + void *message, long *) { - Q_UNUSED(result); + QxtX11Data x11; + + if (!x11.isValid()) + return false; { static bool once_ = false; if (!once_) { - QxtX11Data x11; - if (x11.isValid()) - { - once_ = true; - Bool val; + once_ = true; + Bool val = False; - (void) XkbSetDetectableAutoRepeat(x11.display(), True, &val); + (void) XkbSetDetectableAutoRepeat(x11.display(), True, &val); - if (val) - qDebug() << "qxt-mini: fixed x11 autorepeat"; - else - qDebug() << "qxt-mini: can't fix x11 autorepeat"; - } + if (val) + qDebug() << "qxt-mini: fixed x11 autorepeat"; + else + qDebug() << "qxt-mini: can't fix x11 autorepeat"; } } @@ -357,6 +481,10 @@ bool QxtGlobalShortcutPrivate::nativeEventFilter(const QByteArray & eventType, #endif unsigned int keycode = kev->detail; + + if (keycode == 0) + return false; + unsigned int keystate = 0; if(kev->state & XCB_MOD_MASK_1) // alt keystate |= Mod1Mask; @@ -366,64 +494,43 @@ bool QxtGlobalShortcutPrivate::nativeEventFilter(const QByteArray & eventType, keystate |= Mod4Mask; if(kev->state & XCB_MOD_MASK_SHIFT) //shift keystate |= ShiftMask; -#if 0 - if(key->state & XCB_MOD_MASK_3) // alt gr aka right-alt or ctrl+left-alt -- what mask is it? - keystate |= AltGrMask; -#endif + if(kev->state & XCB_MOD_MASK_2) // numlock + keystate |= Mod2Mask; - // qDebug() << "qxt-mini:" << (is_release ? "keyup" : "keydown") << keycode << keystate; + keystate = filter_evil_mods(keystate); - activateShortcut(keycode, keystate, !is_release); + QPair<KeySym, KeySym> sym_ = keycode_to_keysym(x11.display(), keycode, keystate, kev); + KeySym sym = sym_.first; + + Qt::Key k; Qt::KeyboardModifiers mods; + std::tie(k, mods) = x11_key_to_qt(x11.display(), sym, keystate); + + if (k != 0) + activateShortcut(k, mods, !is_release); } return false; } quint32 QxtGlobalShortcutPrivate::nativeModifiers(Qt::KeyboardModifiers modifiers) { - quint32 native = 0; - if (modifiers & Qt::AltModifier) - native |= Mod1Mask; - if (modifiers & Qt::ControlModifier) - native |= ControlMask; - if (modifiers & Qt::MetaModifier) - native |= Mod4Mask; - if (modifiers & Qt::ShiftModifier) - native |= ShiftMask; - if (modifiers & Qt::KeypadModifier) - native |= Mod2Mask; - if (modifiers & Qt::MetaModifier) // Super aka Windows key - native |= Mod4Mask; - if (modifiers & Qt::KeypadModifier) // numlock - native |= Mod2Mask; - - native &= AllMods; - - return native; + modifiers &= ~evil_qt_mods; + return quint32(modifiers); } quint32 QxtGlobalShortcutPrivate::nativeKeycode(Qt::Key key) { - QxtX11Data x11; - - if (x11.isValid()) - return qt_key_to_x11(x11.display(), key); - - return unsigned(-1); + return quint32(key); } bool QxtGlobalShortcutPrivate::registerShortcut(quint32 nativeKey, quint32 nativeMods) { QxtX11Data x11; - if (nativeKey == unsigned(-1)) - return false; - return x11.isValid() && x11.grabKey(nativeKey, nativeMods, x11.rootWindow()); + return x11.isValid() && x11.grabKey(nativeKey, nativeMods); } bool QxtGlobalShortcutPrivate::unregisterShortcut(quint32 nativeKey, quint32 nativeMods) { QxtX11Data x11; - if (nativeKey == unsigned(-1)) - return false; - return x11.isValid() && x11.ungrabKey(nativeKey, nativeMods, x11.rootWindow()); + return x11.isValid() && x11.ungrabKey(nativeKey, nativeMods); } #endif |