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



相关文章
|
缓存
对YTKNetwork的官方demo的一些补漏
对YTKNetwork的官方demo的一些补漏
269 1
|
Python
python、十六进制的颜色对照表
英文代码  形像颜色  HEX格式  RGB格式 LightPink 浅粉色 #FFB6C1 255,182,193 Pink 粉红 #FFC0CB 255,192,203 Crimson 猩红 #DC143C 220,20,60 LavenderBlush 脸红的淡紫色 #FFF0F5 255.
8177 0
python、十六进制的颜色对照表
|
编解码 测试技术 Android开发
Android经典实战之用 CameraX 库实现高质量的照片和视频拍摄功能
本文详细介绍了如何利用CameraX库实现高质量的照片及视频拍摄功能,包括添加依赖、初始化、权限请求、配置预览与捕获等关键步骤。此外,还特别针对不同分辨率和帧率的视频拍摄提供了性能优化策略,确保应用既高效又稳定。
1371 1
Android经典实战之用 CameraX 库实现高质量的照片和视频拍摄功能
|
8月前
|
网络协议 前端开发 Java
Coolbpf最新特性解读:profiler功能上线,助力性能分析和优化
本文介绍Coolbpf性能分析模块中的profiler功能。
|
存储 Java 调度
FileInputStream,FileOutputStream 和 FileReader ,FileWriter 类的基本使用【 File类+IO流知识回顾②】
这篇文章回顾了Java中FileInputStream、FileOutputStream、FileReader和FileWriter类的基本使用方法,包括读取和写入文件的操作,以及字符流和字节流的区别和应用场景。
FileInputStream,FileOutputStream 和 FileReader ,FileWriter 类的基本使用【 File类+IO流知识回顾②】
|
移动开发 JavaScript 编译器
Vue3 系列教程 — Vue 3 简介
Vue3 系列教程 — Vue 3 简介
|
Java Android开发
Android 12 自定义底部导航栏
Android 12 自定义底部导航栏
462 4
|
人工智能 搜索推荐
强大的AI搜索引擎——秘塔AI搜索
【2月更文挑战第17天】强大的AI搜索引擎——秘塔AI搜索
4825 2
强大的AI搜索引擎——秘塔AI搜索
|
安全 算法 数据安全/隐私保护
【C++入门到精通】智能指针 shared_ptr 简介及C++模拟实现 [ C++入门 ]
【C++入门到精通】智能指针 shared_ptr 简介及C++模拟实现 [ C++入门 ]
376 0
|
Web App开发 JavaScript 前端开发
Vue3.0 Composition API和Hooks使用学习文档
Vue3.0 Composition API和Hooks使用学习文档
7660 0