仿微信二维码极速扫描(MLKit及CameraX初体验) 您所在的位置:网站首页 模拟微信 仿微信二维码极速扫描(MLKit及CameraX初体验)

仿微信二维码极速扫描(MLKit及CameraX初体验)

2023-03-21 23:53| 来源: 网络整理| 查看: 265

一直为网上找的扫码项目无法快速且识别率高的扫描并解析出二维码而苦恼,zxing自己修改继承的难度又比较高(菜是原罪)。好在,经高人指导,接触到了谷歌的MLkit工程,实际体验下来,是真的好用,在此做一个经验分享,也是为了自己以后能记住 先来看看继承之后的效果图

image.png 跟微信扫码比起来,不能说是一模一样,但至少是毫无关系了

image.png 如此迷你的二维码也能解析出来哦

接下来,我们来看下如何具体实现吧

首先,来看下重中之重,MLKit的项目介绍吧 developers.google.cn/ml-kit

可以看到,这个项目除了扫码功能,还有其他各种各样的好用且免费的SDK,像是文字识别,人脸识别等等,都可以依赖这个项目轻松搞定。无奈小弟才疏学浅,只接触了该项目的扫码功能,等来日水平提升了,定将其他的功能也一一体验一遍

对了,这个项目除了Android端,还有IOS端的哦,感兴趣的IOS小伙伴可以自行参考

要使用这个项目,必不可少的,需要先引入工程

dependencies { // ... // Use this dependency to bundle the model with your app implementation 'com.google.mlkit:barcode-scanning:16.1.1' } 复制代码

有google play条件的小伙伴,还可以选择添加google play的相关引用,甚至可以使用google play model

dependencies { // ... // Use this dependency to use the dynamically downloaded model in Google Play Services implementation 'com.google.android.gms:play-services-mlkit-barcode-scanning:16.1.4' } 复制代码 ... 复制代码

CameraX的支持当然也要一并加上

// 版本号 def camerax_version = "1.0.0-rc03" // 对camera 及 camera2的支持,可自行选择 implementation "androidx.camera:camera-core:${camerax_version}" implementation "androidx.camera:camera-camera2:${camerax_version}" 复制代码

项目使用的是基于jetpack lifecycle的框架,所以加上lifecycle的支持

implementation "androidx.camera:camera-lifecycle:${camerax_version}" 复制代码

预览用到了camerax自带的预览控件

implementation "androidx.camera:camera-view:1.0.0-alpha22" 复制代码

接下来就是具体的代码实现了 首先,用到了相机及相册,自然需要添加相应的权限申请

ActivityCompat.requestPermissions(this, arrayOf(Manifest.permission.CAMERA,Manifest.permission.WRITE_EXTERNAL_STORAGE),REQUEST_PERMISSION) 复制代码

先来看看扫码界面的实现,主要包含了预览的PreviewView及用于绘制扫码线条及扫码结果的ScanOverlay

复制代码

接下来看一下如何开启CameraX的预览

CameraX自带了检测相机是否可用的监听,可以在相机可用之后,再进行后续操作

cameraProviderFuture = ProcessCameraProvider.getInstance(this) cameraProviderFuture.addListener(Runnable { val cameraProvider = cameraProviderFuture.get() bindScan(cameraProvider, overlay.width,overlay.height) }, ContextCompat.getMainExecutor(this@BarcodeScanningActivity)) 复制代码

并且,CameraX绑定生命周期控件后,可以根据生命周期,自行释放相机,妈妈再也不用担心忘记关相机啦

val preview : Preview = Preview.Builder().build() //绑定预览 preview.setSurfaceProvider(previewView.surfaceProvider) //使用后置相机 val cameraSelector : CameraSelector = CameraSelector.Builder() .requireLensFacing(CameraSelector.LENS_FACING_BACK) .build() //将相机绑定到当前控件的生命周期 camera = cameraProvider.bindToLifecycle(this as LifecycleOwner, cameraSelector, imageAnalysis, preview) 复制代码

这样,就可以开启预览啦,可以注意到,代码中绑定到生命周期的时候,使用了一个imageAnalysis的useCases,而这,就是用于图片扫描的组件了

//配置图片扫描 val imageAnalysis = ImageAnalysis.Builder() .setTargetResolution(Size(width, height)) .setBackpressureStrategy(ImageAnalysis.STRATEGY_KEEP_ONLY_LATEST) .build() 复制代码

光这样还不行,还需要为图片扫描配置解析扫码内容的解析器QRCodeAnalyser,说了这么多,终于要用到MLKit的二维码解析了 来看一下QRCodeAnalyser的具体实现

@SuppressLint("UnsafeExperimentalUsageError") override fun analyze(imageProxy: ImageProxy) { val mediaImage = imageProxy.image ?: kotlin.run { imageProxy.close() return } val image = InputImage.fromMediaImage(mediaImage, imageProxy.imageInfo.rotationDegrees) detector.process(image) .addOnSuccessListener { barCodes -> if (barCodes.size > 0){ listener.invoke(barCodes[0],imageProxy.width,imageProxy.height) //接收到结果后,就关闭解析 detector.close() } } .addOnFailureListener { Log.d(TAG, "Error: ${issage}") } .addOnCompleteListener { imageProxy.close() } } 复制代码

这个类主要实现了ImageAnalysis.Analyzer,并实现了analyze解析方法,其中imageProxy就是CameraX传递过来的图片扫描内容了 其主要内容,就是根据传递过来的图片扫描内容,使用detector进行解析,而这个detector又是怎么来的呢

//配置当前扫码格式 private val options = BarcodeScannerOptions.Builder() .setBarcodeFormats( Barcode.FORMAT_QR_CODE, Barcode.FORMAT_AZTEC ) .build() //获取解析器 private val detector = BarcodeScanning.getClient(options) 复制代码

这里的BarcodeScanning就是MLKit提供的二维码解析组件了

解析完成后,对扫码结果进行后续操作

//绑定图片扫描解析 imageAnalysis.setAnalyzer(Executors.newSingleThreadExecutor(), QRCodeAnalyser{ barcode,imageWidth,imageHeight-> //解绑当前所有相机操作 cameraProvider.unbindAll() //初始化缩放比例 initScale(imageWidth,imageHeight) barcode.boundingBox?.let {//扫描二维码的外边框矩形 overlay.addRect(translateRect(it)) Log.i(TAG, "bindScan: left:${it.left} right:${it.right} top:${it.top} bottom:${it.bottom}") } Handler().postDelayed({ //延迟1S后返回结果 val intent = Intent() intent.putExtra(SCAN_RESULT,barcode.rawValue) setResult(Activity.RESULT_OK, intent) },1000) }) 复制代码

这里需要注意的是,因为手机上overlay的绘制区域与实际的图片扫描区域并不是一致的,所以需要对返回内容中二维码的外边框矩形进行一个转换,否则,最后绘制的结果点位置会出错

首先,根据绘制区域的大小和图片扫描区域的大小,初始化缩放比例

private fun initScale(imageWidth : Int, imageHeight : Int){ if(isPortraitMode(this)){ scaleY = overlay.height.toFloat() / imageWidth.toFloat() scaleX = overlay.width.toFloat() / imageHeight.toFloat() }else{ scaleY = overlay.height.toFloat() / imageHeight.toFloat() scaleX = overlay.width.toFloat() / imageWidth.toFloat() } } 复制代码

这里需要注意的是,因为实际相机和扫描的角度,在竖屏模式下,有一个90度的差别,所以这里实际上做的比例是用绘制区域的高和扫描区域的宽,绘制区域的宽和扫描区域的高去做一个计算

private fun translateX(x: Float): Float = x * scaleX private fun translateY(y: Float): Float = y * scaleY //将扫描的矩形换算为当前屏幕大小 private fun translateRect(rect: Rect) = RectF( translateX(rect.left.toFloat()), translateY(rect.top.toFloat()), translateX(rect.right.toFloat()), translateY(rect.bottom.toFloat()) ) 复制代码

根据比例去计算实际的二维码外边框矩形,并计算出最后的绘制点,进行绘制

最后的扫码效果,无论是扫码速度,还是解析率,我觉得都要比之前用的基于zxing的扫码工程要高,感兴趣的小伙伴可以自行尝试一下

demo链接,提取码:8sn6

demo已上传百度云,本来想上传github的,无奈家里网络问题,实在上传不上去,等后面有空再试试



【本文地址】

公司简介

联系我们

今日新闻

    推荐新闻

    专题文章
      CopyRight 2018-2019 实验室设备网 版权所有