Android中二維碼掃描的最常用庫是zxing和zbar,zxing項目位址為https://github.com/zxing/zxing,目前還有多個人在維護。zbar主要用C來寫的,對速度有要求的可使用zbar,但目前沒有在維護,項目位址:https://github.com/ZBar/ZBar。
Zxing
之前做畢業設計的時候用到了二維碼掃描功能,github上Zxing項目代碼很多,但其實用到的隻有一部分,由于趕時間是以打算找個精簡的快速內建,這裡分享一下內建過程,比較實用。
國際慣例,先上效果圖:
1.引入jar包
2.copy Zxing包到項目
這裡包名不一樣肯定會報錯,我們暫時不管,先把資源檔案copy過來,後面來做處理。
3.導入相關資源檔案
copy res底下的相關資源檔案,如下:
drawable、drawable-hdpi和layout
raw檔案和values檔案
注:values中相關資源不要直接替換,否則會覆寫之前的,需要打開檔案将内容加到自己項目對應檔案中。
4.AndroidManifest.xml加入相關權限和掃描的Activity
<uses-permission android:name="android.permission.CAMERA" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="android.permission.FLASHLIGHT" />
<activity
android:name=".zxing.android.CaptureActivity"
android:screenOrientation="portrait"
android:theme="@android:style/Theme.NoTitleBar" />
5.Clean Projects,修改報錯的類
報錯無非就是包名不對,修改為自己包名即可
6.capture.xml的ViewfinderView改成自己包名下的
7.調起掃描界面 擷取掃描結果
在需要打開掃描界面的地方直接跳轉到CaptureActivity就OK(使用startActivityForResult)
/**
* 跳轉到掃碼界面掃碼
*/
private void goScan(){
Intent intent = new Intent(MainActivity.this, CaptureActivity.class);
startActivityForResult(intent, REQUEST_CODE_SCAN);
}
在onActivityResult的回調中即可擷取掃描内容
@Override
protected void onActivityResult(int requestCode, int resultCode, Intent data) {
super.onActivityResult(requestCode, resultCode, data);
// 掃描二維碼/條碼回傳
if (requestCode == REQUEST_CODE_SCAN && resultCode == RESULT_OK) {
if (data != null) {
//傳回的文本内容
String content = data.getStringExtra(DECODED_CONTENT_KEY);
//傳回的BitMap圖像
Bitmap bitmap = data.getParcelableExtra(DECODED_BITMAP_KEY);
}
}
}
動态權限申請
由于掃描需要調用相機,打開攝像頭屬于敏感權限,是以需要進行權限的動态申請,如下
//動态權限申請
if (ContextCompat.checkSelfPermission(MainActivity.this, Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {
ActivityCompat.requestPermissions(MainActivity.this, new String[]{Manifest.permission.CAMERA}, 1);
} else {
//掃碼
goScan();
}
@Override
public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
switch (requestCode) {
case 1:
if (grantResults.length > 0 && grantResults[0] == PackageManager.PERMISSION_GRANTED) {
//掃碼
goScan();
} else {
Toast.makeText(this, "你拒絕了權限申請,無法打開相機掃碼喲!", Toast.LENGTH_SHORT).show();
}
break;
default:
}
}
上面的代碼就是動态申請權限的流程,首先判斷使用者是不是已經給我們權限授權了,使用ContextCompat.checkSelfPermission()方法,第一個參數是Context,第二個參數是具體的權限名稱,如果等于PackageManager.PERMISSION_GRANTED表明已授權,不等于就是沒有授權。
如果已授權就直接做後面的操作,如果沒有授權,需要調用ActivityCompat.requestPermissions()方法申請授權,第一個參數是目前Activity執行個體,第二個參數是權限數組,第三個是請求碼。
使用者的選擇将會回調到onRequestPermissionsResult()方法中,授權結果封裝在grantResults參數中,如果grantResults長度大于0且grantResults[0]等于PackageManager.PERMISSION_GRANTED,也就是上面權限數組中加入的第一個打開攝像頭的權限被授權,則可跳轉至掃描界面掃碼,否則提示使用者未打開權限無法使用。
效果
打開個6.0以上模拟器試試
點選按鈕将彈出權限申請,使用者授權後方可進行二維碼掃描。
源碼已上傳至GitHub,需要的可以下載下傳:https://github.com/yangxch/ScanZxing
更多技術幹貨,歡迎關注我的公衆号:ChaoYoung