Android 获取USB扫描枪简易封装

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

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

你可能感兴趣的:(Java,Android,Kotlin)