最近做了个关于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_DEVICESfun 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 节点下

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="" />

更多相关文章

  1. Android事件分发机制完全解析(终极版二)
  2. 【Android(安卓)- 进阶】之事件分发机制
  3. Android(安卓)EditText 属性
  4. Android事件分发机制和一些疑问
  5. android之EditText
  6. Android查询:模拟键盘鼠标事件(adb shell 实现)
  7. Android(安卓)studio项目不能编译,提示设备版本过低
  8. 用网络adb连接调试Android
  9. 解决 Android模拟器无法上网问题——Host is unresolved

随机推荐

  1. android 开源项目(城市定位)
  2. Android中下载文件的使用
  3. Android(安卓)学习之- 单选按钮、复选框
  4. Android(安卓)Applications Tutorial 22.
  5. android security and policy
  6. android canvas 画闹钟 圆弧
  7. 5 Android(安卓)Websites You Should Che
  8. Appcelerator Cloud Push Notification i
  9. as真机安装apk遇到的一个坑 INSTALL_FAIL
  10. 10.Android(安卓)ImageView ScaleType属