Android 相机库CameraView源码解析 (二) : 拍照

1. 前言

这段时间,在使用 natario1/CameraView 来实现带滤镜的预览拍照录像功能。
由于CameraView封装的比较到位,在项目前期,的确为我们节省了不少时间。
但随着项目持续深入,对于CameraView的使用进入深水区,逐渐出现满足不了我们需求的情况。
Github中的issues中,有些BUG作者一直没有修复。

那要怎么办呢 ? 项目迫切地需要实现相关功能,只能自己硬着头皮去看它的源码,去解决这些问题。
而这篇文章是其中关于CameraView怎么进行拍照的源码解析。

以下源码解析基于CameraView 2.7.2

implementation("com.otaliastudios:cameraview:2.7.2")

为了在博客上更好的展示,本文贴出的代码进行了部分精简

在这里插入图片描述

拍照的入口是cameraView.takePicture(),我们从这个方法开始解析。

2. CameraEngine.takePicture

cameraView.takePicture()会调用到mCameraEngine.takePicture()
这个PictureResult.Stub是一个参数封装类,这里重新创建了一个PictureResult.Stub并传入takePicture()方法中。
mCameraEngineCameraEngine抽象类,实现类有Camera1EngineCamera2Engine

public void takePicture() {
    PictureResult.Stub stub = new PictureResult.Stub();
    mCameraEngine.takePicture(stub);
}

我们这里以Camera2为例,可以看到这里对stub参数封装类赋值了一些参数(摄像头ID、图片格式等),并调用了onTakePicture

public  void takePicture(final PictureResult.Stub stub) {
    final boolean metering = mPictureMetering;
    getOrchestrator().scheduleStateful("take picture", CameraState.BIND,
            new Runnable() {
		        @Override
		        public void run() {
		            if (isTakingPicture()) return;
		            if (mMode == Mode.VIDEO) {
		                throw new IllegalStateException("Can't take hq pictures while in VIDEO mode");
		            }
		            stub.isSnapshot = false;
		            stub.location = mLocation;
		            stub.facing = mFacing;
		            stub.format = mPictureFormat;
		            onTakePicture(stub, metering);
		        }
    });
}

3. onTakePicture

接着来看onTakePicture()

设置Rotation

stub.rotation = getAngles().offset(Reference.SENSOR, Reference.OUTPUT, Axis.RELATIVE_TO_SENSOR);

设置设定好拍照图片尺寸

 stub.size = getPictureSize(Reference.OUTPUT);

接着调用mPictureRecorder.take()mPictureRecorderPictureRecorder接口,具体实现是Full2PictureRecorder,专门用来调用Camera2 API捕获图片。

mPictureRecorder = new Full2PictureRecorder(stub, this, builder,mPictureReader);
mPictureRecorder.take();

来看一下完整的重点代码

@EngineThread
@Override
protected void onTakePicture(@NonNull final PictureResult.Stub stub, boolean doMetering) {
    //...省略不重要代码...

	//设置Rotation
    stub.rotation = getAngles().offset(Reference.SENSOR, Reference.OUTPUT, Axis.RELATIVE_TO_SENSOR);
    //设置设定好拍照图片尺寸
    stub.size = getPictureSize(Reference.OUTPUT);
    
    //...省略不重要代码...
        
    CaptureRequest.Builder builder = mCamera.createCaptureRequest(CameraDevice.TEMPLATE_STILL_CAPTURE);
    applyAllParameters(builder, mRepeatingRequestBuilder);
    mPictureRecorder = new Full2PictureRecorder(stub, this, builder,
            mPictureReader);
    mPictureRecorder.take();
}

4. Full2PictureRecorder.take

再来看Full2PictureRecorder.take()

@Override
public void take() {
    mAction.start(mHolder);
}

这里调用了mAction.start(mHolder),来看一下mAction初始化

4.1 初始化BaseAction

mAction = new BaseAction() {
    @Override
    protected void onStart(ActionHolder holder) { //省略了代码,这里只看结构 }

    @Override
    public void onCaptureStarted(ActionHolder holder,CaptureRequest request) { //省略了代码,这里只看结构 }

    @Override
    public void onCaptureCompleted(ActionHolder holder,CaptureRequest request,TotalCaptureResult result) { //省略了代码,这里只看结构 }
};
  • mActionBaseAction抽象类,有onStartonCaptureStartedonCaptureProgressedonCaptureCompleted等方法。
  • mHolder是构造方法传入过来的Camera2Engine,实现了ActionHolder接口。

4.2 BaseAction.onStart

调用了mAction.start(mHolder)后,mActionmHolder会建立关联,也就是BaseActionCamera2Engine会建立关联,具体代码为Camera2Engine.addAction(BaseAction)将其添加到Actions列表中,并在合适的时机回调BaseActiononCaptureStartedonCaptureProgressedonCaptureCompleted方法。

mActionmHolder建立关联后,会调用onStart方法,这里是对mPictureBuilder这个建造者设置了一些值

@Override
protected void onStart(@NonNull ActionHolder holder) {
    super.onStart(holder);

	//mPictureBuilder是一个建造者,这里给建造者设置一些值
    mPictureBuilder.addTarget(mPictureReader.getSurface());
    if (mResult.format == PictureFormat.JPEG) {
        mPictureBuilder.set(CaptureRequest.JPEG_ORIENTATION, mResult.rotation);
    }
    mPictureBuilder.setTag(CameraDevice.TEMPLATE_STILL_CAPTURE);
    
    //应用这个建造者
    holder.applyBuilder(this, mPictureBuilder);
}

再来看onCaptureStarted,调用了dispatchOnShutter来回调

@Override
public void onCaptureStarted(@NonNull ActionHolder holder,
                             @NonNull CaptureRequest request) {
    super.onCaptureStarted(holder, request);
    if (request.getTag() == (Integer) CameraDevice.TEMPLATE_STILL_CAPTURE) {
        dispatchOnShutter(false);
        setState(STATE_COMPLETED);
    }
}

4.3 BaseAction.onCaptureCompleted

再来看onCaptureCompleted,主要是在DNG格式的时候,做了一些特殊处理。

@Override
public void onCaptureCompleted(ActionHolder holder, CaptureRequest request, TotalCaptureResult result) {
    if (mResult.format == PictureFormat.DNG) {
        mDngCreator = new DngCreator(holder.getCharacteristics(this), result);
        mDngCreator.setOrientation(ExifHelper.getExifOrientation(mResult.rotation));
        if (mResult.location != null) {
            mDngCreator.setLocation(mResult.location);
        }
    }
}

结果发现这里不是重点,那么重点在哪里呢 ?

5. 设置OnImageAvailableListener监听

Full2PictureRecorder初始化构造方法中,还有这么一句

mPictureReader.setOnImageAvailableListener(this, WorkerHandler.get().getHandler());

AndroidCamera2 API中,setOnImageAvailableListener方法用于注册一个回调监听器,以在每次图像数据可用时接收通知。

5.1 onImageAvailable回调

来看onImageAvailable回调方法,这里会调用android.media.ImageReader.acquireNextImage()来获取图像数据。
然后如果是JPEG格式,则会调用readJpegImage()方法读取图像数据
最后都会调用dispatchResult来分发数据。

@Override
public void onImageAvailable(ImageReader reader) {
    Image image = null;
    try {
        image = reader.acquireNextImage();
        switch (mResult.format) {
            case JPEG: readJpegImage(image); break;
            case DNG: readRawImage(image); break;
            default: throw new IllegalStateException("Unknown format: " + mResult.format);
        }
    } catch (Exception e) {
        mResult = null;
        mError = e;
        dispatchResult();
        return;
    } finally {
        if (image != null) {
            image.close();
        }
    }
    
    dispatchResult();
}

5.2 读取JPEG数据

我们先来看下readJpegImage()方法

private void readJpegImage(@NonNull Image image) {
	//从Iamge中读取数据
    ByteBuffer buffer = image.getPlanes()[0].getBuffer();
    byte[] bytes = new byte[buffer.remaining()];
    buffer.get(bytes);
    mResult.data = bytes;
    
    //根据Exif设置rotation
    mResult.rotation = 0;
    ExifInterface exif = new ExifInterface(new ByteArrayInputStream(mResult.data));
    int exifOrientation = exif.getAttributeInt(ExifInterface.TAG_ORIENTATION,
            ExifInterface.ORIENTATION_NORMAL);
    mResult.rotation = ExifHelper.getOrientation(exifOrientation);
}

5.3 分发回调

再来看dispatchResult,最终会调用到CameraView中的dispatchOnPictureTaken,这个方法中会遍历mListeners回调列表,调用onPictureTaken()

@Override
public void dispatchOnPictureTaken(final PictureResult.Stub stub) {
	mUiHandler.post(new Runnable() {
	    @Override
	    public void run() {
	        PictureResult result = new PictureResult(stub);
	        for (CameraListener listener : mListeners) {
	            listener.onPictureTaken(result);
	        }
	    }
	});
}

mListeners什么时候被添加呢 ? CameraView中有一个addCameraListener方法,专门直接添加回调。

public void addCameraListener(CameraListener cameraListener) {
    mListeners.add(cameraListener);
}

5.4 设置回调

所以我们只要添加了这个回调,并实现onPictureTaken方法,就可以在onPictureTaken()中获取到拍照后的图像信息了。

binding.cameraView.addCameraListener(object : CameraListener() {
    override fun onPictureTaken(result: PictureResult) {
        super.onPictureTaken(result)
        //拍照回调
        val bitmap = BitmapFactory.decodeByteArray(result.data, 0, result.data.size)
        bitmap?.also {
            Toast.makeText(this@Test2Activity, "拍照成功", Toast.LENGTH_SHORT).show()
            //将Bitmap设置到ImageView上
            binding.img.setImageBitmap(it)
            
            val file = getNewImageFile()
            //保存图片到指定目录
            ImageUtils.save(it, file, Bitmap.CompressFormat.JPEG)
        }
    }
})

6. 其他

6.1 CameraView源码解析系列

Android 相机库CameraView源码解析 (一) : 预览-CSDN博客
Android 相机库CameraView源码解析 (二) : 拍照-CSDN博客

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/190152.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

哈希思想的应用

目录 1.位图 位图的实现 题目变形一 题目变形二 题目变形三 总结: 2.布隆过滤器 概念 布隆过滤器的实现 3.哈希切割的思想 1.位图 哈希表和位图是数据结构中常用的两种技术。哈希表是一种数据结构,通过哈希函数把数据和位置进行映射&#xff0c…

公司人事管理系统

1.问题描述 一个小公司包含四类人员:经理,技术人员,销售人员和销售经理,各类人员的工资计算方法如下:经理:固定月薪(8000);技术人员:月薪按技术等级&#xf…

【LeetCode】挑战100天 Day15(热题+面试经典150题)

【LeetCode】挑战100天 Day15(热题面试经典150题) 一、LeetCode介绍二、LeetCode 热题 HOT 100-172.1 题目2.2 题解 三、面试经典 150 题-173.1 题目3.2 题解 一、LeetCode介绍 LeetCode是一个在线编程网站,提供各种算法和数据结构的题目&…

AI视频生成工具——Runway gen2 全功能超详细使用教程(2)

昨天给大家分享了Runway Gen1的使用教程,一篇文章就能让你轻松掌握使用文字和图像从现有视频生成新的视频技能,还没有看过的同学们可以回看过往文章。 Runway视频生成功能有3大核心成品 Gen1:视频转视频工具Gen2:视频生成编辑工…

阅读笔记——《Removing RLHF Protections in GPT-4 via Fine-Tuning》

【参考文献】Zhan Q, Fang R, Bindu R, et al. Removing RLHF Protections in GPT-4 via Fine-Tuning[J]. arXiv preprint arXiv:2311.05553, 2023.【注】本文仅为作者个人学习笔记,如有冒犯,请联系作者删除。 目录 摘要 一、介绍 二、背景 三、方法…

集线器-交换机-路由器

1.集线器(Hub) 集线器就是将网线集中到一起的机器,也就是多台主机和设备的连接器。集线器的主要功能是对接收到的信号进行同步整形放大,以扩大网络的传输距离,是中继器的一种形式,区别在于集线器能够提供多端口服务,也…

Rust UI开发(三):iced如何打开图片(对话框)并在窗口显示图片?

注:此文适合于对rust有一些了解的朋友 iced是一个跨平台的GUI库,用于为rust语言程序构建UI界面。 这是一个系列博文,本文是第三篇,前两篇的链接: 1、Rust UI开发(一):使用iced构建…

2023年09月 Scratch(三级)真题解析#中国电子学会#全国青少年软件编程等级考试

Scratch等级考试(1~4级)全部真题・点这里 一、单选题(共25题,每题2分,共50分) 第1题 运行下面程序后,角色的x坐标值是?( ) A:100 B:90 C:110 D:120 答案:C 利用变量值作为条件,控制循环的次数。变量从0~10的过程中每次角色的x坐标都增加了10,当变量值为1…

人力资源管理后台 === 左树右表

1.角色管理-编辑角色-进入行内编辑 获取数据之后针对每个数据定义标识-使用$set-代码位置(src/views/role/index.vue) // 针对每一行数据添加一个编辑标记this.list.forEach(item > {// item.isEdit false // 添加一个属性 初始值为false// 数据响应式的问题 数据变化 视图…

牛客 算法 HJ103 Redraiment的走法 golang语言实现

题目 HJ103 Redraiment的走法 实现 package mainimport ("bufio""fmt""os""strconv""strings" )func main() {scanner : bufio.NewScanner(os.Stdin)nums : make([]int, 0)nums_len:0dp:make([]int, 0)for scanner.Scan()…

汇编实验2-2 查找匹配字符串笔记

一、数据段 1.字符串结尾:13,10,$ 2.设置格式控制字符串(这样就不用再写clrf函数了) 3.设置存关键字和句子的地址标签,以关键字为例 二、代码段 1.输入字符串 2.字符串比较 2.1 每次的比较长度,KLEN->CL 2.2 设置目标串起始…

java学习part12多态

99-面向对象(进阶)-面向对象的特征三:多态性_哔哩哔哩_bilibili 1.多态(仅限方法) 父类引用指向子类对象。 调用重写的方法,就会执行子类重写的方法。 编译看引用表面类型,执行看实际变量类型。 2.父子同名属性是否…

游览器缓存讲解

浏览器缓存是指浏览器在本地存储已经请求过的资源的一种机制,以便在将来的请求中能够更快地获取这些资源,减少对服务器的请求,提高页面加载速度。浏览器缓存主要涉及到两个方面:缓存控制和缓存位置。 缓存控制 Expires 头&#…

力扣每日一题-统计和小于目标的下标对数目-2023.11.24

力扣每日一题:统计和小于目标的下标对数目 开篇 今天这道力扣打卡题写得我好狼狈,一开始思路有点问题,后面就是对自己的代码到处缝缝补补,最后蒙混过关。只能分享一下大佬的代码,然后我帮大家分享代码的思路。 题目链…

84基于matlab的数字图像处理

基于matlab的数字图像处理,数据可更换自己的,程序已调通,可直接运行。 84matlab数字图像处理图像增强 (xiaohongshu.com)https://www.xiaohongshu.com/explore/656219d80000000032034dea

python+pytest接口自动化(1)-接口测试基础

一般我们所说的接口即API,那什么又是API呢,百度给的定义如下: API(Application Programming Interface,应用程序接口)是一些预先定义的接口(如函数、HTTP接口),或指软件系…

【数据库基础】

目录: 前言什么是数据库主流数据库服务器,数据库,表关系MySQL架构SQL分类存储引擎 前言 剑指offer:一年又1天 什么是数据库 存储数据用文件就可以了,为什么还要弄个数据库? 文件保存数据有以下几个缺点:…

数据结构之时间复杂度与空间复杂度

1.算法效率 1.1 如何衡量一个算法的好坏&#xff1f; 比方说我们非常熟悉的斐波拉契数列&#xff1a; long long Fib(int N) {if(N < 3)return 1;return Fib(N-1) Fib(N-2); } 递归实现方式非常简洁&#xff0c;但一定好吗&#xff1f;如何衡量其好与坏&#xff1f; 1…

ES6之class类

ES6提供了更接近传统语言的写法&#xff0c;引入了Class类这个概念&#xff0c;作为对象的模板。通过Class关键字&#xff0c;可以定义类&#xff0c;基本上&#xff0c;ES6的class可以看作只是一个语法糖&#xff0c;它的绝大部分功能&#xff0c;ES5都可以做到&#xff0c;新…

AndroidStudio2022.3.1 Patch3使用国内下载源加速

记录一下这个版本的as在使用国内下载源加速碰到的诸多问题。 一、gradle-8.0-bin.zip下载慢 编辑项目文件夹/gradle/wrapper/gradle-wrapper.properties&#xff0c;文件内容改为如下&#xff1a; #Fri Nov 24 18:50:06 CST 2023 distributionBaseGRADLE_USER_HOME distribu…