2016-02-15 12:42:48 +00:00
|
|
|
/********************************************************************
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
|
|
|
|
|
|
|
Copyright (C) 2013, 2016 Martin Gräßlin <mgraesslin@kde.org>
|
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*********************************************************************/
|
|
|
|
#include "keyboard_input.h"
|
2016-05-24 08:57:57 +00:00
|
|
|
#include "input_event.h"
|
2016-02-15 12:42:48 +00:00
|
|
|
#include "abstract_client.h"
|
|
|
|
#include "options.h"
|
|
|
|
#include "utils.h"
|
2016-08-16 08:28:26 +00:00
|
|
|
#include "screenlockerwatcher.h"
|
2016-02-15 12:42:48 +00:00
|
|
|
#include "toplevel.h"
|
|
|
|
#include "wayland_server.h"
|
|
|
|
#include "workspace.h"
|
|
|
|
// KWayland
|
2016-06-20 09:21:16 +00:00
|
|
|
#include <KWayland/Server/datadevice_interface.h>
|
2016-02-15 12:42:48 +00:00
|
|
|
#include <KWayland/Server/seat_interface.h>
|
|
|
|
//screenlocker
|
|
|
|
#include <KScreenLocker/KsldApp>
|
|
|
|
// Frameworks
|
|
|
|
#include <KKeyServer>
|
2017-01-22 10:55:33 +00:00
|
|
|
#include <KLocalizedString>
|
2016-02-19 12:50:39 +00:00
|
|
|
#include <KGlobalAccel>
|
2016-02-15 12:42:48 +00:00
|
|
|
// Qt
|
|
|
|
#include <QDBusConnection>
|
|
|
|
#include <QDBusMessage>
|
|
|
|
#include <QDBusPendingCall>
|
|
|
|
#include <QKeyEvent>
|
|
|
|
#include <QTemporaryFile>
|
|
|
|
// xkbcommon
|
|
|
|
#include <xkbcommon/xkbcommon.h>
|
2016-08-29 15:04:13 +00:00
|
|
|
#include <xkbcommon/xkbcommon-compose.h>
|
2016-02-15 12:42:48 +00:00
|
|
|
#include <xkbcommon/xkbcommon-keysyms.h>
|
|
|
|
// system
|
|
|
|
#include <sys/mman.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2016-07-16 17:14:44 +00:00
|
|
|
Q_LOGGING_CATEGORY(KWIN_XKB, "kwin_xkbcommon", QtCriticalMsg)
|
2016-02-19 08:08:32 +00:00
|
|
|
|
2016-02-15 12:42:48 +00:00
|
|
|
namespace KWin
|
|
|
|
{
|
2016-02-19 08:08:56 +00:00
|
|
|
|
|
|
|
static void xkbLogHandler(xkb_context *context, xkb_log_level priority, const char *format, va_list args)
|
|
|
|
{
|
|
|
|
Q_UNUSED(context)
|
|
|
|
char buf[1024];
|
|
|
|
if (std::vsnprintf(buf, 1023, format, args) <= 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
switch (priority) {
|
|
|
|
case XKB_LOG_LEVEL_DEBUG:
|
|
|
|
qCDebug(KWIN_XKB) << "XKB:" << buf;
|
|
|
|
break;
|
|
|
|
case XKB_LOG_LEVEL_INFO:
|
2016-03-03 15:14:17 +00:00
|
|
|
#if (QT_VERSION >= QT_VERSION_CHECK(5, 5, 0))
|
2016-02-19 08:08:56 +00:00
|
|
|
qCInfo(KWIN_XKB) << "XKB:" << buf;
|
2016-03-03 15:14:17 +00:00
|
|
|
#endif
|
2016-02-19 08:08:56 +00:00
|
|
|
break;
|
|
|
|
case XKB_LOG_LEVEL_WARNING:
|
|
|
|
qCWarning(KWIN_XKB) << "XKB:" << buf;
|
|
|
|
break;
|
|
|
|
case XKB_LOG_LEVEL_ERROR:
|
|
|
|
case XKB_LOG_LEVEL_CRITICAL:
|
|
|
|
default:
|
|
|
|
qCCritical(KWIN_XKB) << "XKB:" << buf;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-15 12:42:48 +00:00
|
|
|
Xkb::Xkb(InputRedirection *input)
|
|
|
|
: m_input(input)
|
|
|
|
, m_context(xkb_context_new(static_cast<xkb_context_flags>(0)))
|
|
|
|
, m_keymap(NULL)
|
|
|
|
, m_state(NULL)
|
|
|
|
, m_shiftModifier(0)
|
2016-08-17 06:39:51 +00:00
|
|
|
, m_capsModifier(0)
|
2016-02-15 12:42:48 +00:00
|
|
|
, m_controlModifier(0)
|
|
|
|
, m_altModifier(0)
|
|
|
|
, m_metaModifier(0)
|
2016-10-05 09:50:20 +00:00
|
|
|
, m_numLock(0)
|
|
|
|
, m_capsLock(0)
|
|
|
|
, m_scrollLock(0)
|
2016-02-15 12:42:48 +00:00
|
|
|
, m_modifiers(Qt::NoModifier)
|
2016-09-14 05:34:00 +00:00
|
|
|
, m_consumedModifiers(Qt::NoModifier)
|
2016-08-29 15:04:13 +00:00
|
|
|
, m_keysym(XKB_KEY_NoSymbol)
|
2016-10-05 09:50:20 +00:00
|
|
|
, m_leds()
|
2016-02-15 12:42:48 +00:00
|
|
|
{
|
2016-10-05 09:50:20 +00:00
|
|
|
qRegisterMetaType<KWin::Xkb::LEDs>();
|
2016-02-15 12:42:48 +00:00
|
|
|
if (!m_context) {
|
2016-02-19 08:08:32 +00:00
|
|
|
qCDebug(KWIN_XKB) << "Could not create xkb context";
|
2016-02-15 12:42:48 +00:00
|
|
|
} else {
|
2016-02-19 08:08:56 +00:00
|
|
|
xkb_context_set_log_level(m_context, XKB_LOG_LEVEL_DEBUG);
|
|
|
|
xkb_context_set_log_fn(m_context, &xkbLogHandler);
|
2016-08-29 15:04:13 +00:00
|
|
|
|
|
|
|
// get locale as described in xkbcommon doc
|
|
|
|
// cannot use QLocale as it drops the modifier part
|
|
|
|
QByteArray locale = qgetenv("LC_ALL");
|
|
|
|
if (locale.isEmpty()) {
|
|
|
|
locale = qgetenv("LC_CTYPE");
|
|
|
|
}
|
|
|
|
if (locale.isEmpty()) {
|
|
|
|
locale = qgetenv("LANG");
|
|
|
|
}
|
|
|
|
if (locale.isEmpty()) {
|
|
|
|
locale = QByteArrayLiteral("C");
|
|
|
|
}
|
|
|
|
|
|
|
|
m_compose.table = xkb_compose_table_new_from_locale(m_context, locale.constData(), XKB_COMPOSE_COMPILE_NO_FLAGS);
|
|
|
|
if (m_compose.table) {
|
|
|
|
m_compose.state = xkb_compose_state_new(m_compose.table, XKB_COMPOSE_STATE_NO_FLAGS);
|
|
|
|
}
|
2016-02-15 12:42:48 +00:00
|
|
|
}
|
2016-08-13 13:25:24 +00:00
|
|
|
|
|
|
|
auto resetModOnlyShortcut = [this] {
|
|
|
|
m_modOnlyShortcut.modifier = Qt::NoModifier;
|
|
|
|
};
|
|
|
|
QObject::connect(m_input, &InputRedirection::pointerButtonStateChanged, resetModOnlyShortcut);
|
|
|
|
QObject::connect(m_input, &InputRedirection::pointerAxisChanged, resetModOnlyShortcut);
|
2016-08-16 08:28:26 +00:00
|
|
|
QObject::connect(ScreenLockerWatcher::self(), &ScreenLockerWatcher::locked, m_input, resetModOnlyShortcut);
|
2016-02-15 12:42:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Xkb::~Xkb()
|
|
|
|
{
|
2016-08-29 15:04:13 +00:00
|
|
|
xkb_compose_state_unref(m_compose.state);
|
|
|
|
xkb_compose_table_unref(m_compose.table);
|
2016-02-15 12:42:48 +00:00
|
|
|
xkb_state_unref(m_state);
|
|
|
|
xkb_keymap_unref(m_keymap);
|
|
|
|
xkb_context_unref(m_context);
|
|
|
|
}
|
|
|
|
|
2016-02-19 10:15:37 +00:00
|
|
|
void Xkb::reconfigure()
|
|
|
|
{
|
|
|
|
if (!m_context) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-08-17 06:35:15 +00:00
|
|
|
xkb_keymap *keymap = nullptr;
|
|
|
|
if (!qEnvironmentVariableIsSet("KWIN_XKB_DEFAULT_KEYMAP")) {
|
|
|
|
keymap = loadKeymapFromConfig();
|
|
|
|
}
|
2016-02-19 10:15:37 +00:00
|
|
|
if (!keymap) {
|
|
|
|
qCDebug(KWIN_XKB) << "Could not create xkb keymap from configuration";
|
|
|
|
keymap = loadDefaultKeymap();
|
|
|
|
}
|
|
|
|
if (keymap) {
|
|
|
|
updateKeymap(keymap);
|
|
|
|
} else {
|
|
|
|
qCDebug(KWIN_XKB) << "Could not create default xkb keymap";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
xkb_keymap *Xkb::loadKeymapFromConfig()
|
|
|
|
{
|
|
|
|
// load config
|
|
|
|
const KConfigGroup config = KSharedConfig::openConfig(QStringLiteral("kxkbrc"), KConfig::NoGlobals)->group("Layout");
|
|
|
|
const QByteArray model = config.readEntry("Model", "pc104").toLocal8Bit();
|
|
|
|
const QByteArray layout = config.readEntry("LayoutList", "").toLocal8Bit();
|
|
|
|
const QByteArray options = config.readEntry("Options", "").toLocal8Bit();
|
|
|
|
|
|
|
|
xkb_rule_names ruleNames = {
|
|
|
|
.rules = nullptr,
|
|
|
|
.model = model.constData(),
|
|
|
|
.layout = layout.constData(),
|
|
|
|
.variant = nullptr,
|
|
|
|
.options = options.constData()
|
|
|
|
};
|
|
|
|
return xkb_keymap_new_from_names(m_context, &ruleNames, XKB_KEYMAP_COMPILE_NO_FLAGS);
|
|
|
|
}
|
|
|
|
|
|
|
|
xkb_keymap *Xkb::loadDefaultKeymap()
|
|
|
|
{
|
|
|
|
return xkb_keymap_new_from_names(m_context, nullptr, XKB_KEYMAP_COMPILE_NO_FLAGS);
|
|
|
|
}
|
|
|
|
|
2016-02-15 12:42:48 +00:00
|
|
|
void Xkb::installKeymap(int fd, uint32_t size)
|
|
|
|
{
|
|
|
|
if (!m_context) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
char *map = reinterpret_cast<char*>(mmap(NULL, size, PROT_READ, MAP_SHARED, fd, 0));
|
|
|
|
if (map == MAP_FAILED) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
xkb_keymap *keymap = xkb_keymap_new_from_string(m_context, map, XKB_KEYMAP_FORMAT_TEXT_V1, XKB_MAP_COMPILE_PLACEHOLDER);
|
|
|
|
munmap(map, size);
|
|
|
|
if (!keymap) {
|
2016-02-19 08:08:32 +00:00
|
|
|
qCDebug(KWIN_XKB) << "Could not map keymap from file";
|
2016-02-15 12:42:48 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
updateKeymap(keymap);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Xkb::updateKeymap(xkb_keymap *keymap)
|
|
|
|
{
|
|
|
|
Q_ASSERT(keymap);
|
|
|
|
xkb_state *state = xkb_state_new(keymap);
|
|
|
|
if (!state) {
|
2016-02-19 08:08:32 +00:00
|
|
|
qCDebug(KWIN_XKB) << "Could not create XKB state";
|
2016-02-15 12:42:48 +00:00
|
|
|
xkb_keymap_unref(keymap);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// now release the old ones
|
|
|
|
xkb_state_unref(m_state);
|
|
|
|
xkb_keymap_unref(m_keymap);
|
|
|
|
|
|
|
|
m_keymap = keymap;
|
|
|
|
m_state = state;
|
|
|
|
|
|
|
|
m_shiftModifier = xkb_keymap_mod_get_index(m_keymap, XKB_MOD_NAME_SHIFT);
|
2016-08-17 06:39:51 +00:00
|
|
|
m_capsModifier = xkb_keymap_mod_get_index(m_keymap, XKB_MOD_NAME_CAPS);
|
2016-02-15 12:42:48 +00:00
|
|
|
m_controlModifier = xkb_keymap_mod_get_index(m_keymap, XKB_MOD_NAME_CTRL);
|
|
|
|
m_altModifier = xkb_keymap_mod_get_index(m_keymap, XKB_MOD_NAME_ALT);
|
|
|
|
m_metaModifier = xkb_keymap_mod_get_index(m_keymap, XKB_MOD_NAME_LOGO);
|
2016-10-05 09:50:20 +00:00
|
|
|
|
|
|
|
m_numLock = xkb_keymap_led_get_index(m_keymap, XKB_LED_NAME_NUM);
|
|
|
|
m_capsLock = xkb_keymap_led_get_index(m_keymap, XKB_LED_NAME_CAPS);
|
|
|
|
m_scrollLock = xkb_keymap_led_get_index(m_keymap, XKB_LED_NAME_SCROLL);
|
|
|
|
|
2016-02-19 13:31:53 +00:00
|
|
|
m_currentLayout = xkb_state_serialize_layout(m_state, XKB_STATE_LAYOUT_EFFECTIVE);
|
2016-02-15 12:42:48 +00:00
|
|
|
|
|
|
|
createKeymapFile();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Xkb::createKeymapFile()
|
|
|
|
{
|
|
|
|
if (!waylandServer()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// TODO: uninstall keymap on server?
|
|
|
|
if (!m_keymap) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ScopedCPointer<char> keymapString(xkb_keymap_get_as_string(m_keymap, XKB_KEYMAP_FORMAT_TEXT_V1));
|
|
|
|
if (keymapString.isNull()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const uint size = qstrlen(keymapString.data()) + 1;
|
|
|
|
|
|
|
|
QTemporaryFile *tmp = new QTemporaryFile(m_input);
|
|
|
|
if (!tmp->open()) {
|
|
|
|
delete tmp;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
unlink(tmp->fileName().toUtf8().constData());
|
|
|
|
if (!tmp->resize(size)) {
|
|
|
|
delete tmp;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
uchar *address = tmp->map(0, size);
|
|
|
|
if (!address) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (qstrncpy(reinterpret_cast<char*>(address), keymapString.data(), size) == nullptr) {
|
|
|
|
delete tmp;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
waylandServer()->seat()->setKeymap(tmp->handle(), size);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Xkb::updateModifiers(uint32_t modsDepressed, uint32_t modsLatched, uint32_t modsLocked, uint32_t group)
|
|
|
|
{
|
|
|
|
if (!m_keymap || !m_state) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
xkb_state_update_mask(m_state, modsDepressed, modsLatched, modsLocked, 0, 0, group);
|
|
|
|
updateModifiers();
|
|
|
|
}
|
|
|
|
|
|
|
|
void Xkb::updateKey(uint32_t key, InputRedirection::KeyboardKeyState state)
|
|
|
|
{
|
|
|
|
if (!m_keymap || !m_state) {
|
|
|
|
return;
|
|
|
|
}
|
2017-01-22 08:44:15 +00:00
|
|
|
const auto oldMods = modifiersRelevantForGlobalShortcuts();
|
2016-02-15 12:42:48 +00:00
|
|
|
xkb_state_update_key(m_state, key + 8, static_cast<xkb_key_direction>(state));
|
2016-08-29 15:04:13 +00:00
|
|
|
if (state == InputRedirection::KeyboardKeyPressed) {
|
|
|
|
const auto sym = toKeysym(key);
|
|
|
|
if (m_compose.state && xkb_compose_state_feed(m_compose.state, sym) == XKB_COMPOSE_FEED_ACCEPTED) {
|
|
|
|
switch (xkb_compose_state_get_status(m_compose.state)) {
|
|
|
|
case XKB_COMPOSE_NOTHING:
|
|
|
|
m_keysym = sym;
|
|
|
|
break;
|
|
|
|
case XKB_COMPOSE_COMPOSED:
|
|
|
|
m_keysym = xkb_compose_state_get_one_sym(m_compose.state);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
m_keysym = XKB_KEY_NoSymbol;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
m_keysym = sym;
|
|
|
|
}
|
|
|
|
}
|
2016-02-15 12:42:48 +00:00
|
|
|
updateModifiers();
|
2016-09-14 05:34:00 +00:00
|
|
|
updateConsumedModifiers(key);
|
2016-02-15 12:42:48 +00:00
|
|
|
if (state == InputRedirection::KeyboardKeyPressed) {
|
|
|
|
m_modOnlyShortcut.pressCount++;
|
2016-08-13 13:25:24 +00:00
|
|
|
if (m_modOnlyShortcut.pressCount == 1 &&
|
2016-08-16 08:28:26 +00:00
|
|
|
!ScreenLockerWatcher::self()->isLocked() &&
|
2016-08-17 06:39:51 +00:00
|
|
|
oldMods == Qt::NoModifier &&
|
2016-08-13 13:25:24 +00:00
|
|
|
m_input->qtButtonStates() == Qt::NoButton) {
|
2017-01-22 08:44:15 +00:00
|
|
|
m_modOnlyShortcut.modifier = Qt::KeyboardModifier(int(modifiersRelevantForGlobalShortcuts()));
|
2016-02-15 12:42:48 +00:00
|
|
|
} else {
|
|
|
|
m_modOnlyShortcut.modifier = Qt::NoModifier;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
m_modOnlyShortcut.pressCount--;
|
2016-08-17 06:39:51 +00:00
|
|
|
if (m_modOnlyShortcut.pressCount == 0 &&
|
2017-01-22 08:44:15 +00:00
|
|
|
modifiersRelevantForGlobalShortcuts() == Qt::NoModifier &&
|
2016-10-06 05:41:43 +00:00
|
|
|
!workspace()->globalShortcutsDisabled()) {
|
2016-02-15 12:42:48 +00:00
|
|
|
if (m_modOnlyShortcut.modifier != Qt::NoModifier) {
|
|
|
|
const auto list = options->modifierOnlyDBusShortcut(m_modOnlyShortcut.modifier);
|
|
|
|
if (list.size() >= 4) {
|
|
|
|
auto call = QDBusMessage::createMethodCall(list.at(0), list.at(1), list.at(2), list.at(3));
|
|
|
|
QVariantList args;
|
|
|
|
for (int i = 4; i < list.size(); ++i) {
|
|
|
|
args << list.at(i);
|
|
|
|
}
|
|
|
|
call.setArguments(args);
|
|
|
|
QDBusConnection::sessionBus().asyncCall(call);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
m_modOnlyShortcut.modifier = Qt::NoModifier;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Xkb::updateModifiers()
|
|
|
|
{
|
|
|
|
Qt::KeyboardModifiers mods = Qt::NoModifier;
|
2016-08-17 06:39:51 +00:00
|
|
|
if (xkb_state_mod_index_is_active(m_state, m_shiftModifier, XKB_STATE_MODS_EFFECTIVE) == 1 ||
|
|
|
|
xkb_state_mod_index_is_active(m_state, m_capsModifier, XKB_STATE_MODS_EFFECTIVE) == 1) {
|
2016-02-15 12:42:48 +00:00
|
|
|
mods |= Qt::ShiftModifier;
|
|
|
|
}
|
|
|
|
if (xkb_state_mod_index_is_active(m_state, m_altModifier, XKB_STATE_MODS_EFFECTIVE) == 1) {
|
|
|
|
mods |= Qt::AltModifier;
|
|
|
|
}
|
|
|
|
if (xkb_state_mod_index_is_active(m_state, m_controlModifier, XKB_STATE_MODS_EFFECTIVE) == 1) {
|
|
|
|
mods |= Qt::ControlModifier;
|
|
|
|
}
|
|
|
|
if (xkb_state_mod_index_is_active(m_state, m_metaModifier, XKB_STATE_MODS_EFFECTIVE) == 1) {
|
|
|
|
mods |= Qt::MetaModifier;
|
|
|
|
}
|
|
|
|
m_modifiers = mods;
|
2016-10-05 09:50:20 +00:00
|
|
|
|
|
|
|
// update LEDs
|
|
|
|
LEDs leds;
|
|
|
|
if (xkb_state_led_index_is_active(m_state, m_numLock) == 1) {
|
|
|
|
leds = leds | LED::NumLock;
|
|
|
|
}
|
|
|
|
if (xkb_state_led_index_is_active(m_state, m_capsLock) == 1) {
|
|
|
|
leds = leds | LED::CapsLock;
|
|
|
|
}
|
|
|
|
if (xkb_state_led_index_is_active(m_state, m_scrollLock) == 1) {
|
|
|
|
leds = leds | LED::ScrollLock;
|
|
|
|
}
|
|
|
|
if (m_leds != leds) {
|
|
|
|
m_leds = leds;
|
|
|
|
emit m_input->keyboard()->ledsChanged(m_leds);
|
|
|
|
}
|
|
|
|
|
2016-02-19 13:31:53 +00:00
|
|
|
const xkb_layout_index_t layout = xkb_state_serialize_layout(m_state, XKB_STATE_LAYOUT_EFFECTIVE);
|
|
|
|
if (layout != m_currentLayout) {
|
|
|
|
m_currentLayout = layout;
|
|
|
|
// notify OSD service about the new layout
|
2016-08-21 16:12:43 +00:00
|
|
|
if (kwinApp()->usesLibinput()) {
|
|
|
|
// only if kwin is in charge of keyboard input
|
|
|
|
QDBusMessage msg = QDBusMessage::createMethodCall(
|
|
|
|
QStringLiteral("org.kde.plasmashell"),
|
|
|
|
QStringLiteral("/org/kde/osdService"),
|
|
|
|
QStringLiteral("org.kde.osdService"),
|
|
|
|
QStringLiteral("kbdLayoutChanged"));
|
2016-02-19 13:31:53 +00:00
|
|
|
|
2017-01-22 10:55:33 +00:00
|
|
|
msg << i18nd("xkeyboard-config", xkb_keymap_layout_get_name(m_keymap, layout));
|
2016-02-19 13:31:53 +00:00
|
|
|
|
2016-08-21 16:12:43 +00:00
|
|
|
QDBusConnection::sessionBus().asyncCall(msg);
|
|
|
|
}
|
2016-02-19 13:31:53 +00:00
|
|
|
}
|
2016-08-12 13:14:49 +00:00
|
|
|
if (waylandServer()) {
|
|
|
|
waylandServer()->seat()->updateKeyboardModifiers(xkb_state_serialize_mods(m_state, xkb_state_component(XKB_STATE_MODS_DEPRESSED)),
|
|
|
|
xkb_state_serialize_mods(m_state, xkb_state_component(XKB_STATE_MODS_LATCHED)),
|
|
|
|
xkb_state_serialize_mods(m_state, xkb_state_component(XKB_STATE_MODS_LOCKED)),
|
|
|
|
layout);
|
|
|
|
}
|
2016-02-15 12:42:48 +00:00
|
|
|
}
|
|
|
|
|
2016-09-14 05:34:00 +00:00
|
|
|
void Xkb::updateConsumedModifiers(uint32_t key)
|
|
|
|
{
|
|
|
|
Qt::KeyboardModifiers mods = Qt::NoModifier;
|
|
|
|
if (xkb_state_mod_index_is_consumed(m_state, key + 8, m_shiftModifier) == 1) {
|
|
|
|
mods |= Qt::ShiftModifier;
|
|
|
|
}
|
|
|
|
if (xkb_state_mod_index_is_consumed(m_state, key + 8, m_altModifier) == 1) {
|
|
|
|
mods |= Qt::AltModifier;
|
|
|
|
}
|
|
|
|
if (xkb_state_mod_index_is_consumed(m_state, key + 8, m_controlModifier) == 1) {
|
|
|
|
mods |= Qt::ControlModifier;
|
|
|
|
}
|
|
|
|
if (xkb_state_mod_index_is_consumed(m_state, key + 8, m_metaModifier) == 1) {
|
|
|
|
mods |= Qt::MetaModifier;
|
|
|
|
}
|
|
|
|
m_consumedModifiers = mods;
|
|
|
|
}
|
|
|
|
|
|
|
|
Qt::KeyboardModifiers Xkb::modifiersRelevantForGlobalShortcuts() const
|
|
|
|
{
|
|
|
|
Qt::KeyboardModifiers mods = Qt::NoModifier;
|
|
|
|
if (xkb_state_mod_index_is_active(m_state, m_shiftModifier, XKB_STATE_MODS_EFFECTIVE) == 1) {
|
|
|
|
mods |= Qt::ShiftModifier;
|
|
|
|
}
|
|
|
|
if (xkb_state_mod_index_is_active(m_state, m_altModifier, XKB_STATE_MODS_EFFECTIVE) == 1) {
|
|
|
|
mods |= Qt::AltModifier;
|
|
|
|
}
|
|
|
|
if (xkb_state_mod_index_is_active(m_state, m_controlModifier, XKB_STATE_MODS_EFFECTIVE) == 1) {
|
|
|
|
mods |= Qt::ControlModifier;
|
|
|
|
}
|
|
|
|
if (xkb_state_mod_index_is_active(m_state, m_metaModifier, XKB_STATE_MODS_EFFECTIVE) == 1) {
|
|
|
|
mods |= Qt::MetaModifier;
|
|
|
|
}
|
Workaround xkbcommon behavior concerning consumed modifiers
Summary:
If a key could be turned into a keysym with a modifier xkbcommon
considers the modifier as consumed even if not pressed.
E.g. Alt+F3 considers alt as consumed as there is a keysym gnerated with
Ctrl+Alt+F3 (vt switching).
This change tries to workaround the problem by ignoring the consumed
modifiers if there are more modifiers consumed than active. It's
possible that this will create regressions for other shortcuts - we need
to test it in the wild. Although this might cause regressions I'm aiming
for Plasma/5.8 branch with the change. It only affects Wayland and fixes
quite important shortcuts from window manager perspective (desktop
switching (ctrl+f1 to ctrl+f4), desktop grid (ctrl+f8), present windows
(ctrl+f9, ctrl+10), cube (ctrl+f11), user actions (alt+f3), close window
(alt+f4)). If it causes regressions they need to be fixed as well in the
Plasma/5.8 branch.
A new API entry point for xkbcommon was proposed, but is not yet merged
and there is no release with it yet. Once that is available the
workaround should get removed and replaced by the new API call.
BUG: 368989
FIXED-IN: 5.8.1
Test Plan: Going to restart session now with the change
Reviewers: #kwin, #plasma_on_wayland
Subscribers: plasma-devel, kwin
Tags: #plasma_on_wayland, #kwin
Differential Revision: https://phabricator.kde.org/D2945
2016-10-05 13:33:21 +00:00
|
|
|
|
|
|
|
// workaround xkbcommon limitation concerning consumed modifiers
|
|
|
|
// if a key could be turned into a keysym with a modifier xkbcommon
|
|
|
|
// considers the modifier as consumed even if not pressed
|
|
|
|
// e.g. alt+F3 considers alt as consumed as there is a keysym generated
|
|
|
|
// with ctrl+alt+F3 (vt switching)
|
|
|
|
// For more information see:
|
|
|
|
// https://bugs.freedesktop.org/show_bug.cgi?id=92818
|
|
|
|
// https://github.com/xkbcommon/libxkbcommon/issues/17
|
|
|
|
|
|
|
|
// the workaround is to not consider the modifiers as consumed
|
|
|
|
// if they are not a currently
|
|
|
|
// this might have other side effects, though. The only proper way to
|
|
|
|
// handle this is through new API in xkbcommon which doesn't exist yet
|
|
|
|
if (m_consumedModifiers & ~m_modifiers) {
|
|
|
|
return mods;
|
|
|
|
}
|
|
|
|
|
2016-10-11 07:47:15 +00:00
|
|
|
Qt::KeyboardModifiers consumedMods = m_consumedModifiers;
|
|
|
|
if ((mods & Qt::ShiftModifier) && (consumedMods == Qt::ShiftModifier)) {
|
|
|
|
// test whether current keysym is a letter
|
|
|
|
// in that case the shift should be removed from the consumed modifiers again
|
|
|
|
// otherwise it would not be possible to trigger e.g. Shift+W as a shortcut
|
|
|
|
// see BUG: 370341
|
|
|
|
if (QChar(toQtKey(m_keysym)).isLetter()) {
|
|
|
|
consumedMods = Qt::KeyboardModifiers();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return mods & ~consumedMods;
|
2016-09-14 05:34:00 +00:00
|
|
|
}
|
|
|
|
|
2016-02-15 12:42:48 +00:00
|
|
|
xkb_keysym_t Xkb::toKeysym(uint32_t key)
|
|
|
|
{
|
|
|
|
if (!m_state) {
|
|
|
|
return XKB_KEY_NoSymbol;
|
|
|
|
}
|
|
|
|
return xkb_state_key_get_one_sym(m_state, key + 8);
|
|
|
|
}
|
|
|
|
|
|
|
|
QString Xkb::toString(xkb_keysym_t keysym)
|
|
|
|
{
|
|
|
|
if (!m_state || keysym == XKB_KEY_NoSymbol) {
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
QByteArray byteArray(7, 0);
|
|
|
|
int ok = xkb_keysym_to_utf8(keysym, byteArray.data(), byteArray.size());
|
|
|
|
if (ok == -1 || ok == 0) {
|
|
|
|
return QString();
|
|
|
|
}
|
|
|
|
return QString::fromUtf8(byteArray.constData());
|
|
|
|
}
|
|
|
|
|
2016-10-11 07:47:15 +00:00
|
|
|
Qt::Key Xkb::toQtKey(xkb_keysym_t keysym) const
|
2016-02-15 12:42:48 +00:00
|
|
|
{
|
|
|
|
int key = Qt::Key_unknown;
|
|
|
|
KKeyServer::symXToKeyQt(keysym, &key);
|
|
|
|
return static_cast<Qt::Key>(key);
|
|
|
|
}
|
|
|
|
|
2016-02-19 09:56:17 +00:00
|
|
|
bool Xkb::shouldKeyRepeat(quint32 key) const
|
|
|
|
{
|
|
|
|
if (!m_keymap) {
|
|
|
|
return false;
|
|
|
|
}
|
2016-08-29 15:14:08 +00:00
|
|
|
return xkb_keymap_key_repeats(m_keymap, key + 8) != 0;
|
2016-02-19 09:56:17 +00:00
|
|
|
}
|
|
|
|
|
2016-02-19 12:50:39 +00:00
|
|
|
void Xkb::switchToNextLayout()
|
|
|
|
{
|
|
|
|
if (!m_keymap || !m_state) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const xkb_layout_index_t numLayouts = xkb_keymap_num_layouts(m_keymap);
|
|
|
|
const xkb_layout_index_t nextLayout = (xkb_state_serialize_layout(m_state, XKB_STATE_LAYOUT_EFFECTIVE) + 1) % numLayouts;
|
|
|
|
const xkb_mod_mask_t depressed = xkb_state_serialize_mods(m_state, xkb_state_component(XKB_STATE_MODS_DEPRESSED));
|
|
|
|
const xkb_mod_mask_t latched = xkb_state_serialize_mods(m_state, xkb_state_component(XKB_STATE_MODS_LATCHED));
|
|
|
|
const xkb_mod_mask_t locked = xkb_state_serialize_mods(m_state, xkb_state_component(XKB_STATE_MODS_LOCKED));
|
|
|
|
xkb_state_update_mask(m_state, depressed, latched, locked, 0, 0, nextLayout);
|
|
|
|
updateModifiers();
|
|
|
|
}
|
|
|
|
|
2016-02-15 12:42:48 +00:00
|
|
|
KeyboardInputRedirection::KeyboardInputRedirection(InputRedirection *parent)
|
|
|
|
: QObject(parent)
|
|
|
|
, m_input(parent)
|
|
|
|
, m_xkb(new Xkb(parent))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2016-10-05 06:26:29 +00:00
|
|
|
KeyboardInputRedirection::~KeyboardInputRedirection() = default;
|
2016-02-15 12:42:48 +00:00
|
|
|
|
|
|
|
void KeyboardInputRedirection::init()
|
|
|
|
{
|
|
|
|
Q_ASSERT(!m_inited);
|
|
|
|
m_inited = true;
|
|
|
|
|
2016-10-05 06:26:29 +00:00
|
|
|
// setup key repeat
|
|
|
|
m_keyRepeat.timer = new QTimer(this);
|
|
|
|
connect(m_keyRepeat.timer, &QTimer::timeout, this,
|
|
|
|
[this] {
|
|
|
|
if (waylandServer()->seat()->keyRepeatRate() != 0) {
|
|
|
|
m_keyRepeat.timer->setInterval(1000 / waylandServer()->seat()->keyRepeatRate());
|
|
|
|
}
|
|
|
|
// TODO: better time
|
|
|
|
processKey(m_keyRepeat.key, InputRedirection::KeyboardKeyAutoRepeat, m_keyRepeat.time);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2016-02-15 12:42:48 +00:00
|
|
|
connect(workspace(), &QObject::destroyed, this, [this] { m_inited = false; });
|
|
|
|
connect(waylandServer(), &QObject::destroyed, this, [this] { m_inited = false; });
|
2016-06-26 13:57:38 +00:00
|
|
|
connect(workspace(), &Workspace::clientActivated, this,
|
|
|
|
[this] {
|
|
|
|
disconnect(m_activeClientSurfaceChangedConnection);
|
|
|
|
if (auto c = workspace()->activeClient()) {
|
|
|
|
m_activeClientSurfaceChangedConnection = connect(c, &Toplevel::surfaceChanged, this, &KeyboardInputRedirection::update);
|
|
|
|
} else {
|
|
|
|
m_activeClientSurfaceChangedConnection = QMetaObject::Connection();
|
|
|
|
}
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
);
|
2016-04-25 06:51:33 +00:00
|
|
|
if (waylandServer()->hasScreenLockerIntegration()) {
|
|
|
|
connect(ScreenLocker::KSldApp::self(), &ScreenLocker::KSldApp::lockStateChanged, this, &KeyboardInputRedirection::update);
|
|
|
|
}
|
2016-02-19 10:15:37 +00:00
|
|
|
|
2016-02-19 12:50:39 +00:00
|
|
|
QAction *switchKeyboardAction = new QAction(this);
|
|
|
|
switchKeyboardAction->setObjectName(QStringLiteral("Switch to Next Keyboard Layout"));
|
|
|
|
switchKeyboardAction->setProperty("componentName", QStringLiteral("KDE Keyboard Layout Switcher"));
|
|
|
|
const QKeySequence sequence = QKeySequence(Qt::ALT+Qt::CTRL+Qt::Key_K);
|
|
|
|
KGlobalAccel::self()->setDefaultShortcut(switchKeyboardAction, QList<QKeySequence>({sequence}));
|
|
|
|
KGlobalAccel::self()->setShortcut(switchKeyboardAction, QList<QKeySequence>({sequence}));
|
|
|
|
m_input->registerShortcut(sequence, switchKeyboardAction);
|
|
|
|
connect(switchKeyboardAction, &QAction::triggered, this,
|
|
|
|
[this] {
|
|
|
|
m_xkb->switchToNextLayout();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2016-02-22 07:17:45 +00:00
|
|
|
QDBusConnection::sessionBus().connect(QString(),
|
|
|
|
QStringLiteral("/Layouts"),
|
|
|
|
QStringLiteral("org.kde.keyboard"),
|
|
|
|
QStringLiteral("reloadConfig"),
|
|
|
|
this,
|
|
|
|
SLOT(reconfigure()));
|
|
|
|
|
|
|
|
m_xkb->reconfigure();
|
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInputRedirection::reconfigure()
|
|
|
|
{
|
2016-02-19 10:15:37 +00:00
|
|
|
m_xkb->reconfigure();
|
2016-02-15 12:42:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInputRedirection::update()
|
|
|
|
{
|
|
|
|
if (!m_inited) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
auto seat = waylandServer()->seat();
|
|
|
|
// TODO: this needs better integration
|
|
|
|
Toplevel *found = nullptr;
|
|
|
|
if (waylandServer()->isScreenLocked()) {
|
|
|
|
const ToplevelList &stacking = Workspace::self()->stackingOrder();
|
|
|
|
if (!stacking.isEmpty()) {
|
|
|
|
auto it = stacking.end();
|
|
|
|
do {
|
|
|
|
--it;
|
|
|
|
Toplevel *t = (*it);
|
|
|
|
if (t->isDeleted()) {
|
|
|
|
// a deleted window doesn't get mouse events
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!t->isLockScreen()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!t->readyForPainting()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
found = t;
|
|
|
|
break;
|
|
|
|
} while (it != stacking.begin());
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
found = workspace()->activeClient();
|
|
|
|
}
|
|
|
|
if (found && found->surface()) {
|
|
|
|
if (found->surface() != seat->focusedKeyboardSurface()) {
|
|
|
|
seat->setFocusedKeyboardSurface(found->surface());
|
2016-06-20 09:21:16 +00:00
|
|
|
auto newKeyboard = seat->focusedKeyboard();
|
|
|
|
if (newKeyboard && newKeyboard->client() == waylandServer()->xWaylandConnection()) {
|
|
|
|
// focus passed to an XWayland surface
|
|
|
|
const auto selection = seat->selection();
|
|
|
|
auto xclipboard = waylandServer()->xclipboardSyncDataDevice();
|
|
|
|
if (xclipboard && selection != xclipboard.data()) {
|
|
|
|
if (selection) {
|
|
|
|
xclipboard->sendSelection(selection);
|
|
|
|
} else {
|
|
|
|
xclipboard->sendClearSelection();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2016-02-15 12:42:48 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
seat->setFocusedKeyboardSurface(nullptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-24 08:57:57 +00:00
|
|
|
void KeyboardInputRedirection::processKey(uint32_t key, InputRedirection::KeyboardKeyState state, uint32_t time, LibInput::Device *device)
|
2016-02-15 12:42:48 +00:00
|
|
|
{
|
|
|
|
if (!m_inited) {
|
|
|
|
return;
|
|
|
|
}
|
2016-02-19 06:53:20 +00:00
|
|
|
QEvent::Type type;
|
|
|
|
bool autoRepeat = false;
|
|
|
|
switch (state) {
|
|
|
|
case InputRedirection::KeyboardKeyAutoRepeat:
|
|
|
|
autoRepeat = true;
|
|
|
|
// fall through
|
|
|
|
case InputRedirection::KeyboardKeyPressed:
|
|
|
|
type = QEvent::KeyPress;
|
|
|
|
break;
|
|
|
|
case InputRedirection::KeyboardKeyReleased:
|
|
|
|
type = QEvent::KeyRelease;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
Q_UNREACHABLE();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!autoRepeat) {
|
|
|
|
emit m_input->keyStateChanged(key, state);
|
|
|
|
const Qt::KeyboardModifiers oldMods = modifiers();
|
|
|
|
m_xkb->updateKey(key, state);
|
|
|
|
if (oldMods != modifiers()) {
|
|
|
|
emit m_input->keyboardModifiersChanged(modifiers(), oldMods);
|
|
|
|
}
|
2016-02-15 12:42:48 +00:00
|
|
|
}
|
2016-02-19 06:53:20 +00:00
|
|
|
|
2016-08-29 15:04:13 +00:00
|
|
|
const xkb_keysym_t keySym = m_xkb->currentKeysym();
|
2016-05-24 08:57:57 +00:00
|
|
|
KeyEvent event(type,
|
|
|
|
m_xkb->toQtKey(keySym),
|
|
|
|
m_xkb->modifiers(),
|
|
|
|
key,
|
|
|
|
keySym,
|
2016-08-29 15:04:13 +00:00
|
|
|
m_xkb->toString(keySym),
|
2016-05-24 08:57:57 +00:00
|
|
|
autoRepeat,
|
|
|
|
time,
|
|
|
|
device);
|
2016-02-19 06:53:20 +00:00
|
|
|
if (state == InputRedirection::KeyboardKeyPressed) {
|
2016-02-19 09:56:17 +00:00
|
|
|
if (m_xkb->shouldKeyRepeat(key) && waylandServer()->seat()->keyRepeatDelay() != 0) {
|
2016-10-05 06:26:29 +00:00
|
|
|
m_keyRepeat.timer->setInterval(waylandServer()->seat()->keyRepeatDelay());
|
|
|
|
m_keyRepeat.key = key;
|
|
|
|
m_keyRepeat.time = time;
|
|
|
|
m_keyRepeat.timer->start();
|
2016-02-19 06:53:20 +00:00
|
|
|
}
|
|
|
|
} else if (state == InputRedirection::KeyboardKeyReleased) {
|
2016-10-05 06:26:29 +00:00
|
|
|
if (key == m_keyRepeat.key) {
|
|
|
|
m_keyRepeat.timer->stop();
|
2016-02-19 06:53:20 +00:00
|
|
|
}
|
|
|
|
}
|
2016-02-15 12:42:48 +00:00
|
|
|
|
|
|
|
const auto &filters = m_input->filters();
|
|
|
|
for (auto it = filters.begin(), end = filters.end(); it != end; it++) {
|
|
|
|
if ((*it)->keyEvent(&event)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInputRedirection::processModifiers(uint32_t modsDepressed, uint32_t modsLatched, uint32_t modsLocked, uint32_t group)
|
|
|
|
{
|
|
|
|
if (!m_inited) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// TODO: send to proper Client and also send when active Client changes
|
|
|
|
Qt::KeyboardModifiers oldMods = modifiers();
|
|
|
|
m_xkb->updateModifiers(modsDepressed, modsLatched, modsLocked, group);
|
|
|
|
if (oldMods != modifiers()) {
|
|
|
|
emit m_input->keyboardModifiersChanged(modifiers(), oldMods);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void KeyboardInputRedirection::processKeymapChange(int fd, uint32_t size)
|
|
|
|
{
|
|
|
|
if (!m_inited) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// TODO: should we pass the keymap to our Clients? Or only to the currently active one and update
|
|
|
|
m_xkb->installKeymap(fd, size);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|