CameraX的使用

<ImageButton

android:id=“@+id/camera_capture_button”

android:layout_width=“92dp”

android:layout_height=“92dp”

android:layout_marginBottom=“80dp”

android:scaleType=“fitCenter”

android:background=“@drawable/ic_shutter”

app:layout_constraintLeft_toLeftOf=“parent”

app:layout_constraintRight_toRightOf=“parent”

app:layout_constraintBottom_toBottomOf=“parent”

android:elevation=“2dp”

/>

<androidx.camera.view.PreviewView

android:id=“@+id/viewFinder”

android:layout_width=“match_parent”

android:layout_height=“match_parent” />

第三步

在AndroidManifest.xml文件中申请权限

在第四步中要在activity的kt文件中添加请求权限的方法

第四步

创建mainactivity的kt文件,代码如下:

import androidx.appcompat.app.AppCompatActivity

import android.os.Bundle

import android.Manifest

import android.content.pm.PackageManager

import android.net.Uri

import android.util.Log

import android.widget.Toast

import androidx.core.app.ActivityCompat

import androidx.core.content.ContextCompat

import java.util.concurrent.Executors

import androidx.camera.core.*

import androidx.camera.lifecycle.ProcessCameraProvider

import kotlinx.android.synthetic.main.activity_main.*

import java.io.File

import java.nio.ByteBuffer

import java.text.SimpleDateFormat

import java.util.*

import java.util.concurrent.ExecutorService

typealias LumaListener = (luma: Double) -> Unit

class MainActivity : AppCompatActivity() {

private var imageCapture: ImageCapture? = null

private lateinit var outputDirectory: File

private lateinit var cameraExecutor: ExecutorService

override fun onCreate(savedInstanceState: Bundle?) {

super.onCreate(savedInstanceState)

setContentView(R.layout.activity_main)

// 请求相机权限

if (allPermissionsGranted()) {

startCamera()

} else {

ActivityCompat.requestPermissions(

this, REQUIRED_PERMISSIONS, REQUEST_CODE_PERMISSIONS)

}

// 设置拍照的监听按钮

camera_capture_button.setOnClickListener { takePhoto() }

outputDirectory = getOutputDirectory()

cameraExecutor = Executors.newSingleThreadExecutor()

}

private fun takePhoto() {

// 获得可修改的图像捕获用例

val imageCapture = imageCapture ?: return

// 创建带时间戳的输出文件用于保存文件

val photoFile = File(

outputDirectory,

SimpleDateFormat(FILENAME_FORMAT, Locale.US

).format(System.currentTimeMillis()) + “.jpg”)

// 创建输出选项对象

val outputOptions = ImageCapture.OutputFileOptions.Builder(photoFile).build()

// 设置图像捕获监听器,拍照后触发

imageCapture.takePicture(

outputOptions, ContextCompat.getMainExecutor(this), object : ImageCapture.OnImageSavedCallback {

override fun onError(exc: ImageCaptureException) {

Log.e(TAG, “Photo capture failed: ${exc.message}”, exc)

}

override fun onImageSaved(output: ImageCapture.OutputFileResults) {

val savedUri = Uri.fromFile(photoFile)

val msg = “Photo capture succeeded: $savedUri”

Toast.makeText(baseContext, msg, Toast.LENGTH_SHORT).show()

Log.d(TAG, msg)

}

})

}

private fun startCamera() {

val cameraProviderFuture = ProcessCameraProvider.getInstance(this)

cameraProviderFuture.addListener(Runnable {

// 用于将摄像机的生命周期绑定到生命周期所有者

val cameraProvider: ProcessCameraProvider = cameraProviderFuture.get()

val preview = Preview.Builder()

.build()

.also {

it.setSurfaceProvider(viewFinder.createSurfaceProvider())

}

imageCapture = ImageCapture.Builder()

.build()

// 默认选择后置摄像头,这边可以设置方法选择摄像头

val cameraSelector = CameraSelector.DEFAULT_BACK_CAMERA

try {

cameraProvider.unbindAll()

// 将用例绑定到相机

cameraProvider.bindToLifecycle(

this, cameraSelector, preview, imageCapture)

} catch(exc: Exception) {

Log.e(TAG, “Use case binding failed”, exc)

}

}, ContextCompat.getMainExecutor(this))

}

private fun allPermissionsGranted() = REQUIRED_PERMISSIONS.all {

ContextCompat.checkSelfPermission(

baseContext, it) == PackageManager.PERMISSION_GRANTED

}

//申请权限的方法

override fun onRequestPermissionsResult(

requestCode: Int, permissions: Array, grantResults:

IntArray) {

if (requestCode == REQUEST_CODE_PERMISSIONS) {

if (allPermissionsGranted()) {

startCamera()

} else {

Toast.makeText(this,

“Permissions not granted by the user.”,

Toast.LENGTH_SHORT).show()

finish()

}

}

}

private fun getOutputDirectory(): File {

val mediaDir = externalMediaDirs.firstOrNull()?.let {

File(it, resources.getString(R.string.app_name)).apply { mkdirs() } }

return if (mediaDir != null && mediaDir.exists())

mediaDir else filesDir

}

override fun onDestroy() {

super.onDestroy()

cameraExecutor.shutdown()

}

companion object {

private const val TAG = “CameraXBasic”

private const val FILENAME_FORMAT = “yyyy-MM-dd-HH-mm-ss-SSS”

private const val REQUEST_CODE_PERMISSIONS = 10

private val REQUIRED_PERMISSIONS = arrayOf(Manifest.permission.CAMERA)

}

}

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

【延伸Android必备知识点】

【Android部分高级架构视频学习资源】

**Android精讲视频学习后更加是如虎添翼!**进军BATJ大厂等(备战)!现在都说互联网寒冬,其实无非就是你上错了车,且穿的少(技能),要是你上对车,自身技术能力够强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水!

**任何市场都是优胜略汰适者生存,只要你技术过硬,到哪里都不存在饱和不饱和的问题,所以重要的还是提升自己。懂得多是自己的加分项 而不是必须项。门槛高了只能证明这个市场在不断成熟化!**另外一千个读者就有一千个哈姆雷特,所以以上只是自己的关键,不喜勿喷!

如果你是卡在缺少学习资源的瓶颈上,那么刚刚好我能帮到你。欢迎关注会持续更新和分享的。

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!

强,公司换掉的代价大,怎么可能会被裁掉,都是淘汰末端的业务Curd而已!现如今市场上初级程序员泛滥,这套教程针对Android开发工程师1-6年的人员、正处于瓶颈期,想要年后突破自己涨薪的,进阶Android中高级、架构师对你更是如鱼得水!

**任何市场都是优胜略汰适者生存,只要你技术过硬,到哪里都不存在饱和不饱和的问题,所以重要的还是提升自己。懂得多是自己的加分项 而不是必须项。门槛高了只能证明这个市场在不断成熟化!**另外一千个读者就有一千个哈姆雷特,所以以上只是自己的关键,不喜勿喷!

如果你是卡在缺少学习资源的瓶颈上,那么刚刚好我能帮到你。欢迎关注会持续更新和分享的。

《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》,点击传送门即可获取!
  • 3
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
非常好的问题,使用CameraX可以轻松地实现拍照功能。首先需要在项目的build.gradle文件中添加以下依赖项: ``` implementation "androidx.camera:camera-core:1.0.0" implementation "androidx.camera:camera-camera2:1.0.0" ``` 然后,在XML布局文件中添加一个ImageView控件用于显示照片,以及一个Button控件用于触发拍照: ``` <ImageView android:id="@+id/photo_view" android:layout_width="match_parent" android:layout_height="match_parent"/> <Button android:id="@+id/camera_button" android:layout_width="wrap_content" android:layout_height="wrap_content" android:text="Take Photo"/> ``` 接下来,在Activity或Fragment中,可以使用以下代码来初始化CameraX,并启动相机预览: ``` val cameraProviderFuture = ProcessCameraProvider.getInstance(this) cameraProviderFuture.addListener(Runnable { val cameraProvider: ProcessCameraProvider = cameraProviderFuture.get() // 创建 Preview 对象 val preview = Preview.Builder() .build() // 创建 ImageCapture 对象 val imageCapture = ImageCapture.Builder() .build() // 绑定预览对象和 ImageCapture 对象到相机上 val cameraSelector = CameraSelector.DEFAULT_BACK_CAMERA cameraProvider.unbindAll() camera = cameraProvider.bindToLifecycle(this, cameraSelector, preview, imageCapture) // 预览的 Surface val previewSurfaceProvider = previewView.previewSurfaceProvider preview.setSurfaceProvider(previewSurfaceProvider) }, ContextCompat.getMainExecutor(this)) ``` 最后,在Button的Click事件中,可以使用以下代码来拍照: ``` val imageCapture = imageCapture ?: return@setOnClickListener val photoFile = File(externalMediaDirs.first(), "${System.currentTimeMillis()}.jpg") val outputOptions = ImageCapture.OutputFileOptions.Builder(photoFile).build() imageCapture.takePicture(outputOptions, ContextCompat.getMainExecutor(this), object : ImageCapture.OnImageSavedCallback { override fun onImageSaved(outputFileResults: ImageCapture.OutputFileResults) { // 显示照片 val savedUri = Uri.fromFile(photoFile) photo_view.post { photo_view.setImageURI(savedUri) } } override fun onError(exception: ImageCaptureException) { // 错误处理 } }) ``` 希望这些代码可以帮助您实现使用CameraX进行拍照。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值