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