我之前已经浏览过几个关于此的问题,到目前为止我找到的最佳答案是这样的:
(char) WinAPI.MapVirtualKey((uint) Keys.A, 2)
但是,这在两个方面不起作用:
它总是返回大写字母。对于Keys.A
我希望获得字符a
,而对于Keys.A | Keys.ShiftKey
,我希望获得A
;但是,我似乎都得到了A
。
似乎没有考虑键盘布局。例如,对于Keys.OemMinus
我似乎总是得到字符-
,即使当前的键盘布局是德语,我希望此键返回ß
。
对此有什么正确的解决方案?
答案 0 :(得分:23)
正确的解决方案是ToUnicode
WinAPI函数:
[DllImport("user32.dll")]
public static extern int ToUnicode(uint virtualKeyCode, uint scanCode,
byte[] keyboardState,
[Out, MarshalAs(UnmanagedType.LPWStr, SizeConst = 64)]
StringBuilder receivingBuffer,
int bufferSize, uint flags);
将这种方法包装成一种明智,方便的方法是:
static string GetCharsFromKeys(Keys keys, bool shift, bool altGr)
{
var buf = new StringBuilder(256);
var keyboardState = new byte[256];
if (shift)
keyboardState[(int) Keys.ShiftKey] = 0xff;
if (altGr)
{
keyboardState[(int) Keys.ControlKey] = 0xff;
keyboardState[(int) Keys.Menu] = 0xff;
}
WinAPI.ToUnicode((uint) keys, 0, keyboardState, buf, 256, 0);
return buf.ToString();
}
现在我们可以检索字符并实际获得预期结果:
Console.WriteLine(GetCharsFromKeys(Keys.E, false, false)); // prints e
Console.WriteLine(GetCharsFromKeys(Keys.E, true, false)); // prints E
// Assuming British keyboard layout:
Console.WriteLine(GetCharsFromKeys(Keys.E, false, true)); // prints é
Console.WriteLine(GetCharsFromKeys(Keys.E, true, true)); // prints É
还可以使用ToUnicodeEx
来检索不是当前活动的键盘布局的字符。签名是相同的,除了一个额外的参数,输入区域设置ID,可以使用LoadKeyboardLayout
函数检索。
答案 1 :(得分:1)
我认为这可以通过这种方法实现:
[DllImportAttribute("User32.dll")]
public static extern int ToAscii(int uVirtKey, int uScanCode, byte[] lpbKeyState,
byte[] lpChar, int uFlags);
可以在此处找到示例用法: http://www.pcreview.co.uk/forums/toascii-function-t1706394.html
答案 2 :(得分:1)
[DllImport("user32.dll")]
static extern int MapVirtualKey(int uCode, uint uMapType);
private void textBox1_OnKeyDown(object sender, KeyEventArgs e)
{
char c = (char)MapVirtualKey((int)e.KeyData, (uint)2);
if (char.IsNumber(c)) DoSomething();
else if (!char.IsNumber(c)) DoNothing();
}