Android設(shè)備獲取掃碼槍掃描的內(nèi)容與可能遇到的問(wèn)題解決
前言
大家應(yīng)該都有所體會(huì),在生活中條形碼掃碼槍可是隨處可見(jiàn),可以很迅速地掃描出條形碼內(nèi)容,比什么手機(jī)相機(jī)掃碼快了不是一點(diǎn)兩點(diǎn)。
為了節(jié)約成本,掃碼槍可以直接通過(guò)藍(lán)牙連接android或其他設(shè)備。
那么android設(shè)備如何通過(guò)藍(lán)牙獲取掃描內(nèi)容的呢?
1. 藍(lán)牙配對(duì),連接設(shè)備
打開(kāi)系統(tǒng)設(shè)置,找到藍(lán)牙,打開(kāi)掃碼槍,配對(duì)掃碼槍設(shè)備。輸入一個(gè)固定的配對(duì)碼,一般掃碼槍說(shuō)明書(shū)里都有寫(xiě)。配對(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.檢測(cè)掃碼槍的連接狀態(tài)
通常來(lái)說(shuō),掃碼槍設(shè)備也相當(dāng)于普通外接輸入設(shè)備類型,外接鍵盤(pán)。
我這款掃碼槍設(shè)備返回的是如下藍(lán)牙類型。
BluetoothClass.Device.Major.PERIPHERAL
一般而言,通過(guò)如下這種方式就可以獲得到我們掃碼槍設(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è)備信息
}
}
開(kāi)發(fā)過(guò)程中,必然會(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版本以上才可用。
后來(lái)轉(zhuǎn)頭一想,既然掃碼槍也是輸入設(shè)備,我們可以不同藍(lán)牙設(shè)備狀態(tài)檢測(cè)入手,改為從輸入設(shè)備檢測(cè)入手。于是,
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();
//大小寫(xiě)區(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();
//字母大小寫(xiě)判斷
checkLetterStatus(event);
if (event.getAction() == KeyEvent.ACTION_DOWN) {
char aChar = getInputCode(event);;
if (aChar != 0) {
mStringBufferResult.append(aChar);
}
if (keyCode == KeyEvent.KEYCODE_ENTER) {
//若為回車(chē)鍵,直接返回
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鍵,表示大寫(xiě)
mCaps = true;
} else {
//松開(kāi)shift鍵,表示小寫(xiě)
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中重寫(xiě)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ī)型無(wú)法判斷外接鍵盤(pán)信息,如三星。
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)解決掃碼槍無(wú)法輸入字母和字符問(wèn)題
問(wèn)題:
在使用掃碼槍掃碼條碼的時(shí)候明明有字母和字符,但是輸入到Android系統(tǒng)卻沒(méi)喲,輸入到電腦是正常的,這就很奇怪,讓一個(gè)搞上層開(kāi)發(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文件。
打開(kāi)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)證,此問(wèn)題解決了。
疑惑問(wèn)題:
- 用相同Android版本的android.jar 查看keyCode對(duì)應(yīng)的值和Generic.kl文件里描述的不一樣,此問(wèn)題還沒(méi)有弄明白為什么,系統(tǒng)說(shuō)兩個(gè)是不相關(guān)的?
- 發(fā)現(xiàn)在兩個(gè)平臺(tái)上有大部分按鍵值在一致的,但存在分別的是不樣的,不明白怎么定義的。依據(jù)是啥?
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
詳解 Android中Libgdx使用ShapeRenderer自定義Actor解決無(wú)法接收到Touch事件的問(wèn)題
這篇文章主要介紹了詳解 Android中Libgdx使用ShapeRenderer自定義Actor解決無(wú)法接收到Touch事件的問(wèn)題的相關(guān)資料,希望通過(guò)本文能幫助到大家解決這樣的問(wèn)題,需要的朋友可以參考下2017-09-09
Android CountDownTimer實(shí)現(xiàn)倒計(jì)時(shí)器
這篇文章主要為大家詳細(xì)介紹了Android CountDownTimer實(shí)現(xiàn)倒計(jì)時(shí)效果的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-02-02
Android實(shí)現(xiàn)仿通訊錄側(cè)邊欄滑動(dòng)SiderBar效果代碼
這篇文章主要介紹了Android實(shí)現(xiàn)仿通訊錄側(cè)邊欄滑動(dòng)SiderBar效果代碼,實(shí)例分析了通訊錄側(cè)邊欄滑動(dòng)效果的實(shí)現(xiàn)技巧,并附帶完整實(shí)例代碼供讀者下載參考,需要的朋友可以參考下2015-10-10
Android實(shí)現(xiàn)加載廣告圖片和倒計(jì)時(shí)的開(kāi)屏布局
這篇文章主要介紹了Android實(shí)現(xiàn)加載廣告圖片和倒計(jì)時(shí)的開(kāi)屏布局,需要的朋友可以參考下2014-07-07
Android開(kāi)發(fā)之Activity管理工具類完整示例
這篇文章主要介紹了Android開(kāi)發(fā)之Activity管理工具類,集合完整實(shí)例形式分析了Android操作Activity創(chuàng)建、添加、獲取、移除等相關(guān)操作技巧,需要的朋友可以參考下2018-01-01
Android串口開(kāi)發(fā)之使用JNI實(shí)現(xiàn)ANDROID和串口通信詳解
這篇文章主要給大家介紹了關(guān)于Android串口開(kāi)發(fā)之使用JNI實(shí)現(xiàn)ANDROID和串口通信的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2018-01-01
使用DrawerLayout組件實(shí)現(xiàn)側(cè)滑抽屜的功能
DrawerLayout組件同樣是V4包中的組件,也是直接繼承于ViewGroup類,所以說(shuō)是一個(gè)容器類,下面通過(guò)本文給大家介紹使用DrawerLayout組件實(shí)現(xiàn)側(cè)滑抽屜的功能,感興趣的朋友一起看下吧2016-08-08
Android開(kāi)發(fā)實(shí)現(xiàn)圖片的上傳下載
這篇文章主要為大家詳細(xì)介紹了Android開(kāi)發(fā)實(shí)現(xiàn)圖片的上傳下載,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-09-09
Android開(kāi)發(fā)中Intent傳遞對(duì)象的方法分析
這篇文章主要介紹了Android開(kāi)發(fā)中Intent傳遞對(duì)象的方法,結(jié)合實(shí)例分析了Intent傳遞對(duì)象所涉及的具體方法、實(shí)現(xiàn)步驟與相關(guān)注意事項(xiàng),需要的朋友可以參考下2016-02-02
Flutter質(zhì)感設(shè)計(jì)之表單輸入
這篇文章主要為大家詳細(xì)介紹了Flutter質(zhì)感設(shè)計(jì)之表單輸入,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-08-08

