使用XKeycodeToKeysym
(在C ++中)我可以获得按键事件的KeySym。从我的理解,这已经应该尊重键盘布局。但是,当我切换键盘布局(英语和希伯来语之间)时,我得到了相同的KeySym。我怀疑Xlib Keysym只尊重X11级别中定义的键盘布局?在我的系统中,键盘布局仅在桌面环境级别(Mate)中定义。如果是这样,有没有办法在不使用像Qt / GTK这样的工具包的情况下获得正确的角色?我必须单独处理每个桌面环境吗?
[编辑]
我已经尝试了以下(根据安德烈的建议),但这并不起作用:
#include <X11/XKBlib.h>
#include <cstring>
#include <cassert>
#include <iostream>
int main() {
Display *display = XOpenDisplay(nullptr);
Window root, window;
XSetWindowAttributes swa;
root = DefaultRootWindow( display );
XSelectInput( display, root, MappingNotify );
std::memset(&swa, 0, sizeof swa );
swa.event_mask = MappingNotify | KeyPressMask;
window = XCreateWindow( display, root, 50, 200, 1024, 768, 0, CopyFromParent, InputOutput, CopyFromParent,
CWEventMask, &swa );
XMapRaised( display, window );
int xkbEventCode=0, n0=0, n1=0, n2=0, n3=0;
bool isOk = XkbQueryExtension( display, &n0, &xkbEventCode, &n1, &n2, &n3 ) ==True;
assert( isOk );
isOk = XkbSelectEvents( display, XkbUseCoreKbd, XkbAllEventsMask, XkbAllEventsMask ) ==True;
assert( isOk );
while (true) {
XEvent event;
std::memset( &event, 0, sizeof event );
XNextEvent( display, &event );
if (event.type == xkbEventCode)
switch (reinterpret_cast<XkbEvent*>(&event)->any.xkb_type) {
case XkbNewKeyboardNotify:
case XkbMapNotify: {
std::cout << "Keyboard mapping has changed." << std::endl;
break;
}
default: break;
}
else
switch (event.type) {
case KeyPress: {
KeyCode keyCode = event.xkey.keycode;
int keySymsPerkeyCode=0;
KeySym *keySyms( XGetKeyboardMapping(display, keyCode, 1, &keySymsPerkeyCode)),
*keySym = keySyms;
while (keySymsPerkeyCode-- && *keySym != NoSymbol) {
std::cout << *keySym << std::endl;
++keySym;
}
XFree( keySyms );
break;
}
case MappingNotify: {
std::cout << "Keyboard mapping has changed." << std::endl;
break;
}
default: break;
}
}
return 0;
}
答案 0 :(得分:1)
我只能在较低级别回答,不知道XKeycodeToKeysym
做了什么。
您需要收听MappingNotify事件,并且每次收到该事件时都会使用XGetKeyboardMapping
重新阅读新映射(通常这意味着键盘布局/语言已更改)这是核心X协议,桌面环境/ WM独立的一部分
<强>更新强>
看起来xlib默认启用XKeyboard扩展。启用后,您需要明确表达对映射更改事件的兴趣(与核心协议不同,其中MappingNotify始终发送给所有客户端) - 请参阅http://www.x.org/archive/X11R7.5/doc/man/man3/XkbSelectEvents.3.html
XkbSelectEvents (display, XkbUseCoreKbd, XkbAllEventsMask, XkbAllEventsMask);
之后,您回复XkbMapNotify
事件并执行XkbGetMap()以获得新的布局