Работа со сканером штрихкодов
Необходимо реализовать работу со сканером. WinFroms. У меня уже довольно давно работает следующий костыль. Учитывая что сканер работает в режиме эмулятора клавиатуры, в нужной форме, я ловлю все нажатия клавиш и складываю их в строку. А эту самую строку очищаю раз в 0.3 секунды. При каждом добавлении проверяю, не 13 ли символов в строке, именно такая длинна штрихкода. Если да, то передаю строку в обработчик штрихкода. С клавиатуры заполнить строку длинной 13 символов за 300мс - невозможно, сканер же делает это куда быстрее, так что выходит довольно точно вытягивать штрихкоды. В данном решении есть несколько проблем:
- Мы можем использовать только штрихкоды, длиной 13 символов. (стандарт EAN-13).
- В конце штрихкода, сканер еще и отправляет нажатие клавиши Enter, что приводит к случайным нажатиям активных клавиш, и подобному.
- И самое главное, сканер все еще воспринимается как клавиатура, так что если в форме активно текстовое поле, он будет писать в нем штрихкод. Таким образом, когда кассир добавил товар, отредактировал цену и сканирует следующий товар, то получает в дополнение к новому товару, цену в формате 4823432123342.
Что нужно? Верно, нужно знать от какой клавиатуры мы получили этот Char, что б понять, сканер это или нет. Ну а выбор сканера из списка клавиатур сделать где то в параметрах.
Я не первый делаю сканер штрихкодов, так что пошел искать ответы в гугл. Как я понял, самым адекватным способом реализовать подобный функционал, будет использование хуков. Из интернета был скопипащен следующий код:
using System;
using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text;
using System.Threading.Tasks;
using System.Windows.Input;
namespace BRUA
{
public class BarcodeScanerListener
{
private const int WH_KEYBOARD_LL = 13;
private const int WM_KEYDOWN = 0x0100;
private const int WM_SYSKEYDOWN = 0x0104;
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern IntPtr SetWindowsHookEx(int idHook, LowLevelKeyboardProc lpfn, IntPtr hMod, uint dwThreadId);
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
[return: MarshalAs(UnmanagedType.Bool)]
private static extern bool UnhookWindowsHookEx(IntPtr hhk);
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern IntPtr CallNextHookEx(IntPtr hhk, int nCode, IntPtr wParam, IntPtr lParam);
[DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
private static extern IntPtr GetModuleHandle(string lpModuleName);
public delegate IntPtr LowLevelKeyboardProc(int nCode, IntPtr wParam, IntPtr lParam);
public event EventHandler<KeyPressedArgs> OnKeyPressed;
private LowLevelKeyboardProc _proc;
private IntPtr _hookID = IntPtr.Zero;
public BarcodeScanerListener()
{
_proc = HookCallback;
}
public void HookKeyboard()
{
_hookID = SetHook(_proc);
}
public void UnHookKeyboard()
{
UnhookWindowsHookEx(_hookID);
}
private IntPtr SetHook(LowLevelKeyboardProc proc)
{
using (Process curProcess = Process.GetCurrentProcess())
using (ProcessModule curModule = curProcess.MainModule)
{
return SetWindowsHookEx(WH_KEYBOARD_LL, proc, GetModuleHandle(curModule.ModuleName), 0);
}
}
private IntPtr HookCallback(int nCode, IntPtr wParam, IntPtr lParam)
{
if (nCode >= 0 && wParam == (IntPtr)WM_KEYDOWN || wParam == (IntPtr)WM_SYSKEYDOWN)
{
int vkCode = Marshal.ReadInt32(lParam);
if (OnKeyPressed != null) { OnKeyPressed(this, new KeyPressedArgs(KeyInterop.KeyFromVirtualKey(vkCode))); }
}
return CallNextHookEx(_hookID, nCode, wParam, lParam);
}
}
public class KeyPressedArgs : EventArgs
{
public Key KeyPressed { get; private set; }
public KeyPressedArgs(Key key)
{
KeyPressed = key;
}
}
}
И теперь интегрируем его в приложение.
public partial class Form1 : Form
{
public BarcodeScanerListener barcodeScanerListener = new BarcodeScanerListener();
public Form1()
{
InitializeComponent();
barcodeScanerListener = new BarcodeScanerListener();
barcodeScanerListener.OnKeyPressed += BarcodeScanerListener_OnKeyPressed;
barcodeScanerListener.HookKeyboard();
}
private void BarcodeScanerListener_OnKeyPressed(object sender, KeyPressedArgs e)
{
MessageBox.Show(e.KeyPressed.ToString());
}
}
Ура, Хук работает, мы перехватываем нажатия клавиатуры. Вот только, не той клавиатуры, что нужно. Как нам сделать так, что б хук работал на определенную клавиатуру? Другие способы реализации работы сканера тоже рассмотрел бы.