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() {@Overridepublic 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) {//...省略不重要代码...//设置Rotationstub.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() {@Overrideprotected void onStart(ActionHolder holder) { //省略了代码,这里只看结构 }@Overridepublic void onCaptureStarted(ActionHolder holder,CaptureRequest request) { //省略了代码,这里只看结构 }@Overridepublic 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设置rotationmResult.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() {@Overridepublic 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博客

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

WordPress无需插件禁用WP生成1536×1536和2048×2048尺寸图片

我们在使用WordPress上传图片媒体文件的时候,是不是看到媒体库中有15361536和20482048的图片文件,当然这么大的文件会占用我们的服务器空间,如何禁止掉呢? function remove_default_image_sizes( $sizes) {unset( $sizes[1536x15…

spring-webmvc练习-日程管理-访问后端展示列表数据

1、util/request.js import axios from "axios";let request axios.create({baseURL: "http://localhost:8080",timeout: 50000 });export default request 2、api/schedule.js import request from "../util/request.js";export let getSchedu…

[架构之路-253]:目标系统 - 设计方法 - 软件工程 - 软件设计 - 结构化设计的主要评估指标:高内聚(模块内部)、低耦合(模块之间)的含义

目录 前言: 一、软件工程中的软件设计种类:根据宏观到微观分 (1)软件架构设计(层次划分、模块划分、职责分工): (2)软件高层设计、概要设计(功能模块的接…

【c++随笔14】虚函数表

【c随笔14】虚函数表 一、虚函数表(Virtual Function Table)1、定义2、查看虚函数表2.1、 问题:三种类型,包含一个int类型的class、一个int类型的变量、int类型的指针:这三个大小分别是多少呢?2.2、怎么发现…

IT问题解答类型网站源码

问答网是一款为IT工程师提供的问答平台,旨在帮助用户在线获取专业知识和相关问题的答案。在问答网,用户可以轻松找到其他人的问答问题,并在这里寻求解答。如果您有任何想要解决的问题,都可以在此发布问题并得到其他同行的解答。 …

CSS之弹性盒子Flexible Box

我想大家在做布局的时候,没接触flex布局之前,大家都是用浮动来布局的,但现在我们接触了flex布局之后,我只能说:“真香”。让我为大家介绍一下弹性盒子模型吧! Flexible Box 弹性盒子 在我们使用弹性盒子时&…

【算法】链表-20231127

这里写目录标题 一、面试题 02.02. 返回倒数第 k 个节点二、82. 删除排序链表中的重复元素 II三、141. 环形链表 一、面试题 02.02. 返回倒数第 k 个节点 提示 简单 130 相关企业 实现一种算法,找出单向链表中倒数第 k 个节点。返回该节点的值。 注意:本…

Linux(8):BASH

硬件、核心与 Shell 操作系统其实是一组软件,由于这组软件在控制整个硬件与管理系统的活动监测,如果这组软件能被用户随意的操作,若使用者应用不当,将会使得整个系统崩溃。因为操作系统管理的就是整个硬件功能。 应用程序在最外层…

前端(HTML + CSS + JS)

文章目录 一、HTML1. 概念(1)HTML 文件基本结构(2)HTML代码框架 2. 、HTML常见标签 二、CSS1. CSS基本语法规范2. 用法(1) 引用方式(2)选择器(3)常用元素属性…

NX二次开发UF_CURVE_ask_trim 函数介绍

文章作者:里海 来源网站:https://blog.csdn.net/WangPaiFeiXingYuan UF_CURVE_ask_trim Defined in: uf_curve.h int UF_CURVE_ask_trim(tag_t trim_feature, UF_CURVE_trim_p_t trim_info ) overview 概述 Retrieve the current parameters of an a…

利用STM32和MFRC522 IC实现智能卡的读取和数据存储

利用STM32微控制器和MFRC522 RFID读写器芯片,可以实现智能卡的读取和数据存储功能。智能卡是一种集成了RFID技术和存储芯片的卡片,它可以用于身份验证、门禁控制、支付系统等应用场景。下面将介绍如何使用STM32和MFRC522芯片进行智能卡的读取和数据存储&…

3.OpenResty系列之Nginx反向代理

1. Nginx简介 Nginx (engine x) 是一款轻量级的 Web 服务器 、反向代理服务器及电子邮件(IMAP/POP3)代理服务器 什么是反向代理? 反向代理(Reverse Proxy)方式是指以代理服务器来接受 internet 上的连接请求&#x…

4面试题--数据库(补充)

隔离性问题 若不考虑隔离性则会出现以下问题 1. 脏读:指⼀个事务在处理数据的过程中,读取到另⼀个 未提交 事务的数据 2. 不可重复读:指对于数据库中的某个数据(同⼀个数据项),⼀个事务内的多次查询却…

docker打包前端镜像

文章目录 一、构建镜像二、查看本地镜像三、启动容器四、查看启动的容器五、保存镜像六、读取镜像七、创建镜像八、最后 docker官网 一、构建镜像 -t是给镜像命名,.(点)是基于当前目录的Dockerfile来构建镜像 docker build -t image_web .二、查看本地镜像 docke…

使用echars实现数据可视化

生活随笔 展翅飞翔之际 请下定决心不再回头 echars实现数据可视化 在搭建后台页面时,可能会遇到很多的表格,但有时表格所展现的数据并不能直观的体现出当前用户的宏观信息,所以就可以引入一个新的表格插件——echars 快速上手 - Handbook…

某软件商店app抓包分析与sign加密算法实现

文章目录 1. 写在前面2. 抓包配置3. 抓包分析4. 接口测试5. sign加密算法6. 数据效果展示 【作者主页】:吴秋霖 【作者介绍】:Python领域优质创作者、阿里云博客专家、华为云享专家。长期致力于Python与爬虫领域研究与开发工作! 【作者推荐】…

通用电气调查网络攻击和数据盗窃指控

通用电气正在调查有关威胁行为者在网络攻击中破坏了公司开发环境并泄露据称被盗数据的指控。 通用电气 (GE) 是一家美国跨国公司,业务涉及电力、可再生能源和航空航天行业。 本月早些时候,一个名为 IntelBroker 的威胁行为者试图在黑客论坛上以 500 美…

人工智能_机器学习051_支持向量机SVM概念介绍_理解support vector machine---人工智能工作笔记0091

在出现深度学习,神经网络算法之前,支持向量机已经可以解决很多问题了,我们自然界中的问题,无非就是可以转换为回归问题和分类问题. 然后从现在开始我们来看支持向量机,首先看一下这几个字 support 是支持 vector是向量的意思,然后 machine指的是机器 那么我们之前用到的模型…

常见树种(贵州省):021冬青、连香树、白辛树、香合欢、云贵鹅耳枥、肥牛树、杜英、格木、黄连木、圆果化香树、南天竹

摘要:本专栏树种介绍图片来源于PPBC中国植物图像库(下附网址),本文整理仅做交流学习使用,同时便于查找,如有侵权请联系删除。 图片网址:PPBC中国植物图像库——最大的植物分类图片库 一、冬青 …

AIGC ChatGPT 4 快速整理不规则数据

从业务系统中采集到的数据如下: 序号 省份 英文 2022年销售额 2021年销售额 增量 1 广东guangDOng129068.58 124319.67 4748.91 2 江苏 JiangSu 122825.6 116314.2 6511.4 3 山东ShAnDong 87385 83045.9 4339.1 4 浙江…