从使用Raw Input API检索的按键集合中获取一个字符串

本文关键字:获取 字符串 一个 集合 Raw Input API 检索 | 更新日期: 2023-09-27 18:10:37

我使用Raw Input API从键盘(实际上是一个模拟键盘的磁条读卡器)获取按键的集合。下面是一些代码摘录,这样您就可以了解我是如何获得密钥的。

[StructLayout(LayoutKind.Sequential)]
internal struct RAWKEYBOARD
{
    [MarshalAs(UnmanagedType.U2)]
    public ushort MakeCode;
    [MarshalAs(UnmanagedType.U2)]
    public ushort Flags;
    [MarshalAs(UnmanagedType.U2)]
    public ushort Reserved;
    [MarshalAs(UnmanagedType.U2)]
    public ushort VKey;
    [MarshalAs(UnmanagedType.U4)]
    public uint Message;
    [MarshalAs(UnmanagedType.U4)]
    public uint ExtraInformation;
}
[StructLayout(LayoutKind.Explicit)]
internal struct RAWINPUT
{
    [FieldOffset(0)]
    public RAWINPUTHEADER header;
    [FieldOffset(16)]
    public RAWMOUSE mouse;
    [FieldOffset(16)]
    public RAWKEYBOARD keyboard;
    [FieldOffset(16)]
    public RAWHID hid;
}
Queue<char> MyKeys = new Queue<char>();
// buffer has the result of a GetRawInputData() call
RAWINPUT raw = (RAWINPUT)Marshal.PtrToStructure(buffer, typeof(RAWINPUT));
MyKeys.Enqueue((char)raw.keyboard.VKey);

运行代码时,读卡器输出字符串%B40^TEST,但在MyKeys集合中,我有以下值:

{ 16 '',  53 '5', 16 '', 66 'B',
  52 '4', 48 '0', 16 '', 54 '6',
  16 '',  84 'T', 16 '', 69 'E',
  16 '',  83 'S', 16 '', 84 'T' }

这些看起来像是实际按键的集合(呸!),而不是它们所代表的字符串。键码16似乎是Shift,所以在读卡器当前配置的键盘映射中,使用Shift+5产生%字符,用{16,53}表示。后面的字符大写BShift+B或{16,66}。其他字符也是如此。

显然,简单地将这些转换为char(就像我现在所做的)不是要走的路。所以,我的问题是:我如何将这个数组的按键转换成它们所代表的字符串?

从使用Raw Input API检索的按键集合中获取一个字符串

在做了一些额外的研究之后,我自己找到了答案。我把它贴出来给其他读到这篇文章的人。下面是一个小的测试应用程序,演示了如何将一组虚拟键(ushort键代码)转换为字符串表示形式。我使用问题中描述的集合作为输入。

class Program
{
    [DllImport("user32.dll")]
    static extern int MapVirtualKey(uint uCode, uint uMapType);
    [DllImport("user32.dll")]
    private static extern int ToAscii(uint uVirtKey, uint uScanCode, byte[] lpKeyState, [Out] StringBuilder lpChar, uint uFlags);
    static void Main(string[] args)
    {
        byte[] keyState = new byte[256];
        ushort[] input = { 16, 53, 16, 66, 52, 48, 16, 54, 16, 84, 16, 69, 16, 83, 16, 84 };
        StringBuilder output = new StringBuilder();
        foreach (ushort vk in input)
            AppendChar(output, vk, ref keyState);
        Console.WriteLine(output);
        Console.ReadKey(true);
    }
    private static void AppendChar(StringBuilder output, ushort vKey, ref byte[] keyState)
    {
        if (MapVirtualKey(vKey, 2) == 0)
        {
            keyState[vKey] = 0x80;
        }
        else
        {
            StringBuilder chr = new StringBuilder(2);
            int n = ToAscii(vKey, 0, keyState, chr, 0);
            if (n > 0)
                output.Append(chr.ToString(0, n));
            keyState = new byte[256];
        }
    }
}