公司的设备以前接入的都是串口的扫码头,优点是直接通过串口读取流里面的数据就OK了,缺点是你需要知道每一款扫码器的型号以获取波特率及Android设备的串口地址。因为现在usb扫码器越来越方便且即插即用,不需要额外供电以及价格便宜等特点,公司以后开发的设备都打算采用usb扫码器。所以我开始尝试接入usb扫码器,下面就是我在接入时的方法以及遇到的一些问题。

1. USB扫码器接入

  • 前面我有说过,usb扫码器接入方便,即插即用,但是有个很大的坑,因为它的实质其实就是相当于设备的外接键盘,也就是它必须在有光标的地方才能进行扫码,且是直接把扫到的内容自动输入到输入框中,并不受我们的控制。但是我们在很多时候并不需要一个edittext的输入框的,而这时要么就是重新改设计,要么就是我们自己想办法解决这个问题。
  • 最开始的时候我是利用1个宽高都为1px的Edittext作为扫码头的接收器,并且让它自动获取焦点,这样我们就能实时地获取到扫码头传过来的数据了。但是这种方法并不能作为一个通用的方法,且每个项目都不能复用,还会和我们页面中其它Edittext输入框冲突,这个方法也就被我弃用了。
  • 接下来说到的就是我现在用到的方法,我们知道,USB扫码器实质就是一个外接键盘,那么我们可不可以对它进行键盘的输入拦截呢,安卓系统中有这么个方法dispatchKeyEvent(KeyEvent event),它就是用来处理我们键盘的输入事件的,如果我们拦截该方法,把它交给我们自己去处理,这样我们就可以不通过Edittext从而获取到扫码头传过来的数据了。
  • 我自定义了一个叫ScanKeyManager的拦截键盘事件并将它转化成我们需要的数据的管理类代码如下:
import android.view.KeyEvent;

public class ScanKeyManager {

    private StringBuilder mResult;
public OnScanValueListener mListener;
private boolean mCaps; public interface OnScanValueListener {
void onScanValue(String value);
} public ScanKeyManager(OnScanValueListener listener) {
this.mListener = listener;
this.mResult = new StringBuilder();
} /**
* 扫码设备事件解析
*/
public void analysisKeyEvent(KeyEvent event) {
int keyCode = event.getKeyCode();
checkLetterStatus(event);
if (event.getAction() == KeyEvent.ACTION_DOWN) {
char aChar = getInputCode(mCaps, event.getKeyCode());
if (aChar != 0) {
mResult.append(aChar);
}
if (keyCode == KeyEvent.KEYCODE_ENTER) {
if (mListener != null) {
mListener.onScanValue(mResult.toString());
}
mResult.delete(0, mResult.length());
}
}
} /**
* 判断大小写
*/
private void checkLetterStatus(KeyEvent event) {
int keyCode = event.getKeyCode();
if (keyCode == KeyEvent.KEYCODE_SHIFT_RIGHT || keyCode == KeyEvent.KEYCODE_SHIFT_LEFT) {
mCaps = event.getAction() == KeyEvent.ACTION_DOWN;
}
} /**
* 将keyCode转为char
*
* @param caps 是不是大写
* @param keyCode 按键
* @return 按键对应的char
*/
private char getInputCode(boolean caps, int keyCode) {
if (keyCode >= KeyEvent.KEYCODE_A && keyCode <= KeyEvent.KEYCODE_Z) {
return (char) ((caps ? 'A' : 'a') + keyCode - KeyEvent.KEYCODE_A);
} else {
return keyValue(caps, keyCode);
}
} /**
* 按键对应的char表
*/
private char keyValue(boolean caps, int keyCode) {
switch (keyCode) {
case KeyEvent.KEYCODE_0:
return caps ? ')' : '0';
case KeyEvent.KEYCODE_1:
return caps ? '!' : '1';
case KeyEvent.KEYCODE_2:
return caps ? '@' : '2';
case KeyEvent.KEYCODE_3:
return caps ? '#' : '3';
case KeyEvent.KEYCODE_4:
return caps ? '$' : '4';
case KeyEvent.KEYCODE_5:
return caps ? '%' : '5';
case KeyEvent.KEYCODE_6:
return caps ? '^' : '6';
case KeyEvent.KEYCODE_7:
return caps ? '&' : '7';
case KeyEvent.KEYCODE_8:
return caps ? '*' : '8';
case KeyEvent.KEYCODE_9:
return caps ? '(' : '9';
case KeyEvent.KEYCODE_NUMPAD_SUBTRACT:
return '-';
case KeyEvent.KEYCODE_MINUS:
return '_';
case KeyEvent.KEYCODE_EQUALS:
return '=';
case KeyEvent.KEYCODE_NUMPAD_ADD:
return '+';
case KeyEvent.KEYCODE_GRAVE:
return caps ? '~' : '`';
case KeyEvent.KEYCODE_BACKSLASH:
return caps ? '|' : '\\';
case KeyEvent.KEYCODE_LEFT_BRACKET:
return caps ? '{' : '[';
case KeyEvent.KEYCODE_RIGHT_BRACKET:
return caps ? '}' : ']';
case KeyEvent.KEYCODE_SEMICOLON:
return caps ? ':' : ';';
case KeyEvent.KEYCODE_APOSTROPHE:
return caps ? '"' : '\'';
case KeyEvent.KEYCODE_COMMA:
return caps ? '<' : ',';
case KeyEvent.KEYCODE_PERIOD:
return caps ? '>' : '.';
case KeyEvent.KEYCODE_SLASH:
return caps ? '?' : '/';
default:
return 0;
}
}
}

在该类中,我拦截处理了键盘输入事件的绝大部分字符,并把它转化成我们需要的数据通过接口回调传给我们需要用到的页面。

  • 用法如下
  1. 在activity中使用
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.KeyEvent; public class MainActivity extends AppCompatActivity {
private ScanKeyManager scanKeyManager;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
//拦截扫码器回调,获取扫码内容
scanKeyManager = new ScanKeyManager(new ScanKeyManager.OnScanValueListener() {
@Override
public void onScanValue(String value) {
Log.e("ScanValue", value);
}
});
} /*监听键盘事件,除了返回事件都将它拦截,使用我们自定义的拦截器处理该事件*/
@Override
public boolean dispatchKeyEvent(KeyEvent event) {
if (event.getKeyCode() != KeyEvent.KEYCODE_BACK) {
scanKeyManager.analysisKeyEvent(event);
return true;
}
return super.dispatchKeyEvent(event);
}
}
  1. 在dialog中使用
public class ScanDialog extends Dialog {

    private ScanKeyManager scanKeyManager;
private OnScanDialogListener mListener; public ScanDialog(Context context) {
super(context, R.style.LoadDialogStyle);
} @Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
this.setContentView(R.layout.dialog_scan);
setCanceledOnTouchOutside(false);
scanKeyManager = new ScanKeyManager(new ScanKeyManager.OnScanValueListener() {
@Override
public void onScanValue(String value) {
Log.e("dialog", value);
if (mListener == null) return;
mListener.onScanResult(value);
}
});
} @Override
public boolean dispatchKeyEvent(@NonNull KeyEvent event) {
scanKeyManager.analysisKeyEvent(event);
return true;
} public interface OnScanDialogListener {
void onScanResult(String scanValue);
} public void setOnScanDialogListener(OnScanDialogListener listener) {
mListener = listener;
}
}

使用起来都是比较简单,只需要创建键盘拦截管理类,并设置监听回调,以及复写dispatchKeyEvent(KeyEvent event)拦截键盘事件就ok了,如果是dialog中使用则多了一个接口回调的步骤。

2. 遇到的问题

如果按照上述的方法去做是可以做到完美监听扫码器扫码事件的,但是我在实际使用中还遇到了一个比较严重问题,就是如果页面中有Edittext输入框的时候,如果我使用该方法拦截键盘事件之后,会出现数字,一些符号,以及删除键等一些字符输入无效的问题,好像这样我们在使用键盘拦截的时候就没法使用键盘输入了,那该怎么办呢?

3. 解决办法

上面我们说到,使用键盘拦截的时候就没法使用键盘输入了,那么有没有一种可能,就是我在键盘输入的时候不拦截键盘事件,不输入的时候我才进行拦截呢?这个时候就需要监听软键盘的显示和隐藏事件了,但是安卓系统所提供的api当中是没有监听键盘事件的,那么如果需要监听键盘事件就需要我们自定义了,我的另一篇文章中就对键盘的显示隐藏事件监听有比较好的解决办法,有兴趣的同学可以去看看Android软键盘显示隐藏事件监听。下面就开始上代码:

import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.KeyEvent; public class SecondActivity extends AppCompatActivity { private ScanKeyManager scanKeyManager;
/*是否是输入状态(输入时扫码监听不拦截)*/
private boolean isInput = false; @Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_second);
onKeyBoardListener();
//拦截扫码器回调,获取扫码内容
scanKeyManager = new ScanKeyManager(new ScanKeyManager.OnScanValueListener() {
@Override
public void onScanValue(String value) {
Log.e("ScanValue", value);
}
});
} /*监听键盘事件,除了返回事件都将它拦截,使用我们自定义的拦截器处理该事件*/
@Override
public boolean dispatchKeyEvent(KeyEvent event) {
if (event.getKeyCode() != KeyEvent.KEYCODE_BACK && !isInput) {
scanKeyManager.analysisKeyEvent(event);
return true;
}
return super.dispatchKeyEvent(event);
} //监听软件盘是否弹起
private void onKeyBoardListener() {
SoftKeyBoardListener.setListener(this, new SoftKeyBoardListener.OnSoftKeyBoardChangeListener() {
@Override
public void keyBoardShow(int height) {
Log.e("软键盘", "键盘显示 高度" + height);
isInput = true;
} @Override
public void keyBoardHide(int height) {
Log.e("软键盘", "键盘隐藏 高度" + height);
isInput = false;
}
});
}
}

那么,这个时候就可以完美解决usb扫码器的监听问题啦,可能还会有小伙伴会问,那我在键盘弹起的时候岂不是就不能用扫码头了?这个其实不用担心,当我们不拦截键盘事件的时候,且Edittext获取焦点时,那么扫到的内容就会被自动输入到输入框中了。

由于安卓系统并没有提供软键盘弹入和弹出的系统方法,所以我们有时候需要用到监听软件盘的时候会非常的麻烦,今天我根据监听视图树中全局布局发生改变或者视图树中的某个视图的可视状态发生改变超过一定的数值来监听软键盘的弹入弹出事件,使用起来也非常简单.

首先,我们需要自定义一个监听工具类代码如下:

public class SoftKeyBoardListener {
private View rootView;//activity的根视图
int rootViewVisibleHeight;//纪录根视图的显示高度
private OnSoftKeyBoardChangeListener onSoftKeyBoardChangeListener; public SoftKeyBoardListener(Activity activity) {
//获取activity的根视图
rootView = activity.getWindow().getDecorView();
//监听视图树中全局布局发生改变或者视图树中的某个视图的可视状态发生改变
rootView.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
//获取当前根视图在屏幕上显示的大小
Rect r = new Rect();
rootView.getWindowVisibleDisplayFrame(r);
int visibleHeight = r.height();
System.out.println(""+visibleHeight);
if (rootViewVisibleHeight == 0) {
rootViewVisibleHeight = visibleHeight;
return;
} //根视图显示高度没有变化,可以看作软键盘显示/隐藏状态没有改变
if (rootViewVisibleHeight == visibleHeight) {
return;
} //根视图显示高度变小超过300,可以看作软键盘显示了,该数值可根据需要自行调整
if (rootViewVisibleHeight - visibleHeight > 200) {
if (onSoftKeyBoardChangeListener != null) {
onSoftKeyBoardChangeListener.keyBoardShow(rootViewVisibleHeight - visibleHeight);
}
rootViewVisibleHeight = visibleHeight;
return;
} //根视图显示高度变大超过300,可以看作软键盘隐藏了,该数值可根据需要自行调整
if (visibleHeight - rootViewVisibleHeight > 200) {
if (onSoftKeyBoardChangeListener != null) {
onSoftKeyBoardChangeListener.keyBoardHide(visibleHeight - rootViewVisibleHeight);
}
rootViewVisibleHeight = visibleHeight;
return;
} }
});
} private void setOnSoftKeyBoardChangeListener(OnSoftKeyBoardChangeListener onSoftKeyBoardChangeListener) {
this.onSoftKeyBoardChangeListener = onSoftKeyBoardChangeListener;
}
public interface OnSoftKeyBoardChangeListener {
void keyBoardShow(int height);
void keyBoardHide(int height);
} public static void setListener(Activity activity, OnSoftKeyBoardChangeListener onSoftKeyBoardChangeListener) {
SoftKeyBoardListener softKeyBoardListener = new SoftKeyBoardListener(activity);
softKeyBoardListener.setOnSoftKeyBoardChangeListener(onSoftKeyBoardChangeListener);
}
}

其次,我们在需要使用监听的activity清单文件中设置configChanges:

<activity android:name=".MainActivity"
android:launchMode="singleTop"
android:configChanges="keyboardHidden"
android:windowSoftInputMode="adjustPan|stateHidden"/>

最后,使用方法也非常的简单,只需要在activity类中调用方法就ok了,代码如下:

    //监听软件盘是否弹起
private void onKeyBoardListener() {
SoftKeyBoardListener.setListener(MainActivity.this, new SoftKeyBoardListener.OnSoftKeyBoardChangeListener() {
@Override
public void keyBoardShow(int height) {
Log.e("软键盘", "键盘显示 高度" + height);
} @Override
public void keyBoardHide(int height) {
Log.e("软键盘", "键盘隐藏 高度" + height);
}
});
}

如果该篇文章对你有用,希望你能够点个喜欢支持一下,如果其中有错误希望多多指正。

Android外接USB扫码枪的更多相关文章

  1. [RK3288] 外接USB设备出现丢数

    CPU:RK3288 系统:Android 5.1 主板外接 USB 接口的外设,经常会出现丢数的现象,这种问题在很多 USB 接口的外设上都遇到过,例如:USB读卡器.USB扫描枪等 有一个共同点是 ...

  2. android 与usb 设备通信(二)

    再次遇到android  mUsbManager.getDevicelist() 得不到usb 设备的问题.于是深入去探讨android 与usb 外围设备通信的问题.第一篇文章写的有点乱,本质就是需 ...

  3. android的USB MTP && USB CDC/USBnet(ECM, NCM, ACM) && USB gardget

    MTP的全称是Media Transfer Protocol(媒体传输协议),它是微软公司提出的一套媒体文件传输协议.早在智能手机普及前,数码相机和MP3播放器等都使用了MTP的前身PTP(Pictu ...

  4. 【Android】Android之USB

    [转载请注明出处] 首先介绍一个概念:USB Host and Accessory Android通过两种模式支持一系列的USB外围设备和Android USB附件(实现了Android附件协议的硬件 ...

  5. 用代码实现断开Android手机USB连接【转】

    本文转载自:https://blog.csdn.net/phoebe_2012/article/details/47025309 用代码实现断开Android手机USB连接               ...

  6. Android studio USB连接失败

    Android studio USB连接失败,可能是因为adb的端口被占了,此时在其自带的cmd中输入netstat -aon|findstr "5037",并且启动任务管理器关掉 ...

  7. android 连接蓝牙扫码枪,程序崩溃之onConfigurationChanged

    当android手机通过蓝牙连接扫码枪时,程序崩溃的原因之一是:键盘弹出或隐藏,触发程序走了onDestory->onCreate的生命周期,从而可能使得页面的某些初始化数据被清除了. 解决方法 ...

  8. android 更改USB显示名称

    能力 kernel\drivers\usb\gadget\Android.c 在这个例子中,下列的变化 #define PRODUCT_STRING "Sergeycao" 版权声 ...

  9. usbmanger android 底下USB的工作模式

    Android USB开发麻烦还是比较多的. 第一种:host模式 这种模式比较不错,由Android设备提供电源,然后与外部设备通信.举个例子来说:电脑连接USB设备,都是这个模式,非常常见的模式. ...

随机推荐

  1. Java JDBC 数据源

    数据源有2种: 普通数据源     即数据库驱动自带的数据源 连接池     包括数据库驱动自带的连接池,以及DBCP.C3P0等常用的第三方连接池. 数据库驱动自带的数据源 //从propertie ...

  2. can解析

  3. 打造属于你的提供者(Provider = Strategy + Factory Method) 设计模式 - Provider Pattern(提供者模式)

    打造属于你的提供者(Provider = Strategy + Factory Method)   1.1.1 摘要 在日常系统设计中,我们也许听说过提供者模式,甚至几乎每天都在使用它,在.NET F ...

  4. 关于mysql数据库涉及的一些规范

    tips:如果本文对你有用,请爱心点个赞,提高排名,让这篇文章帮助更多的人.谢谢大家!比心❤~ 如果解决不了,可以在文末加我微信,进群交流. 设计规范,在分工协作的工作场景中尤其重要,否则团队之间互相 ...

  5. 【Linux】在linux上java工具jps jstat jinfo等命令找不到怎么办

    一.yum安装方式 1)搜索openjdk-devel相关的安装包 yum search java|grep jdk 2)安装对应的版本 yum install -y java-1.8.0-openj ...

  6. mysql设计规范和原则

    DB设计流程: 1,需求分析 2,ER设计 3,物理设计 需求分析的最佳实践是头脑风暴,把需求理解透彻.根据公司的现况和未来的发展,与pm一起来讨论. ER(EntiyRelation)设计阶段要确定 ...

  7. 使用Matplotlab画图

    1.绘制折线图 #! /usr/bin/env python#encoding=utf-8 # 用于python2import sys reload(sys) sys.setdefaultencodi ...

  8. python-Arduino串口传输数据到电脑并保存至excel表格

    起因:学校运河杯报了个项目,制作一个天气预测的装置.我用arduino跑了BME280模块,用蓝牙模块实现两块arduino主从机透传.但是为了分析,还需要提取出数据.因此我用python写了个上位机 ...

  9. 攻防世界WEB高手进阶之blgdel

    CISCN final 打开页面 扫描目录 Robots.txt Config.txt 代码审计 <?php class master { private $path; private $nam ...

  10. Statefulset:部署有状态的多副本应用

    10.1.什么是Statefulset StatefulSet是Kubernetes提供的管理有状态应用的负载管理控制器API.        特点: 1.具有固定的网络标记(主机名) 2.具有持久化 ...