使用c#捕获usb扫描枪扫描二维码、条形码结果(支持中文版)

目录

前因

代码

思路

左右shift及其他

尝试获得中文返回内容

处理中文信息

注意编码及解码


前因

之前在18年写过一个使用c#捕获usb扫描枪扫描二维码、条形码结果,当时我是没有遇到需要使用中文的情况,因为二维码都是我自己控制生成的,如果遇到中文,我会提前进行编码编程unicode编码,所以,没有专门针对中文符号做支持

但一年多以来,不少人询问,或者在博客下留言,提出了中文扫码支持的需要,于是老顾就从新把程序从故纸堆里翻了出来,毕竟很久没弄这个了,还得自己重新读一遍代码,真晕。。。。。

代码

好在,中文支持已经实现了

        public class ScanerCodes
        {
            private int ts = 500; // 指定输入间隔为500毫秒以内时为连续输入
            private List<List<EventMsg>> _keys = new List<List<EventMsg>>();
            private List<int> _keydown = new List<int>();   // 保存组合键状态
            private List<string> _result = new List<string>();  // 返回结果集
            private List<string> _alt = new List<string>();     // 保存alt按下时的各键位
            private DateTime _last = DateTime.Now;
            private byte[] _state = new byte[256];
            private string _key = string.Empty;
            private string _cur = string.Empty;
            public EventMsg Event
            {
                get
                {
                    if (_keys.Count == 0)
                    {
                        return new EventMsg();
                    }
                    else
                    {
                        return _keys[_keys.Count - 1][_keys[_keys.Count - 1].Count - 1];
                    }
                }
            }
            public List<int> KeyDowns
            {
                get
                {
                    return _keydown;
                }
            }
            public DateTime LastInput
            {
                get
                {
                    return _last;
                }
            }
            public byte[] KeyboardState
            {
                get
                {
                    return _state;
                }
            }
            public int KeyDownCount
            {
                get
                {
                    return _keydown.Count;
                }
            }
            public string Result
            {
                get
                {
                    if (_result.Count > 0)
                    {
                        return _result[_result.Count - 1].Trim();
                    }
                    else
                    {
                        return null;
                    }
                }
            }
            public string CurrentKey
            {
                get
                {
                    return _key;
                }
            }
            public string CurrentChar
            {
                get
                {
                    return _cur;
                }
            }
            public bool isShift
            {
                get
                {
                    return _keydown.Contains(160) || _keydown.Contains(161);
                }
            }
            public bool isAlt
            {
                get
                {
                    return _keydown.Contains(164) || _keydown.Contains(165);
                }
            }
            public bool isControl
            {
                get
                {
                    return _keydown.Contains(162) || _keydown.Contains(163);
                }
            }
            public void Clear()
            {
                _keys.Clear();
                _result.Clear();
            }
            public void Add(EventMsg msg)
            {
                #region 记录按键信息
                // 首次按下按键
                if (_keys.Count == 0)
                {
                    _keys.Add(new List<EventMsg>());
                    _keys[0].Add(msg);
                    _result.Add(string.Empty);
                }
                // 未释放其他按键时按下按键
                else if (_keydown.Count > 0)
                {
                    _keys[_keys.Count - 1].Add(msg);
                }
                // 单位时间内按下按键
                else if (((TimeSpan)(DateTime.Now - _last)).TotalMilliseconds < ts)
                {
                    _keys[_keys.Count - 1].Add(msg);
                }
                // 从新记录输入内容
                else
                {
                    _keys.Add(new List<EventMsg>());
                    _keys[_keys.Count - 1].Add(msg);
                    _result.Add(string.Empty);
                }
                if (_keys.Count > 10)
                {
                    _keys.RemoveAt(0);
                }
                #endregion
                _last = DateTime.Now;
                #region 获取键盘状态
                // 记录正在按下的按键
                if (msg.paramH < 128 && !_keydown.Contains(msg.message))
                {
                    _keydown.Add(msg.message);
                }
                // 清除已松开的按键
                if (msg.paramH > 127 && _keydown.Contains(msg.message))
                {
                    _keydown.Remove(msg.message);
                }
                #endregion
                #region 计算按键信息
                int v = msg.message & 0xff;
                int c = msg.paramL & 0xff;
                StringBuilder strKeyName = new StringBuilder(500);
                if (GetKeyNameText(c * 65536, strKeyName, 255) > 0)
                {
                    _key = strKeyName.ToString().Trim(new char[] { ' ', '\0' });
                    GetKeyboardState(_state);
                    if (_key.Length == 1 && msg.paramH < 128)
                    {
                        // 根据键盘状态和shift缓存判断输出字符
                        _cur = ShiftChar(_key, isShift, _state).ToString();
                        _result[_result.Count - 1] += _cur;
                    }
                    else if (msg.paramH < 128 && isAlt)
                    {

                        _alt.Add(_key);
                    }
                    else if (msg.paramH > 127 && (msg.message == 164 || msg.message == 165))
                    {
                        string composite = string.Empty;
                        for (int i = 0; i < _alt.Count; i++)
                        {
                            composite += _alt[i];
                        }
                        composite = Regex.Replace(composite, @"[^0-9]+", "");
                        string hex = Convert.ToInt32(composite).ToString("x");
                        if (hex.Length % 2 == 1)
                        {
                            hex = "0" + hex;
                        }
                        string str = Regex.Replace(hex, @"([0-9a-f]{2})", "%$1");
                        _cur = System.Web.HttpUtility.UrlDecode(str, Encoding.GetEncoding(936));
                        _result[_result.Count - 1] += _cur;
                        _alt.Clear();
                    }
                    else
                    {
                        _cur = string.Empty;
                    }
                }
                #endregion
            }
            private char ShiftChar(string k, bool isShiftDown, byte[] state)
            {
                bool capslock = state[0x14] == 1;
                bool numlock = state[0x90] == 1;
                bool scrolllock = state[0x91] == 1;
                bool shiftdown = state[0xa0] == 1;
                char chr = (capslock ? k.ToUpper() : k.ToLower()).ToCharArray()[0];
                if (isShiftDown)
                {
                    if (chr >= 'a' && chr <= 'z')
                    {
                        chr = (char)((int)chr - 32);
                    }
                    else if (chr >= 'A' && chr <= 'Z')
                    {
                        chr = (char)((int)chr + 32);
                    }
                    else
                    {
                        string s = "`1234567890-=[];',./";
                        string u = "~!@#$%^&*()_+{}:\"<>?";
                        if (s.IndexOf(chr) >= 0)
                        {
                            return (u.ToCharArray())[s.IndexOf(chr)];
                        }
                    }
                }
                return chr;
            }
        }

说起来,其实也没有修改很多东西,只对ScanerCodes这个类做了下微调,其他代码我就不再复制过来了,请参考原文

思路

以下是中文支持的思路和尝试过程

左右shift及其他

首先,根据原文中不吃辣的阿银同学提出的问题,左右shift键paramH值不同的问题专门测试了一下,发现的确存在这种问题

那么第一步就是调整isShift,这里有个测试结果,有兴趣继续魔改的可以看一下

paramH,message
0,160	// Lshift down
128,160 // Lshift up
1,161	// Rshift down
129,161 // Rshift up
0,162   // Lctrl down
128,162 // Lctrl up
1,163	// Rctrl down
129,163	// Rctrl up
32,164	// Lalt down
128,164	// Lalt up
33,165	// Ralt down
129,165 // Ralt up

我这里使用的还是科密扫描枪,如果其他型号扫描枪有不同的结果,请自行尝试

isShift修改为判断已按下的键中,是否存在左右shift键对应的编号

尝试获得中文返回内容

然后,尝试进行中文扫码,直接一脸懵逼了。。。。所有的_key得到的结果都是长度大于1的内容,这都什么和什么啊,然后发现捕捉到了32,164的组合,这不是按alt键了吗?一脸疑惑中。。。。

然后通过调试发现,之后的键盘信息_key给我的是 Numpad 4,Numpad 5,Numpad 7,Numpad 9,Numpad 4。。。。。。

一脸惊讶!我是知道按住alt不送,直接在数字小键盘上输入数字可以出现任意符号的,难道扫描枪扫到中文也是这么搞得?

直接用数字小键盘尝试了下Alt+45794。。。。出现了一个“测”字,好吧。。。。这个是我生成的二维码。。。。。

处理中文信息

那么有了这个结果,后边就相对简单了,直接加一个状态,isAlt,嗯,顺便支持左右Alt

然后重新定义一个数组_alt,用来记录alt键释放前的所有键盘信息

最后,在alt键释放的时候将_alt数组中的信息组合起来,得到小键盘数字顺序,并清空_alt数组,以便下一次继续获取中文符号

注意编码及解码

好了,得到了数字后就需要把数字转成中文了,然后当我使用(char)45794转出来的内容是个韩文字符的时候,我差点崩溃了。。。NND,扫描枪居然有字符集设置,而不是使用的unicode!

行吧,如果有人购买的扫描枪在数字转字符时不对,那一定是编码的锅,自己测试到底是什么编码吧!

于是,我就将数字转成16进制,然后补%,冒充urldecode的结果,然后使用gb2312进行urldecode,得到了正确结果,中文支持完成

对于使用得力扫描枪扫描二维码,你可以使用以下步骤在C#中实现: 1. 确保你的得力扫描枪已经连接到计算机,并且被识别为HID(Human Interface Device)设备。 2. 使用Windows API来读取扫描枪的输入数据。你可以通过使用`RegisterRawInputDevices`函数来注册输入设备,并使用`WM_INPUT`消息来接收输入数据。 下面是一个基本的示例代码,演示了如何使用C#来处理得力扫描枪的输入数据: ```csharp using System; using System.Runtime.InteropServices; using System.Windows.Forms; class Program { private const int WM_INPUT = 0x00FF; private const int RID_INPUT = 0x10000003; private const int RIDEV_INPUTSINK = 0x00000100; private const int RIDEV_DEVNOTIFY = 0x00002000; [StructLayout(LayoutKind.Sequential)] struct RAWINPUTDEVICELIST { public IntPtr hDevice; public int dwType; } [DllImport("user32.dll")] static extern uint GetRawInputDeviceList(IntPtr pRawInputDeviceList, ref uint uiNumDevices, uint cbSize); [DllImport("user32.dll")] static extern bool RegisterRawInputDevices(RAWINPUTDEVICELIST[] pRawInputDeviceList, uint uiNumDevices, uint cbSize); [DllImport("user32.dll")] static extern uint GetRawInputData(IntPtr hRawInput, uint uiCommand, IntPtr pData, ref uint pcbSize, uint cbSizeHeader); static void Main() { // 注册输入设备 RAWINPUTDEVICELIST[] rawInputDeviceList = new RAWINPUTDEVICELIST[1]; rawInputDeviceList[0].dwType = RID_INPUT; rawInputDeviceList[0].hDevice = IntPtr.Zero; if (!RegisterRawInputDevices(rawInputDeviceList, 1, (uint)Marshal.SizeOf<RAWINPUTDEVICELIST>())) { Console.WriteLine("无法注册输入设备。"); return; } // 创建窗口来接收输入消息 NativeWindow window = new NativeWindow(); window.CreateHandle(new CreateParams()); // 循环接收输入消息 while (true) { Application.DoEvents(); } } protected override void WndProc(ref Message m) { if (m.Msg == WM_INPUT) { // 读取输入数据 uint size = 0; GetRawInputData(m.LParam, RID_INPUT, IntPtr.Zero, ref size, (uint)Marshal.SizeOf<RAWINPUTHEADER>()); IntPtr buffer = Marshal.AllocHGlobal((int)size); try { GetRawInputData(m.LParam, RID_INPUT, buffer, ref size, (uint)Marshal.SizeOf<RAWINPUTHEADER>()); // 在这里处理扫描枪的输入数据 // 解析二维码数据并进行相关操作 // 例如,你可以通过ZXing库来解码二维码 // 你可以在这里将二维码数据传递给其他方法进行处理 ProcessQRCodeData(buffer); } finally { Marshal.FreeHGlobal(buffer); } } base.WndProc(ref m); } private void ProcessQRCodeData(IntPtr buffer) { // 在这里处理扫描枪输入的二维码数据 // 解析二维码并进行相关操作 // 例如,你可以使用ZXing库来解码二维码 // 以下是一个使用ZXing库解码二维码的示例 // 你需要安装ZXing.Net库,通过NuGet包管理器安装 ZXing.BarcodeReader barcodeReader = new ZXing.BarcodeReader(); ZXing.Result result = barcodeReader.Decode(new Bitmap(/* 这里是图像数据来源 */)); if (result != null) { string decodedText = result.Text; Console.WriteLine("解码结果: " + decodedText); } else { Console.WriteLine("未能解码二维码。"); } } } ``` 请注意,上述代码只是一个基本示例,你需要根据实际情况进行适当的修改和调整。在代码中,我们使用`WM_INPUT`消息来接收扫描枪的输入数据,并通过`RegisterRawInputDevices`函数注册输入设备。然后,我们创建了一个窗口来接收输入消息,并在`WndProc`方法中处理输入数据。你可以在`ProcessQRCodeData`方法中解析扫描枪输入的二维码数据。 希望这可以帮助到你!如果你有任何其他问题,请随时提问。
评论 20
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

文盲老顾

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值