使用CameraX拍摄视频

简介: 使用CameraX拍摄视频

概览


概念和简单的使用建议跳到上一篇文章去看juejin.cn/post/696831…


使用(简单拍摄一段视频并展示)


打开相机并设置相关参数

private fun openCamera() {
        cameraProviderFuture = ProcessCameraProvider.getInstance(this)//获得provider实例
        cameraProviderFuture.addListener({
            cameraProvider = cameraProviderFuture.get()
            bindPreview()
        }, ContextCompat.getMainExecutor(this))
    }
    @SuppressLint("RestrictedApi")
    private fun bindPreview(
    ) {
        val preview: Preview = Preview.Builder()
            .build()
        val cameraSelector: CameraSelector = CameraSelector.Builder()
            .requireLensFacing(CameraSelector.LENS_FACING_BACK)
            .build()
        preview.setSurfaceProvider(previdew.surfaceProvider)
        videoCapture = VideoCapture.Builder()
            .setBitRate(1024 * 1024)
            .setVideoFrameRate(36)
            .setTargetRotation(previdew.display.rotation)
            .build()
        var camera = cameraProvider?.bindToLifecycle(
            this,
            cameraSelector,
            videoCapture,
            preview
        )
    }
复制代码


拍摄视频代码

@SuppressLint("RestrictedApi", "MissingPermission")
    fun takeVideo() {
        val faceVideoFile = getExternalFilesDir("videoFile")?.also {
            if (it.exists()) {
                it.mkdirs()
            }
        }
        val destVideoFile = File(faceVideoFile, "${System.currentTimeMillis()}.mp4")
        val outFileOption = VideoCapture.OutputFileOptions.Builder(destVideoFile).build()
        videoCapture.startRecording(
            outFileOption,
            Executors.newSingleThreadExecutor(),
            object : VideoCapture.OnVideoSavedCallback {
                override fun onVideoSaved(outputFileResults: VideoCapture.OutputFileResults) {
                    lifecycleScope.launch(Dispatchers.Main) {
                        Toast.makeText(this@TakeVideoActivity, "拍摄成功", Toast.LENGTH_LONG).show()
                        startActivity(
                            Intent(
                                this@TakeVideoActivity,
                                ShowVideoActivity::class.java
                            ).apply {
                                putExtra("path", destVideoFile.absolutePath)
                            })
                    }
                }
                override fun onError(videoCaptureError: Int, message: String, cause: Throwable?) {
                    lifecycleScope.launch(Dispatchers.Main) {
                        Toast.makeText(this@TakeVideoActivity, "拍摄失败", Toast.LENGTH_LONG).show()
                    }
                }
            })
    }
复制代码


结束拍摄

videoCapture.stopRecording()
复制代码


activity销毁的时候停止相机

@SuppressLint("RestrictedApi")
    override fun onDestroy() {
        super.onDestroy()
        cameraProvider?.shutdown()//必备的一步调用
    }
复制代码


最终效果

image.png


重要api讲解


当我们构建VideoCapture的时候有一些重要的参数设置可以决定最终我们生成视频的结果


setBitRate设置视频的比特率

.setBitRate(1024 * 1024)
复制代码

方法参数的单位是比特,表示我们生成的视频一秒钟的的比特大小

我亲测的结果是CameraX会尽量按照我们设置的比特率来做,如果设置的不合理CameraX会自动为我们调整以保证最终生成的视频的质量


setVideoFrameRate设置视频的帧率

.setVideoFrameRate(100)
复制代码

相当于设置视频的帧率,不过参数的单位是毫秒,比如本例代码我们设置100ms,那么就表示1s内回输出10张视频帧


setTargetRotation设置相机的旋转角度

.setTargetRotation(previdew.display.rotation)
复制代码

这个api没啥好说的,大家都熟悉


setAudioBitRate设置音频的码率

.setAudioBitRate(1024)
复制代码

设置音频的码率,和setBitRate类似


setTargetResolution设置生成的视频的分辨率

.setTargetResolution(Size(300,200))
复制代码

设置最终生成视频的宽高像素比,当我们想极限压缩视频大小的时候这个会非常有用,压缩收益会远大于控制帧率。


切换摄像头的方法

val cameraSelector: CameraSelector = CameraSelector.Builder()
            .requireLensFacing(CameraSelector.LENS_FACING_BACK)//设置后置摄像头,非自拍
//            .requireLensFacing(CameraSelector.LENS_FACING_FRONT)//设置前置摄像头,自拍
            .build()
复制代码

这里必须说一下安卓前置摄像头坑爹的地方:使用前置摄像头拍摄出来的图像是左右颠倒的,使用CameraX因为无法自定义SurfaceView所以我们无法对视频方向进行处理



目录
打赏
0
0
0
0
4
分享
相关文章
python、十六进制的颜色对照表
英文代码  形像颜色  HEX格式  RGB格式 LightPink 浅粉色 #FFB6C1 255,182,193 Pink 粉红 #FFC0CB 255,192,203 Crimson 猩红 #DC143C 220,20,60 LavenderBlush 脸红的淡紫色 #FFF0F5 255.
7885 0
python、十六进制的颜色对照表
Android14 适配之——targetSdkVersion 升级到 34 需要注意些什么?(下)
Android14 适配之——targetSdkVersion 升级到 34 需要注意些什么?(下)
1638 0
【C++入门到精通】智能指针 shared_ptr 简介及C++模拟实现 [ C++入门 ]
【C++入门到精通】智能指针 shared_ptr 简介及C++模拟实现 [ C++入门 ]
251 0
Android S内置APK时AndroidManifest使用uses-library编译报错
Android S内置APK时AndroidManifest使用uses-library编译报错
592 0
kotlin使用spring mvc将接收的字符串生成二维码并响应
使用kotlin制作一个基于`spring mvc`的小demo:接收请求中的字符串参数,用`zxing`将字符串生成出一个二维码,再返回给前端。可以通过此例来了解kotlin与spring的搭配
443 1
梦的解析 —— 周公版,包括【周公解梦】在内的超多免费可用 API 推荐(二)
梦是人们日常生活中经常出现的现象,人们对梦的解析一直是一个重要的研究领域。周公解梦是中国古代传统文化中的一部分,它对于梦的解析提供了很多有价值的参考,成为了梦解释的重要参考工具之一。
411 0
梦的解析 —— 周公版,包括【周公解梦】在内的超多免费可用 API  推荐(二)
AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等

登录插画

登录以查看您的控制台资源

管理云资源
状态一览
快捷访问