使用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所以我们无法对视频方向进行处理



相关文章
|
存储 Linux Android开发
Rockchip系列之VendorStorage uboot/kernel/user space 阶段接口使用介绍(2)
Rockchip系列之VendorStorage uboot/kernel/user space 阶段接口使用介绍(2)
1827 0
|
人工智能 搜索推荐
强大的AI搜索引擎——秘塔AI搜索
【2月更文挑战第17天】强大的AI搜索引擎——秘塔AI搜索
5561 2
强大的AI搜索引擎——秘塔AI搜索
|
存储 Shell 程序员
使用 Python 和 Pygame 制作游戏:第一章到第五章
使用 Python 和 Pygame 制作游戏:第一章到第五章
886 0
|
Android开发 Kotlin 容器
Jetpack-Compose 学习笔记(二)—— Compose 布局你学会了么?(上)
Jetpack-Compose 学习笔记(二)—— Compose 布局你学会了么?(上)
736 1
|
JSON Java 应用服务中间件
使用 Caddy 代替 acme.sh 来为自己的网站颁发免费 SSL 证书
使用 Caddy 代替 acme.sh 来为自己的网站颁发免费 SSL 证书
使用 Caddy 代替 acme.sh 来为自己的网站颁发免费 SSL 证书
|
JSON IDE 开发工具
AAB 扶正!APK 再见!
AAB 扶正!APK 再见!
1010 0
|
Web App开发 JSON 算法
【原创】抖音个人视频列表采集
【原创】抖音个人视频列表采集
1603 0
【原创】抖音个人视频列表采集
|
Web App开发 Java 程序员
全方位测评|M1 这款小小芯片真的全面领跑顶配 i9 Mac 嘛?你想知道的我都告诉你...
大家好,我是小羽。我一直觉得一个东西好不好用,并不是由自己说了算的,也不是别人说了算的,而应该是大多数人用了之后,觉得很不错,那它就是一件好东西。今天小羽除了介绍 M1 芯片的 Mac 的...
1704 0