最近做了个关于Android设备Usb外接扫码器的项目,在此记录下。扫码器有以下这两种模式:
- USB HID-KBW:扫码器会将扫描出来的内容转化为键盘事件,就是Android中就是KeyEvent里面对应的常量(0 = KeyEvent.KEYCODE_0)。
- USB 虚拟串口:可使用android-serialport-api 连接到UsbDevice进行通信,读取数据。(设备要支持串口)
支持 Android 热插拔USB扫描枪会在有EditText时,扫描枪扫描内容自动输入到编辑框了,在没有EditText的情况下呢?还会响应获焦控件的点击事件(如Button),因为标准扫描枪扫描数据会触发KEYCODE_ENTER键。
通过USB 虚拟串口方式,这个我喜欢,可是它不支持! 项目需求:
- 扫码枪扫商品条形码时返回内容(通常一串数字),作为购买时唯一标识
扫码枪是基于键盘输入的,那事件会先分发到获取焦点的Activity、Dialog 中的,dispatchKeyEvent(KeyEvent event)
.所以很好解决了由安卓事件分发机制看,只要消费了扫码器产生的事件,就不需要EditText,也不会触发到其它组件了。那好,现在新的问题又来了,dispatchKeyEvent(KeyEvent event)
是按键事件分发的第一个要塞,而且没办法统一为应用设置监听,只能在每个Activity、Dialog作监听。这里可以基类(BaseActivity)处理扫码器的输入事件,也可以通过AccessibilityService 的 onKeyEvent(KeyEvent event)
事件去处理,但无障碍辅助需要手动开启,不太友好。
查看 KeyEvent 源码一看继承 InputEvent,正好可以通过 InputDevice getDevice()
获取输入设备,根据输入设备正好判断该事件输入扫码枪输入
以下是BarCodeHelper.kt 处理扫码器输入事件且回调条形码number
//扫码设备名称
const val BARCODE_DEVICES = "Barcode Reader"
var scannerResult = StringBuilder()
fun hasBarcodeInputDeviceExist(): Boolean {
InputDevice.getDeviceIds().forEach {
val name = InputDevice.getDevice(it).name.trim()
Log.i("InputDevice", name)
if (BARCODE_DEVICES == name) {
return true
}
}
return false
}
fun KeyEvent.isBarcodeKeyEvent() = this.device.name.trim() == BARCODE_DEVICES
fun Activity.transformBarCodeKeyEvent(event: KeyEvent, listener: (result: String) -> Unit): Boolean {
if (event.action == KeyEvent.ACTION_DOWN && event.repeatCount == 0) {
val keyCode = event.keyCode
if (keyCode >= KeyEvent.KEYCODE_0 && keyCode <= KeyEvent.KEYCODE_9) {
scannerResult.append(keyCode - KeyEvent.KEYCODE_0)
return true
}
if (keyCode == KeyEvent.KEYCODE_ENTER) {
listener.invoke(scannerResult.toString())
scannerResult = StringBuilder()
return true
}
}
return false
}
fun Dialog.transformBarCodeKeyEvent(event: KeyEvent, listener: (result: String) -> Unit): Boolean {
if (event.action == KeyEvent.ACTION_DOWN && event.repeatCount == 0) {
val keyCode = event.keyCode
if (keyCode >= KeyEvent.KEYCODE_0 && keyCode <= KeyEvent.KEYCODE_9) {
scannerResult.append(keyCode - KeyEvent.KEYCODE_0)
return true
}
if (keyCode == KeyEvent.KEYCODE_ENTER) {
listener.invoke(scannerResult.toString())
scannerResult = StringBuilder()
return true
}
}
return false
}
BaseActivity
override fun dispatchKeyEvent(event: KeyEvent): Boolean {
//多数activity 不需要扫码输入,只要是扫码设备事件都消费掉,以防止触碰到控件
return if (event.isBarcodeKeyEvent()) {
this.transformBarCodeKeyEvent(event) { barCode ->
Log.i("Barcode", "barCode: " + barCode)
//EventBus
EventBus.getDefault().post(barCode, PRODUCT_BAR_CODE_EVENT)
}
} else super.dispatchKeyEvent(event)
}
另外你要是采取AccessibilityService 方式的话,又通过以下方式去设置的话,onKeyEvent(KeyEvent event)不回调,只能通过xml 方式注册
@Override
protected void onServiceConnected() {
super.onServiceConnected();
Log.v(TAG, "on Service Connected");
AccessibilityServiceInfo info = new AccessibilityServiceInfo();
info.packageNames = null;
info.eventTypes = AccessibilityEvent.TYPES_ALL_MASK;
info.notificationTimeout = 0;
info.feedbackType = AccessibilityEvent.TYPES_ALL_MASK;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) {
info.flags = AccessibilityServiceInfo.FLAG_REQUEST_FILTER_KEY_EVENTS;
}
setServiceInfo(info);
System.out.println(getServiceInfo());
}
这里meta-data 只能写在 service 节点下
<meta-data
android:name="android.accessibilityservice"
android:resource="@xml/serviceconfig" />
<accessibility-service xmlns:android="http://schemas.android.com/apk/res/android"
android:accessibilityEventTypes="typeAllMask"
android:accessibilityFeedbackType="feedbackAllMask"
android:accessibilityFlags="flagRequestFilterKeyEvents"
android:canRequestFilterKeyEvents="true"
android:canRetrieveWindowContent="true"
android:description="@string/app_name"
android:notificationTimeout="100"
android:packageNames="" />
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。感谢