个人博客:haichenyi.com。感谢关注
做过二维码扫描的童鞋都知道,要用Zxing,可是google官方推的Zxing,我们集成扫描的时候很慢,有时候就直接扫不出来,体验太差了。因为他适配了太多东西,我们只要一个扫描功能,然后就去网上搜索,怎么使zxing扫描变的很快,zxing的优化,然后跟着网上的博客,一步一步走,即使最终做完了,感觉体验还是不够好。我这里推荐一个二维码扫描的开源框架——BGAQRCode-Android
QRCode 扫描二维码、扫描条形码、相册获取图片后识别、生成带 Logo 二维码、支持微博微信、QQ 二维码扫描样式。他把Zbar和Zxing,都做了优化,扫描速度非常快,用户体验很好,用起来也非常方便
依赖
implementation 'cn.bingoogolapple:bga-qrcode-zbar:1.2.1'
或者
implementation 'cn.bingoogolapple:bga-qrcode-zxing:1.2.1'
用法
以Zbar为例(Zxing类似)
第一步:XML文件
<cn.bingoogolapple.qrcode.zbar.ZBarView
android:id="@+id/zbarview"
style="@style/MatchMatch"
app:qrcv_animTime="1000"
app:qrcv_borderColor="@android:color/white"
app:qrcv_borderSize="1dp"
app:qrcv_cornerColor="@color/colorPrimaryDark"
app:qrcv_cornerLength="20dp"
app:qrcv_cornerSize="3dp"
app:qrcv_isShowDefaultScanLineDrawable="true"
app:qrcv_maskColor="#33FFFFFF"
app:qrcv_rectWidth="200dp"
app:qrcv_scanLineColor="@color/colorPrimaryDark"
app:qrcv_topOffset="90dp" />
自定义属性说明
属性名 | 说明 | 默认值 |
---|---|---|
qrcv_topOffset | 扫描框距离 toolbar 底部的距离 | 90dp |
qrcv_cornerSize | 扫描框边角线的宽度 | 3dp |
qrcv_cornerLength | 扫描框边角线的长度 | 20dp |
qrcv_cornerColor | 扫描框边角线的颜色 | @android:color/white |
qrcv_rectWidth | 扫描框的宽度 | 200dp |
qrcv_barcodeRectHeight | 条码扫样式描框的高度 | 140dp |
qrcv_maskColor | 除去扫描框,其余部分阴影颜色 | #33FFFFFF |
qrcv_scanLineSize | 扫描线的宽度 | 1dp |
qrcv_scanLineColor | 扫描线的颜色「扫描线和默认的扫描线图片的颜色」 | @android:color/white |
qrcv_scanLineMargin | 扫描线距离上下或者左右边框的间距 | 0dp |
qrcv_isShowDefaultScanLineDrawable | 是否显示默认的图片扫描线「设置该属性后 qrcv_scanLineSize 将失效,可以通过 qrcv_scanLineColor 设置扫描线的颜色,避免让你公司的UI单独给你出特定颜色的扫描线图片」 | false |
qrcv_customScanLineDrawable | 扫描线的图片资源「默认的扫描线图片样式不能满足你的需求时使用,设置该属性后 qrcv_isShowDefaultScanLineDrawable、qrcv_scanLineSize、qrcv_scanLineColor 将失效」 | null |
qrcv_borderSize | 扫描边框的宽度 | 1dp |
qrcv_borderColor | 扫描边框的颜色 | @android:color/white |
qrcv_animTime | 扫描线从顶部移动到底部的动画时间「单位为毫秒」 | 1000 |
qrcv_isCenterVertical | 扫描框是否垂直居中,该属性为true时会忽略 qrcv_topOffset 属性 | false |
qrcv_toolbarHeight | Toolbar 的高度,通过该属性来修正由 Toolbar 导致扫描框在垂直方向上的偏差 | 0dp |
qrcv_isBarcode | 是否是扫条形码 | false |
qrcv_tipText | 提示文案 | null |
qrcv_tipTextSize | 提示文案字体大小 | 14sp |
qrcv_tipTextColor | 提示文案颜色 | @android:color/white |
qrcv_isTipTextBelowRect | 提示文案是否在扫描框的底部 | false |
qrcv_tipTextMargin | 提示文案与扫描框之间的间距 | 20dp |
qrcv_isShowTipTextAsSingleLine | 是否把提示文案作为单行显示 | false |
qrcv_isShowTipBackground | 是否显示提示文案的背景 | false |
qrcv_tipBackgroundColor | 提示文案的背景色 | #22000000 |
qrcv_isScanLineReverse | 扫描线是否来回移动 | true |
qrcv_isShowDefaultGridScanLineDrawable | 是否显示默认的网格图片扫描线 | false |
qrcv_customGridScanLineDrawable | 扫描线的网格图片资源 | null |
qrcv_isOnlyDecodeScanBoxArea | 是否只识别扫描框区域的二维码 | false |
第二步:获取控件,并实现QRCodeView.Delegate接口
QRCodeView mQRCodeView = (ZBarView) findViewById(R.id.zbarview);
mQRCodeView.setDelegate(this);
@Override
public void onScanQRCodeSuccess(String result) {
vibrate();
Toast.makeText(this,result,Toast.LENGTH_SHORT).show();
mQRCodeView.startSpot();
}
@Override
public void onScanQRCodeOpenCameraError() {
Toast.makeText(this,"错误",Toast.LENGTH_SHORT).show();
}
第四步:实现onStart,onStop,onDestroy方法
@Override
protected void onStart() {
super.onStart();
mQRCodeView.startCamera();//打开相机
mQRCodeView.showScanRect();//显示扫描框
mQRCodeView.startSpot();//开始识别二维码
//mQRCodeView.openFlashlight();//开灯
//mQRCodeView.closeFlashlight();//关灯
}
@Override
protected void onStop() {
mQRCodeView.stopCamera();
super.onStop();
}
@Override
protected void onDestroy() {
mQRCodeView.onDestroy();
super.onDestroy();
}
private void vibrate() {
Vibrator vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);
vibrator.vibrate(200);
}
第五步:添加权限
<uses-permission android:name="android.permission.CAMERA"/>
<uses-permission android:name="android.permission.VIBRATE"/>
经过上面5步,扫描功能就已经出来了,Zbar扫描比Zxing快,Zbar是基于C实现的,Zxing是基于C++。有个问题就是Zbar无法生成二维码,如果你是集成的BGA的Zxing可以生成二维码。Zxing生成二维码的方式如下:
耗时操作,新开线程
//中文
Bitmap bitmap = QRCodeEncoder.syncEncodeQRCode("海晨忆", BGAQRCodeUtil.dp2px(this, 150));
//带logo
Bitmap logoBitmap = BitmapFactory.decodeResource(TestGeneratectivity.this.getResources(), R.mipmap.logo);
QRCodeEncoder.syncEncodeQRCode("海晨忆", BGAQRCodeUtil.dp2px(TestGeneratectivity.this, 150), Color.parseColor("#ff0000"), logoBitmap);
解析图片二维码
//mEnglishLogoIv:显示二维码的控件
mEnglishLogoIv.setDrawingCacheEnabled(true);
Bitmap bitmap = mEnglishLogoIv.getDrawingCache();
//返回二维码图片里的内容 或 null
String result = QRCodeDecoder.syncDecodeQRCode(bitmap);
String result = QRCodeDecoder.syncDecodeQRCode(picturePath);