当前位置: 首页 > 编程笔记 >

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

傅宏恺
2023-03-14
本文向大家介绍Android设备获取扫码枪扫描的内容与可能遇到的问题解决,包括了Android设备获取扫码枪扫描的内容与可能遇到的问题解决的使用技巧和注意事项,需要的朋友参考一下

前言

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

为了节约成本,扫码枪可以直接通过蓝牙连接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入手。

事件html" target="_blank">解析类

/**
 * 扫码枪事件解析类
 */
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 显示扫描内容
 }

详细代码参看:https://github.com/czhzero/scangon

注意点:

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文件里描述的不一样,此问题还没有弄明白为什么,系统说两个是不相关的?
  • 发现在两个平台上有大部分按键值在一致的,但存在分别的是不样的,不明白怎么定义的。依据是啥?

总结

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

 类似资料:
  • 我是Java新手,在过去从未遇到过麻烦的事情上苦苦挣扎。不管出于什么原因,我不能扫描代码中的int(或double),但我可以扫描字符串。我正在发布扫描程序无法正常工作的代码片段,请告诉我是否应该包含程序的其余部分。 在底部附近,不接受任何扫描值。我试着把它放在一个不同于的方法中,我试着重新排列它,为它创建一个单独的扫描仪,以及其他一些事情。调试器显示,不管怎样,我都会被卡在那行代码上。我做错了什

  • 我对DynamoDB的性能有问题,我想澄清一下我有点困惑的地方。 执行扫描表中的100条记录,条件是使用(例如)。如果在表中找到20条记录,DynamoDB是否还扫描其他80条记录? 扫描时分页是如何工作的? 消耗超过分配的RCU和WCU的后果是什么?

  • 我正在开发一款Android应用程序,可以实现蓝牙功能。蓝牙扫描,一切正常(我在这里参考了android bletooth聊天代码示例:(https://android.googlesource.com/platform/development//25b6aed7b2e01ce7bdc0dfa1a79eaf009ad178fe/samples/BluetoothChat),但当蓝牙扫描运行时,我的

  • 我正在创建一个Android应用程序,该应用程序使用Android Studio扫描所有附近的BLE(蓝牙低功耗)设备。 我想我已经实施了所有这些文件: https://developer.android.com/guide/topics/connectivity/bluetooth/ble-overview https://developer.android.com/guide/topics/c

  • 开启蓝牙扫描 参数说明 字段 类型 必须? 说明 blePrefix String 是 设备名称类型前缀 示例代码 Swift: RokidMobileSDK.binder.startBLEScan(blePrefix: String) ->Bool Objc: [RokidMobileSDK.binder startBLEScanWithBlePrefix:@"Rokid-"]; 手机蓝牙未打

  • 扫描设备 介绍 需要传入扫描蓝牙设备的名称的前缀,回调均在主线程。如果传空,是无法获取设备列表。 1、单前缀蓝牙设备 参数说明 字段 类型 必须? 说明 type String 是 设备名称类型前缀 举个大栗子 String type = "Rokid-" RokidMobileSDK.binder.startBTScan(type, new IBTScanCallBack() { @Ov