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



相关文章
|
Python
python、十六进制的颜色对照表
英文代码  形像颜色  HEX格式  RGB格式 LightPink 浅粉色 #FFB6C1 255,182,193 Pink 粉红 #FFC0CB 255,192,203 Crimson 猩红 #DC143C 220,20,60 LavenderBlush 脸红的淡紫色 #FFF0F5 255.
8503 0
python、十六进制的颜色对照表
|
10月前
|
网络协议 前端开发 Java
Coolbpf最新特性解读:profiler功能上线,助力性能分析和优化
本文介绍Coolbpf性能分析模块中的profiler功能。
|
存储 Java 调度
FileInputStream,FileOutputStream 和 FileReader ,FileWriter 类的基本使用【 File类+IO流知识回顾②】
这篇文章回顾了Java中FileInputStream、FileOutputStream、FileReader和FileWriter类的基本使用方法,包括读取和写入文件的操作,以及字符流和字节流的区别和应用场景。
FileInputStream,FileOutputStream 和 FileReader ,FileWriter 类的基本使用【 File类+IO流知识回顾②】
|
存储 Kubernetes API
使用Kubernetes管理容器化应用的深度解析
【5月更文挑战第20天】本文深度解析Kubernetes在管理容器化应用中的作用。Kubernetes是一个开源平台,用于自动化部署、扩展和管理容器,提供API对象描述应用资源并维持其期望状态。核心组件包括负责集群控制的Master节点(含API Server、Scheduler、Controller Manager和Etcd)和运行Pod的工作节点Node(含Kubelet、Kube-Proxy和容器运行时环境)。
|
编解码 测试技术 Android开发
Android经典实战之用 CameraX 库实现高质量的照片和视频拍摄功能
本文详细介绍了如何利用CameraX库实现高质量的照片及视频拍摄功能,包括添加依赖、初始化、权限请求、配置预览与捕获等关键步骤。此外,还特别针对不同分辨率和帧率的视频拍摄提供了性能优化策略,确保应用既高效又稳定。
1499 1
Android经典实战之用 CameraX 库实现高质量的照片和视频拍摄功能
|
人工智能 搜索推荐
强大的AI搜索引擎——秘塔AI搜索
【2月更文挑战第17天】强大的AI搜索引擎——秘塔AI搜索
5024 2
强大的AI搜索引擎——秘塔AI搜索
|
存储 Shell 程序员
使用 Python 和 Pygame 制作游戏:第一章到第五章
使用 Python 和 Pygame 制作游戏:第一章到第五章
603 0
|
安全 算法 数据安全/隐私保护
【C++入门到精通】智能指针 shared_ptr 简介及C++模拟实现 [ C++入门 ]
【C++入门到精通】智能指针 shared_ptr 简介及C++模拟实现 [ C++入门 ]
491 0
|
Java Android开发
Android 12 自定义底部导航栏
Android 12 自定义底部导航栏
508 4
解决:java.lang.UnsatisfiedLinkError: dalvik.system.PathClassLoader[DexPathList[[zip file
解决:java.lang.UnsatisfiedLinkError: dalvik.system.PathClassLoader[DexPathList[[zip file