Android接入USB掃碼模塊的方法
前言
USB掃碼模塊可以是掃描盒子或者掃碼槍之類的,一根USB線作為供電和數(shù)據(jù)通信使用,有些掃碼模塊支持虛擬串口模式,虛擬串口模式讀取數(shù)據(jù)會比較簡單一點,和普通的串口一樣操作即可,就是通過虛擬串口口+波特率即可獲取到數(shù)據(jù),這里主要講讀取USB模式下的數(shù)據(jù)。
1.讀取USB模式下的數(shù)據(jù)
USB模式下的掃碼模塊相當于一個外接鍵盤,也就是它必須在有光標的地方才能進行掃碼,且是直接把掃到的內容自動輸入到輸入框中,并不受我們的控制,所以我們必須另外想辦法,安卓系統(tǒng)中有這么個方dispatchKeyEvent(KeyEvent event),它就是用來處理我們鍵盤的輸入事件的,如果我們攔截該方法,把它交給我們自己去處理,這樣我們就可以不通過Edittext從而獲取到掃碼頭傳過來的數(shù)據(jù)了。
值得注意的是掃碼輸出字符是連續(xù)輸出的,也就是說掃碼的數(shù)據(jù)并不是一下子輸出所有的數(shù)據(jù),而是有間隔的輸出字符,間隔是比較短的,大概有10ms左右,有些掃碼模塊有回車結束符,有些沒有,所以我們需要封裝代碼使用,如下:
import android.os.Build;
import android.os.Handler;
import android.util.Log;
import android.view.KeyEvent;
import org.greenrobot.eventbus.EventBus;
import java.util.ArrayList;
import java.util.List;
public class ScanGunHelper {
private final static long MESSAGE_DELAY = 200;
private StringBuffer mStringBufferResult = new StringBuffer();
private Handler mHandler = new Handler();
private Runnable mScanningFishedRunnable = new Runnable() {
@Override
public void run() {
performScanSuccess();
}
};
private List<Integer> keyCodeList = new ArrayList<>();
private static class SingletonHolder {
private static final ScanGunHelper instance = new ScanGunHelper();
}
private ScanGunHelper() {
}
public static ScanGunHelper getInstance() {
return ScanGunHelper.SingletonHolder.instance;
}
/**
* 返回掃碼成功后的結果
*/
private void performScanSuccess() {
try {
boolean mCaps = keyCodeList.contains(KeyEvent.KEYCODE_SHIFT_RIGHT) || keyCodeList.contains(KeyEvent.KEYCODE_SHIFT_LEFT);
for (int keyCode : keyCodeList) {
char aChar = getInputCode(keyCode, mCaps);
if (aChar != 0) {
mStringBufferResult.append(aChar);
}
}
String barcode = mStringBufferResult.toString();
Log.e("ScanGunHelper", "barcode==" + barcode);
mStringBufferResult.setLength(0);
keyCodeList.clear();
if (mHandler != null) {
mHandler.removeCallbacks(mScanningFishedRunnable);
}
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 掃碼槍事件解析
*/
public void analysisKeyEvent(KeyEvent event) {
//Virtual是我所使用機器的內置軟鍵盤的名字
//在這判斷是因為項目中避免和軟鍵盤沖突(掃碼槍和軟鍵盤都屬于按鍵事件)
int keyCode = event.getKeyCode();
if (event.getAction() == KeyEvent.ACTION_UP) {
keyCodeList.add(keyCode);
if (keyCode == KeyEvent.KEYCODE_ENTER) {
mHandler.removeCallbacks(mScanningFishedRunnable);
mHandler.post(mScanningFishedRunnable);
} else {
mHandler.removeCallbacks(mScanningFishedRunnable);
mHandler.postDelayed(mScanningFishedRunnable, MESSAGE_DELAY);
}
}
}
//獲取掃描內容
private char getInputCode(int keyCode, boolean mCaps) {
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 {
//其他符號
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 static boolean isScanGunExist(KeyEvent event) {
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
Log.e("ScanGunHelper", "ProductId==" + event.getDevice().getProductId() + ",VendorId==" + event.getDevice().getVendorId());
if (event.getDevice().getVendorId() == 1409 && event.getDevice().getProductId() == 262) {
return true;
}
}
return false;
}
}
如果用的掃碼模塊有確定的ProductId和VendorId,可以直接過濾,寫死ProductId和VendorId會使得程序很不靈活,萬一換了一款掃碼模塊就不適配了,所以建議做成可配置,避免很多麻煩
如果不用ProductId和VendorId過濾輸入事件,那就區(qū)分一下軟鍵盤即可,系統(tǒng)自動軟件盤的DeviceId為-1,只要不等于-1,就是掃碼模塊輸入的字符。
單列模式封裝代碼,主要是考慮到一臺安卓一體機不可能接入兩個掃碼模塊,當成唯一設備作為處理即可。
2.在基類的Activity中使用,確保每個界面能攔截到數(shù)據(jù),方便使用,不需要每個界面都攔截
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v7.app.AppCompatActivity;
import android.view.KeyEvent;
import android.view.WindowManager;
public abstract class BaseActivity extends AppCompatActivity {
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
}
@Override
public boolean dispatchKeyEvent(KeyEvent event) {
if (event.getDeviceId() > 0) {
ScanGunHelper.getInstance().analysisKeyEvent(event);
return true;
}
return super.dispatchKeyEvent(event);
}
}
總結:
現(xiàn)在的USB掃碼模塊是比較流行,但是掃碼模塊與終端設備的通信方式花樣百出,有wifi、藍牙、USB、串口等等,但是萬變不離其宗,對于安卓設備而言,掃碼模塊就是個外接鍵盤,不管是什么方式通信也好(除了串口以外),最終呈現(xiàn)的形態(tài)就是個外接鍵盤,當然,我們也可以使用藍牙讀取掃碼數(shù)據(jù),這些都是so esay了,串口更加簡單,有實際的串口線更好,虛擬串口會麻煩一點,每次插拔虛擬串口號會不固定,不同的安卓設備表現(xiàn)出的串口號也盡不相同,所以虛擬串口模式需要做很多適配工作,不能寫死一個虛擬串口號,這樣會照成程序不靈活、不適配,做成可配置,才能應付多變的招標現(xiàn)場或者客戶需求現(xiàn)場。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
Android自定義PopupWindow仿點擊彈出分享功能
這篇文章主要為大家詳細介紹了Android自定義PopupWindow仿點擊彈出分享功能,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-02-02
Android從0到完整項目(1)使用Android studio 創(chuàng)建項目詳解
本篇文章主要介紹了Android從0到完整項目(1)使用Android studio 創(chuàng)建項目詳解,具有一定的參考價值,有興趣的可以了解一下2017-07-07
Android studio利用gradle打jar包并混淆的方法詳解
昨天準備把寫好的代碼使用gradle打jar包出來,并打算加混淆。打jar包容易,結果在混淆上走了彎路。所以這篇文章主要介紹了關于Android studio利用gradle打jar包并混淆的方法,需要的朋友可以參考下。2017-03-03
Android FaceDetector實現(xiàn)人臉檢測功能
這篇文章主要為大家詳細介紹了Android FaceDetector實現(xiàn)人臉檢測功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-05-05
android顯示TextView文字的倒影效果實現(xiàn)代碼
這篇文章主要介紹了android顯示TextView文字的倒影效果實現(xiàn)代碼,需要的朋友可以參考下2014-02-02
僅5步搞定Android開發(fā)環(huán)境部署 Android開發(fā)環(huán)境搭建教程
僅5步搞定Android開發(fā)環(huán)境部署,這篇文章主要為大家詳細介紹了Android開發(fā)環(huán)境搭建教程,感興趣的小伙伴們可以參考一下2016-02-02

