欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android设备获取扫码枪扫描的内容与可能遇到的问题解决

程序员文章站 2023-11-21 16:37:52
前言 大家应该都有所体会,在生活中条形码扫码枪可是随处可见,可以很迅速地扫描出条形码内容,比什么手机相机扫码快了不是一点两点。 为了节约成本,扫码枪可以直接通过蓝...

前言

大家应该都有所体会,在生活中条形码扫码枪可是随处可见,可以很迅速地扫描出条形码内容,比什么手机相机扫码快了不是一点两点。

为了节约成本,扫码枪可以直接通过蓝牙连接android或其他设备。

那么android设备如何通过蓝牙获取扫描内容的呢?

1. 蓝牙配对,连接设备

打开系统设置,找到蓝牙,打开扫码枪,配对扫码枪设备。输入一个固定的配对码,一般扫码枪说明书里都有写。配对完成后,显示设备已连接。就ok。

2.androidmanifest中配置权限

android项目中的androidmanifest.xml文件添加蓝牙权限。

 <uses-permission android:name="android.permission.bluetooth" />
 <uses-permission android:name="android.permission.bluetooth_admin" />

3.检测扫码枪的连接状态

通常来说,扫码枪设备也相当于普通外接输入设备类型,外接键盘。

我这款扫码枪设备返回的是如下蓝牙类型。

bluetoothclass.device.major.peripheral

一般而言,通过如下这种方式就可以获得到我们扫码枪设备的信息。

set<bluetoothdevice> bluedevices = mbluetoothadapter.getbondeddevices();

if (bluedevices == null || bluedevices.size() <= 0) {
 return false;
}

for (iterator<bluetoothdevice> iterator = bluedevices.iterator(); iterator.hasnext(); ) {
 bluetoothdevice bluetoothdevice = iterator.next();

 if (bluetoothdevice.getbluetoothclass().getmajordeviceclass() == bluetoothclass.device.major.peripheral) {
 //todo 获取扫码枪设备信息
 }

}

开发过程中,必然会需要实时判断设备是否正常连接。

mbluetoothadapter.getbondeddevices()

这个方法仅仅只能够判断设备是否已配对绑定。但是绑定不代表连接,所以只能放弃。

public list getconnecteddevices (int profile) 
public int getconnectionstate (bluetoothdevice device, int profile)

接着又尝试了这两个方法,方法是可用,但是必须要求设备sdk>18,即android 4.3版本以上才可用。

后来转头一想,既然扫码枪也是输入设备,我们可以不同蓝牙设备状态检测入手,改为从输入设备检测入手。于是,

private void hasscangun() {
 configuration cfg = getresources().getconfiguration();
 return cfg.keyboard != configuration.keyboard_nokeys;
}

搞定。

4.获取扫码枪扫描内容

扫描枪,既然是一个外接输入设备,那么很自然的,我们就从keyevent入手。

事件解析类

/**
 * 扫码枪事件解析类
 */
public class scangunkeyeventhelper {

 //延迟500ms,判断扫码是否完成。
 private final static long message_delay = 500;
 //扫码内容
 private stringbuffer mstringbufferresult = new stringbuffer();
 //大小写区分
 private boolean mcaps;
 private onscansuccesslistener monscansuccesslistener;
 private handler mhandler = new handler();

 private final runnable mscanningfishedrunnable = new runnable() {
 @override
 public void run() {
  performscansuccess();
 }
 };

 //返回扫描结果
 private void performscansuccess() {
 string barcode = mstringbufferresult.tostring();
 if (monscansuccesslistener != null)
  monscansuccesslistener.onscansuccess(barcode);
 mstringbufferresult.setlength(0);
 }

 //key事件处理
 public void analysiskeyevent(keyevent event) {

 int keycode = event.getkeycode();

 //字母大小写判断
 checkletterstatus(event);

 if (event.getaction() == keyevent.action_down) {

  char achar = getinputcode(event);;

  if (achar != 0) {
  mstringbufferresult.append(achar);
  }

  if (keycode == keyevent.keycode_enter) {
  //若为回车键,直接返回
  mhandler.removecallbacks(mscanningfishedrunnable);
  mhandler.post(mscanningfishedrunnable);
  } else {
  //延迟post,若500ms内,有其他事件
  mhandler.removecallbacks(mscanningfishedrunnable);
  mhandler.postdelayed(mscanningfishedrunnable, message_delay);
  }

 }
 }

 //检查shift键
 private void checkletterstatus(keyevent event) {
 int keycode = event.getkeycode();
 if (keycode == keyevent.keycode_shift_right || keycode == keyevent.keycode_shift_left) {
  if (event.getaction() == keyevent.action_down) {
  //按着shift键,表示大写
  mcaps = true;
  } else {
  //松开shift键,表示小写
  mcaps = false;
  }
 }
 }


 //获取扫描内容
 private char getinputcode(keyevent event) {

 int keycode = event.getkeycode();

 char achar;

 if (keycode >= keyevent.keycode_a && keycode <= keyevent.keycode_z) {
  //字母
  achar = (char) ((mcaps ? 'a' : 'a') + keycode - keyevent.keycode_a);
 } else if (keycode >= keyevent.keycode_0 && keycode <= keyevent.keycode_9) {
  //数字
  achar = (char) ('0' + keycode - keyevent.keycode_0);
 } else {
  //其他符号
  switch (keycode) {
  case keyevent.keycode_period:
   achar = '.';
   break;
  case keyevent.keycode_minus:
   achar = mcaps ? '_' : '-';
   break;
  case keyevent.keycode_slash:
   achar = '/';
   break;
  case keyevent.keycode_backslash:
   achar = mcaps ? '|' : '\\';
   break;
  default:
   achar = 0;
   break;
  }
 }

 return achar;

 }




 public interface onscansuccesslistener {
 public void onscansuccess(string barcode);
 }

 public void setonbarcodecatchlistener(onscansuccesslistener onscansuccesslistener) {
 monscansuccesslistener = onscansuccesslistener;
 }

 public void ondestroy() {
 mhandler.removecallbacks(mscanningfishedrunnable);
 monscansuccesslistener = null;
 }

}

activity中重写dispatchkeyevent方法,截取key事件。

 /**
 * activity截获按键事件.发给scangunkeyeventhelper
 *
 * @param event
 * @return
 */
 @override
 public boolean dispatchkeyevent(keyevent event) {

 if (isscangunevent(event)) {
  mscangunkeyeventhelper.analysiskeyevent(event);
  return true;
 }
 return super.dispatchkeyevent(event);
 }

 /**
 * 显示扫描内容
 * @param barcode 条形码
 */
 @override
 public void onscansuccess(string barcode) {
 //todo 显示扫描内容
 }

详细代码参看:

注意点:

1.部分机型无法判断外接键盘信息,如三星。

private void hasscangun() {
 configuration cfg = getresources().getconfiguration();
 return cfg.keyboard != configuration.keyboard_nokeys;
}

三星手机cfg.keyboard返回值等于configuration.keyboard_nokeys。

因此为了更好的兼容,可以采用如下方法,

/**
 * 判断是否已经连接扫码枪
 * @return
 */
protected boolean hasscangun() {

 set<bluetoothdevice> bluedevices = mbluetoothadapter.getbondeddevices();

 if (bluedevices == null || bluedevices.size() <= 0) {
 return false;
 }

 for (iterator<bluetoothdevice> iterator = bluedevices.iterator(); iterator.hasnext(); ) {
 bluetoothdevice bluetoothdevice = iterator.next();

 if (bluetoothdevice.getbluetoothclass().getmajordeviceclass() == bluetoothclass.device.major.peripheral) {
  return isinputdeviceused(bluetoothdevice.getname());
 }

 }

 return false;

}



private boolean isinputdeviceused(string devicename) {

 int[] deviceids = inputdevice.getdeviceids();

 for (int id : deviceids) {
 if (inputdevice.getdevice(id).getname().equals(devicename)) {
  return true;
 }
 }

 return false;

}

anroid系统解决扫码枪无法输入字母和字符问题

问题:

在使用扫码枪扫码条码的时候明明有字母和字符,但是输入到android系统却没哟,输入到电脑是正常的,这就很奇怪,让一个搞上层开发的摸不着头脑,最后和系统讨论才知道是系统按键部分映射被删除导致的。

解决办法:

在android系统层frameworks/base/data/keyboards文件夹下面有generic.kl这个文件,此文件为android默认的按键映射对应表,还有其他的比如:qwerty.kl文件,以及一些自定义码值的kl文件。
打开generic.kl看看类型也许就明白了.

key 11 0
key 2 1
key 3 2
key 4 3
key 5 4
key 6 5
key 7 6
key 8 7
key 9 8
key 10 9
key 12 minus
key 13 equals
key 14 del
key 15 tab

里面是键与键值的映射,比如:键值11 对应的按键为 0 这个,以此类推。那解决就明朗了,将所有字母和字符的按键映射添加进行就ok了,至于按键值是多少我这边直接参考了另外一个平台的generic.kl文件。重新编译系统验证,此问题解决了。

疑惑问题:

  • 用相同android版本的android.jar 查看keycode对应的值和generic.kl文件里描述的不一样,此问题还没有弄明白为什么,系统说两个是不相关的?
  • 发现在两个平台上有大部分按键值在一致的,但存在分别的是不样的,不明白怎么定义的。依据是啥?

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。