USB扫码模块可以是扫描盒子或者扫码枪之类的,一根USB线作为供电和数据通信使用,有些扫码模块支持虚拟串口模式,虚拟串口模式读取数据会比较简单一点,和普通的串口一样操作即可,就是通过虚拟串口口+波特率即可获取到数据,这里主要讲读取USB模式下的数据。

1.读取USB模式下的数据

USB模式下的扫码模块相当于一个外接键盘,也就是它必须在有光标的地方才能进行扫码,且是直接把扫到的内容自动输入到输入框中,并不受我们的控制,所以我们必须另外想办法,安卓系统中有这么个方dispatchKeyEvent(KeyEvent event),它就是用来处理我们键盘的输入事件的,如果我们拦截该方法,把它交给我们自己去处理,这样我们就可以不通过Edittext从而获取到扫码头传过来的数据了。

值得注意的是扫码输出字符是连续输出的,也就是说扫码的数据并不是一下子输出所有的数据,而是有间隔的输出字符,间隔是比较短的,大概有10ms左右,有些扫码模块有回车结束符,有些没有,所以我们需要封装代码使用,如下
import android.os.Build;
import android.os.Handler;
import android.util.Log;
import android.view.KeyEvent;

  1. import org.greenrobot.eventbus.EventBus;
  2. import java.util.ArrayList;
  3. import java.util.List;
  4. public class ScanGunHelper {
  5. private final static long MESSAGE_DELAY = 200;
  6. private StringBuffer mStringBufferResult = new StringBuffer();
  7. private Handler mHandler = new Handler();
  8. private Runnable mScanningFishedRunnable = new Runnable() {
  9. @Override
  10. public void run() {
  11. performScanSuccess();
  12. }
  13. };
  14. private List<Integer> keyCodeList = new ArrayList<>();
  15. private static class SingletonHolder {
  16. private static final ScanGunHelper instance = new ScanGunHelper();
  17. }
  18. private ScanGunHelper() {
  19. }
  20. public static ScanGunHelper getInstance() {
  21. return ScanGunHelper.SingletonHolder.instance;
  22. }
  23. /**
  24. * 返回扫码成功后的结果
  25. */
  26. private void performScanSuccess() {
  27. try {
  28. boolean mCaps = keyCodeList.contains(KeyEvent.KEYCODE_SHIFT_RIGHT) || keyCodeList.contains(KeyEvent.KEYCODE_SHIFT_LEFT);
  29. for (int keyCode : keyCodeList) {
  30. char aChar = getInputCode(keyCode, mCaps);
  31. if (aChar != 0) {
  32. mStringBufferResult.append(aChar);
  33. }
  34. }
  35. String barcode = mStringBufferResult.toString();
  36. Log.e("ScanGunHelper", "barcode==" + barcode);
  37. mStringBufferResult.setLength(0);
  38. keyCodeList.clear();
  39. if (mHandler != null) {
  40. mHandler.removeCallbacks(mScanningFishedRunnable);
  41. }
  42. } catch (Exception e) {
  43. e.printStackTrace();
  44. }
  45. }
  46. /**
  47. * 扫码枪事件解析
  48. */
  49. public void analysisKeyEvent(KeyEvent event) {
  50. //Virtual是我所使用机器的内置软键盘的名字
  51. //在这判断是因为项目中避免和软键盘冲突(扫码枪和软键盘都属于按键事件)
  52. int keyCode = event.getKeyCode();
  53. if (event.getAction() == KeyEvent.ACTION_UP) {
  54. keyCodeList.add(keyCode);
  55. if (keyCode == KeyEvent.KEYCODE_ENTER) {
  56. mHandler.removeCallbacks(mScanningFishedRunnable);
  57. mHandler.post(mScanningFishedRunnable);
  58. } else {
  59. mHandler.removeCallbacks(mScanningFishedRunnable);
  60. mHandler.postDelayed(mScanningFishedRunnable, MESSAGE_DELAY);
  61. }
  62. }
  63. }
  64. //获取扫描内容
  65. private char getInputCode(int keyCode, boolean mCaps) {
  66. char aChar;
  67. if (keyCode >= KeyEvent.KEYCODE_A && keyCode <= KeyEvent.KEYCODE_Z) {
  68. //字母
  69. aChar = (char) ((mCaps ? 'A' : 'a') + keyCode - KeyEvent.KEYCODE_A);
  70. } else if (keyCode >= KeyEvent.KEYCODE_0 && keyCode <= KeyEvent.KEYCODE_9) {
  71. //数字
  72. aChar = (char) ('0' + keyCode - KeyEvent.KEYCODE_0);
  73. } else {
  74. //其他符号
  75. switch (keyCode) {
  76. case KeyEvent.KEYCODE_PERIOD:
  77. aChar = '.';
  78. break;
  79. case KeyEvent.KEYCODE_MINUS:
  80. aChar = mCaps ? '_' : '-';
  81. break;
  82. case KeyEvent.KEYCODE_SLASH:
  83. aChar = '/';
  84. break;
  85. case KeyEvent.KEYCODE_BACKSLASH:
  86. aChar = mCaps ? '|' : '\\';
  87. break;
  88. default:
  89. aChar = 0;
  90. break;
  91. }
  92. }
  93. return aChar;
  94. }
  95. /**
  96. * 输入设备是否存在
  97. */
  98. public static boolean isScanGunExist(KeyEvent event) {
  99. if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
  100. Log.e("ScanGunHelper", "ProductId==" + event.getDevice().getProductId() + ",VendorId==" + event.getDevice().getVendorId());
  101. if (event.getDevice().getVendorId() == 1409 && event.getDevice().getProductId() == 262) {
  102. return true;
  103. }
  104. }
  105. return false;
  106. }
  107. }

如果用的扫码模块有确定的ProductId和VendorId,可以直接过滤,写死ProductId和VendorId会使得程序很不灵活,万一换了一款扫码模块就不适配了,所以建议做成可配置,避免很多麻烦

如果不用ProductId和VendorId过滤输入事件,那就区分一下软键盘即可,系统自动软件盘的DeviceId为-1,只要不等于-1,就是扫码模块输入的字符。

单列模式封装代码,主要是考虑到一台安卓一体机不可能接入两个扫码模块,当成唯一设备作为处理即可。

2.在基类的Activity中使用,确保每个界面能拦截到数据,方便使用,不需要每个界面都拦截
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.view.KeyEvent;
import android.view.WindowManager;

  1. public abstract class BaseActivity extends AppCompatActivity {
  2. @Override
  3. protected void onCreate(@Nullable Bundle savedInstanceState) {
  4. super.onCreate(savedInstanceState);
  5. }
  6. @Override
  7. public boolean dispatchKeyEvent(KeyEvent event) {
  8. if (event.getDeviceId() > 0) {
  9. ScanGunHelper.getInstance().analysisKeyEvent(event);
  10. return true;
  11. }
  12. return super.dispatchKeyEvent(event);
  13. }
  14. }

总结:
现在的USB扫码模块是比较流行,但是扫码模块与终端设备的通信方式花样百出,有wifi、蓝牙、USB、串口等等,但是万变不离其宗,对于安卓设备而言,扫码模块就是个外接键盘,不管是什么方式通信也好(除了串口以外),最终呈现的形态就是个外接键盘,当然,我们也可以使用蓝牙读取扫码数据,这些都是so esay了,串口更加简单,有实际的串口线更好,虚拟串口会麻烦一点,每次插拔虚拟串口号会不固定,不同的安卓设备表现出的串口号也尽不相同,所以虚拟串口模式需要做很多适配工作,不能写死一个虚拟串口号,这样会照成程序不灵活、不适配,做成可配置,才能应付多变的招标现场或者客户需求现场。

更多相关文章

  1. 华纳公司现场经理
  2. 腾龙公司在线
  3. 百胜公司在线
  4. 如何设计营销型网站首页?营销型网站首页九大模块设置
  5. HTML注册表单控件实例、CSS模块化理解与CSS基本/上下文选择器的
  6. 简单的注册和模块+CSS选择器
  7. 表单案例和css模块化与选择器
  8. 选择器的使用和模块化组件思想
  9. php之tp6安装与扩展安装

随机推荐

  1. BULLSEYE:没有覆盖linux中的共享库
  2. Linux - Nginx配置反向代理。
  3. windows linux 创建oracle用户及表空间
  4. 压缩解压常用命令
  5. mysql数据库忘记ROOT密码时的解决办法
  6. Linux Bash Shell编程快速入门
  7. linux下lftp命令中mirror的用法详解
  8. script学习,如何用linux监控你的同事?
  9. Linux系统中使用fdisk命令为U盘分区,并将
  10. Linux内核分析-9/进程的调度时机