C#下usb条码扫描枪的钩子实现的改进
2014-02-22 12:33
1426 查看
【目前的条形码扫描器有点类似外接键盘(其实从消息传送上它就相当于一个键盘),把输入焦点定位到可输入的控件上,一扫描相应的条形码信息就输入到文本框中去了,但是如果没有输入焦点,或另一个不相干的程序获得输入焦点,那就有点乱套了。我想实现的是,不管什么情况,只要扫描器一工作,我的程序就能自动激活,并能获得当前输入的条形码信息。 实现思路:我用的是litele牌的USB口的红外条形码扫描器,仔细分析了一下,扫描成功后,以键盘按键消息的形式把条形码输入信息通知给系统。这样通过键盘钩子就可以方便的获得该信息了。但是,怎样区分信息是键盘还是条形码输入的哪?很简单,条形码扫描器在很短的时间内输入了至少3个字符以上信息,并且以“回车”作为结束字符,在这种思想指引下,很完美的实现了预定功能。】
frmMain:
public BarCodeHook BarCode = new BarCodeHook();
public delegate void ShowInfoDelegate(BarCodeHook.BarCodes barCode);
void ShowInfo(BarCodeHook.BarCodes barCode){
textBox_barCode.Text = barCode.BarCode;
buttonX2.Focus();
}
public void BarCode_BarCodeEvent(BarCodeHook.BarCodes barCode)
{
ShowInfo(barCode);
}
public frmMain()
{
InitializeComponent();
BarCode.BarCodeEvent += new BarCodeHook.BarCodeDelegate(BarCode_BarCodeEvent);
}
如果焦点本来就在textBox上,会产生多余的字符,所以在showInfo函数里,每次都手动让buttonX2成为焦点。
是如果扫到的是英文字符的话,会有一个多余的码无法从键盘码转到ascii码,需要去掉这个码。同时设为30ms可以过滤掉一直按住一个键的情况。
frmMain:
public BarCodeHook BarCode = new BarCodeHook();
public delegate void ShowInfoDelegate(BarCodeHook.BarCodes barCode);
void ShowInfo(BarCodeHook.BarCodes barCode){
textBox_barCode.Text = barCode.BarCode;
buttonX2.Focus();
}
public void BarCode_BarCodeEvent(BarCodeHook.BarCodes barCode)
{
ShowInfo(barCode);
}
public frmMain()
{
InitializeComponent();
BarCode.BarCodeEvent += new BarCodeHook.BarCodeDelegate(BarCode_BarCodeEvent);
}
如果焦点本来就在textBox上,会产生多余的字符,所以在showInfo函数里,每次都手动让buttonX2成为焦点。
public class BarCodeHook { public delegate void BarCodeDelegate(BarCodes barCode); public event BarCodeDelegate BarCodeEvent; public struct BarCodes { public int VirtKey; //虚拟码 public int ScanCode; //扫描码 public string KeyName; //键名 public uint AscII; //AscII public char Chr; //字符 public string BarCode; //条码信息 public bool IsValid; //条码是否有效 public DateTime Time; //扫描时间 } private struct EventMsg { public int message; public int paramL; public int paramH; public int Time; public int hwnd; } [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)] private static extern int SetWindowsHookEx(int idHook, HookProc lpfn, IntPtr hInstance, int threadId); [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)] private static extern bool UnhookWindowsHookEx(int idHook); [DllImport("user32.dll", CharSet = CharSet.Auto, CallingConvention = CallingConvention.StdCall)] private static extern int CallNextHookEx(int idHook, int nCode, Int32 wParam, IntPtr lParam); [DllImport("user32", EntryPoint = "GetKeyNameText")] private static extern int GetKeyNameText(int lParam, StringBuilder lpBuffer, int nSize); [DllImport("user32", EntryPoint = "GetKeyboardState")] private static extern int GetKeyboardState(byte[] pbKeyState); [DllImport("user32", EntryPoint = "ToAscii")] private static extern bool ToAscii(int VirtualKey, int ScanCode, byte[] lpKeyState, ref uint lpChar, int uFlags); delegate int HookProc(int nCode, Int32 wParam, IntPtr lParam); BarCodes barCode = new BarCodes(); int hKeyboardHook = 0; public string strBarCode = ""; public int length; private int KeyboardHookProc(int nCode, Int32 wParam, IntPtr lParam) { barCode.IsValid = false; bool notChar = false; if (nCode == 0) { EventMsg msg = (EventMsg)Marshal.PtrToStructure(lParam, typeof(EventMsg)); if (wParam == 0x100) //WM_KEYDOWN = 0x100 { barCode.VirtKey = msg.message & 0xff; //虚拟码 barCode.ScanCode = msg.paramL & 0xff; //扫描码 StringBuilder strKeyName = new StringBuilder(255); if (GetKeyNameText(barCode.ScanCode * 65536, strKeyName, 255) > 0) { barCode.KeyName = strKeyName.ToString().Trim(new char[] { ' ', '\0' }); } else { barCode.KeyName = ""; } byte[] kbArray = new byte[256]; uint uKey = 0; GetKeyboardState(kbArray); if (ToAscii(barCode.VirtKey, barCode.ScanCode, kbArray, ref uKey, 0)) { barCode.AscII = uKey; barCode.Chr = Convert.ToChar(uKey); } else { notChar = true; //转到ascii字符失败,这不是一个正常字符,要去掉 } if (DateTime.Now.Subtract(barCode.Time).TotalMilliseconds > 30) //30ms可以过滤掉连续按住一个键时的情况 { if (notChar == false) strBarCode = barCode.Chr.ToString(); else strBarCode = ""; barCode.IsValid = false; } else { if (strBarCode.Length >= 5) { barCode.IsValid = true; //isValid为true表明这是个条码 } if (notChar == false) { strBarCode += barCode.Chr.ToString(); } barCode.BarCode = strBarCode; } barCode.Time = DateTime.Now; if (BarCodeEvent != null && barCode.IsValid) BarCodeEvent(barCode); //触发事件 } } return CallNextHookEx(hKeyboardHook, nCode, wParam, lParam); } // 安装钩子 public bool Start() { if (hKeyboardHook == 0) { //WH_KEYBOARD_LL = 13 hKeyboardHook = SetWindowsHookEx(13, new HookProc(KeyboardHookProc), Marshal.GetHINSTANCE(Assembly.GetExecutingAssembly().GetModules()[0]), 0); } return (hKeyboardHook != 0); } // 卸载钩子 public bool Stop() { if (hKeyboardHook != 0) { bool result = UnhookWindowsHookEx(hKeyboardHook); hKeyboardHook = 0; //将hKeyboardHook 置为0 if (result) { //MessageBox.Show("true"); } return result; } return true; } }
是如果扫到的是英文字符的话,会有一个多余的码无法从键盘码转到ascii码,需要去掉这个码。同时设为30ms可以过滤掉一直按住一个键的情况。
相关文章推荐
- c#调用COM组件
- c#中虚函数的相关使用方法
- 浅析C# web访问mysql数据库-整理归纳总结
- C# IP地址与整数之间转换的具体方法
- C#中的多态深入理解
- C#实现的阴历阳历互相转化类实例
- C# 的关键字详细介绍
- c# 获得局域网主机列表实例
- C#基本概念列举详解
- C#中Convert.ToInt32()和int.Parse()的区别介绍
- C#面向对象特征的具体实现及作用详解
- c#静态方法和非静态方法详细介绍
- 关于C# Math 处理奇进偶不进的实现代码
- C#中调用Windows API的技术要点说明
- c# 随机函数的使用详解
- c#之圆形无标题栏椭圆窗体的实现详解
- C#页面之间跳转功能的小结
- 浅析C# 基础语法的使用
- C#中将字符串转换为整型的三种解决方法总结
- 深入c# GDI+简单绘图的具体操作步骤(一)