我想编写一个应用程序,其中键绑定特定于键盘上键的位置,而不是它们映射到的字符。例如,美式键盘上介于 t 和 u 之间的键应该执行特定功能,无论它是在美国是 Y 还是在德国是 Z。
我认为这样做的方法是获取键盘给操作系统的实际扫描代码,以表示按下的键。我怎样才能在java中做到这一点?
还是有其他方法可以实现相同的功能?
摘自 Oracle 的 KeyEvent 源代码:
//set from native code.
private transient long rawCode = 0;
private transient long primaryLevelUnicode = 0;
private transient long scancode = 0; // for MS Windows only
private transient long extendedKeyCode = 0;
首先,我考虑解析 KeyEvent 的toString()返回值,因为它包含扫描码。但后来我写了一个使用反射的实用方法(在 Windows 8 上成功尝试。) :
final public static Integer getScancodeFromKeyEvent(final KeyEvent keyEvent) {
Integer ret;
Field field;
try {
field = KeyEvent.class.getDeclaredField("scancode");
} catch (NoSuchFieldException nsfe) {
System.err.println("ATTENTION! The KeyEvent object does not have a field named \"scancode\"! (Which is kinda weird.)");
nsfe.printStackTrace();
return null;
}
try {
field.setAccessible(true);
} catch (SecurityException se) {
System.err.println("ATTENTION! Changing the accessibility of the KeyEvent class' field \"scancode\" caused a security exception!");
se.printStackTrace();
return null;
}
try {
ret = (int) field.getLong(keyEvent);
} catch (IllegalAccessException iae) {
System.err.println("ATTENTION! It is not allowed to read the field \"scancode\" of the KeyEvent instance!");
iae.printStackTrace();
return null;
}
return ret;
}
显然,之后重新设置字段的可访问性是不必要的,因为我在使用此方法时遇到了异常,而 setAccessible() 行被注释掉了。(我来回更改它并在运行时仍在运行时在 IntelliJ 中重新编译。在 Eclipse 中应该是相同的。)但是,首先使用 isAccessible() 方法很容易实现。
我需要扫描码来在键盘上播放音乐,因为在 QWERTZ 和 QWERTY 之间更改键盘语言会自动交换音符。很遗憾,我们无法合法访问类似扫描码的值。上述解决方案成功忽略了当前的键盘布局配置。
顺便说一句,美国键盘布局的“z”和“y”的 toString 返回值:
[KEY_PRESSED,keyCode=90,keyText=Z,keyChar='z',keyLocation=KEY_LOCATION_STANDARD,rawCode=90,primaryLevelUnicode=122,scancode=44,extendedKeyCode=0x5a] on frame0]
[KEY_PRESSED,keyCode=89,keyText=Y,keyChar='y',keyLocation=KEY_LOCATION_STANDARD,rawCode=89,primaryLevelUnicode=121,scancode=21,extendedKeyCode=0x59] on frame0]
使用 DE 键盘布局:
[KEY_PRESSED,keyCode=89,keyText=Y,keyChar='y',keyLocation=KEY_LOCATION_STANDARD,rawCode=89,primaryLevelUnicode=121,scancode=44,extendedKeyCode=0x59] on frame0]
[KEY_PRESSED,keyCode=90,keyText=Z,keyChar='z',keyLocation=KEY_LOCATION_STANDARD,rawCode=90,primaryLevelUnicode=122,scancode=21,extendedKeyCode=0x5a] on frame0]
注意扫描码。
作为奖励,题外话:
[KEY_PRESSED,keyCode=10,keyText=Enter,keyChar=Enter,keyLocation=KEY_LOCATION_STANDARD,rawCode=13,primaryLevelUnicode=13,scancode=28,extendedKeyCode=0xa] on frame0]
[KEY_PRESSED,keyCode=10,keyText=Enter,keyChar=Enter,keyLocation=KEY_LOCATION_NUMPAD,rawCode=13,primaryLevelUnicode=13,scancode=28,extendedKeyCode=0xa] on frame0]
[KEY_PRESSED,keyCode=49,keyText=1,keyChar='1',keyLocation=KEY_LOCATION_STANDARD,rawCode=49,primaryLevelUnicode=49,scancode=2,extendedKeyCode=0x31] on frame0]
[KEY_PRESSED,keyCode=97,keyText=NumPad-1,keyChar='1',keyLocation=KEY_LOCATION_NUMPAD,rawCode=97,primaryLevelUnicode=49,scancode=79,extendedKeyCode=0x61] on frame0]
正如MadProgrammer所说:你必须使用 JNA 或 JNI。您还可以查看这些项目:
JIntellitype 是一个 Java API,用于与 Microsoft Intellitype 命令交互以及在 Java 应用程序中注册全局热键。API 是一个 Java JNI 库,它使用 C++ DLL 与 Windows 进行所有通信。
JNativeHook 是一个为 Java 提供全局键盘和鼠标侦听器的库。这将允许您监听全局快捷方式或鼠标移动,否则使用纯 Java 是不可能的。为了完成这项任务,JNativeHook 通过 Java 的本机接口利用平台相关的本机代码来创建低级系统范围的挂钩并将这些事件传递给您的应用程序。
仅限 Windows,支持 Win 7 / 8(32 位和 64 位)