溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Android設(shè)備如何獲取掃碼槍掃描的內(nèi)容

發(fā)布時(shí)間:2021-05-25 09:41:19 來源:億速云 閱讀:496 作者:小新 欄目:移動(dòng)開發(fā)

這篇文章將為大家詳細(xì)講解有關(guān)Android設(shè)備如何獲取掃碼槍掃描的內(nèi)容,小編覺得挺實(shí)用的,因此分享給大家做個(gè)參考,希望大家閱讀完這篇文章后可以有所收獲。

那么android設(shè)備如何通過藍(lán)牙獲取掃描內(nèi)容的呢?

1. 藍(lán)牙配對(duì),連接設(shè)備

打開系統(tǒng)設(shè)置,找到藍(lán)牙,打開掃碼槍,配對(duì)掃碼槍設(shè)備。輸入一個(gè)固定的配對(duì)碼,一般掃碼槍說明書里都有寫。配對(duì)完成后,顯示設(shè)備已連接。就ok。

2.AndroidManifest中配置權(quán)限

android項(xiàng)目中的AndroidManifest.xml文件添加藍(lán)牙權(quán)限。

 <uses-permission android:name="android.permission.BLUETOOTH" />
 <uses-permission android:name="android.permission.BLUETOOTH_ADMIN" />

3.檢測掃碼槍的連接狀態(tài)

通常來說,掃碼槍設(shè)備也相當(dāng)于普通外接輸入設(shè)備類型,外接鍵盤。

我這款掃碼槍設(shè)備返回的是如下藍(lán)牙類型。

BluetoothClass.Device.Major.PERIPHERAL

一般而言,通過如下這種方式就可以獲得到我們掃碼槍設(shè)備的信息。

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 獲取掃碼槍設(shè)備信息
 }

}

開發(fā)過程中,必然會(huì)需要實(shí)時(shí)判斷設(shè)備是否正常連接。

mBluetoothAdapter.getBondedDevices()

這個(gè)方法僅僅只能夠判斷設(shè)備是否已配對(duì)綁定。但是綁定不代表連接,所以只能放棄。

public List getConnectedDevices (int profile) 
public int getConnectionState (BluetoothDevice device, int profile)

接著又嘗試了這兩個(gè)方法,方法是可用,但是必須要求設(shè)備sdk>18,即android 4.3版本以上才可用。

后來轉(zhuǎn)頭一想,既然掃碼槍也是輸入設(shè)備,我們可以不同藍(lán)牙設(shè)備狀態(tài)檢測入手,改為從輸入設(shè)備檢測入手。于是,

private void hasScanGun() {
 Configuration cfg = getResources().getConfiguration();
 return cfg.keyboard != Configuration.KEYBOARD_NOKEYS;
}

搞定。

4.獲取掃碼槍掃描內(nèi)容

掃描槍,既然是一個(gè)外接輸入設(shè)備,那么很自然的,我們就從KeyEvent入手。

事件解析類

/**
 * 掃碼槍事件解析類
 */
public class ScanGunKeyEventHelper {

 //延遲500ms,判斷掃碼是否完成。
 private final static long MESSAGE_DELAY = 500;
 //掃碼內(nèi)容
 private StringBuffer mStringBufferResult = new StringBuffer();
 //大小寫區(qū)分
 private boolean mCaps;
 private OnScanSuccessListener mOnScanSuccessListener;
 private Handler mHandler = new Handler();

 private final Runnable mScanningFishedRunnable = new Runnable() {
 @Override
 public void run() {
  performScanSuccess();
 }
 };

 //返回掃描結(jié)果
 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內(nèi),有其他事件
  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;
  }
 }
 }


 //獲取掃描內(nèi)容
 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) {
  //數(shù)字
  aChar = (char) ('0' + keyCode - KeyEvent.KEYCODE_0);
 } else {
  //其他符號(hào)
  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截獲按鍵事件.發(fā)給ScanGunKeyEventHelper
 *
 * @param event
 * @return
 */
 @Override
 public boolean dispatchKeyEvent(KeyEvent event) {

 if (isScanGunEvent(event)) {
  mScanGunKeyEventHelper.analysisKeyEvent(event);
  return true;
 }
 return super.dispatchKeyEvent(event);
 }

 /**
 * 顯示掃描內(nèi)容
 * @param barcode 條形碼
 */
 @Override
 public void onScanSuccess(String barcode) {
 //TODO 顯示掃描內(nèi)容
 }

詳細(xì)代碼參看:https://github.com/czhzero/scangon

注意點(diǎn):

1.部分機(jī)型無法判斷外接鍵盤信息,如三星。

private void hasScanGun() {
 Configuration cfg = getResources().getConfiguration();
 return cfg.keyboard != Configuration.KEYBOARD_NOKEYS;
}

三星手機(jī)cfg.keyboard返回值等于Configuration.KEYBOARD_NOKEYS。

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

/**
 * 判斷是否已經(jīng)連接掃碼槍
 * @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系統(tǒng)解決掃碼槍無法輸入字母和字符問題

問題:

在使用掃碼槍掃碼條碼的時(shí)候明明有字母和字符,但是輸入到Android系統(tǒng)卻沒喲,輸入到電腦是正常的,這就很奇怪,讓一個(gè)搞上層開發(fā)的摸不著頭腦,最后和系統(tǒng)討論才知道是系統(tǒng)按鍵部分映射被刪除導(dǎo)致的。

解決辦法:

在Android系統(tǒng)層frameworks/base/data/keyboards文件夾下面有Generic.kl這個(gè)文件,此文件為Android默認(rèn)的按鍵映射對(duì)應(yīng)表,還有其他的比如: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 對(duì)應(yīng)的按鍵為 0 這個(gè),以此類推。那解決就明朗了,將所有字母和字符的按鍵映射添加進(jìn)行就ok了,至于按鍵值是多少我這邊直接參考了另外一個(gè)平臺(tái)的Generic.kl文件。重新編譯系統(tǒng)驗(yàn)證,此問題解決了。

疑惑問題:

  • 用相同Android版本的android.jar 查看keyCode對(duì)應(yīng)的值和Generic.kl文件里描述的不一樣,此問題還沒有弄明白為什么,系統(tǒng)說兩個(gè)是不相關(guān)的?

  • 發(fā)現(xiàn)在兩個(gè)平臺(tái)上有大部分按鍵值在一致的,但存在分別的是不樣的,不明白怎么定義的。依據(jù)是啥?

Android是什么

Android是一種基于Linux內(nèi)核的自由及開放源代碼的操作系統(tǒng),主要使用于移動(dòng)設(shè)備,如智能手機(jī)和平板電腦,由美國Google公司和開放手機(jī)聯(lián)盟領(lǐng)導(dǎo)及開發(fā)。

關(guān)于“Android設(shè)備如何獲取掃碼槍掃描的內(nèi)容”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,使各位可以學(xué)到更多知識(shí),如果覺得文章不錯(cuò),請(qǐng)把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI