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

news2024/11/28 20:54:14

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博客

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

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

相关文章

2006-2023年2月地级市城投债数据

2006-2023年2月地级市城投债数据 1、时间:2006-2023年2月 2、指标:省份、城市、证券代码、证券简称、债券简称、证券全称、债券初始面值单位元、债券最新面值交易日期20221231、发行总额单位亿元、债券余额日期20221231单位亿、起息日期、计息截止日、…

2018年10月9日 Go生态洞察:Go Cloud的Wire与编译时依赖注入

🌷🍁 博主猫头虎(🐅🐾)带您 Go to New World✨🍁 🦄 博客首页——🐅🐾猫头虎的博客🎐 🐳 《面试题大全专栏》 🦕 文章图文…

火柴棒等式

枚举 只要在保证等式正确的基础上判断火柴棒有没有用完就可以 因为数比较小,而且我不知道最大的等式中的数是多少,索性就设置为999了 还好对效率要求不大(doge) 要不然就得自己慢慢改最大数来试了 代码如下: #in…

树套树 (线段树+splay)

树套树,就是线段树、平衡树、树状数组等数据结构的嵌套。 最简单的是线段树套set,可以解决一些比较简单的问题,而且代码根线段树是一样的只是一些细节不太一样。 本题中用的是线段树套splay,代码较长。 树套树中的splay和单一的…

基于springboot实现农机电招平台系统项目【项目源码+论文说明】

基于springboot实现农机电招平台系统演示 摘要 随着农机电招行业的不断发展,农机电招在现实生活中的使用和普及,农机电招行业成为近年内出现的一个新行业,并且能够成为大群众广为认可和接受的行为和选择。设计农机电招平台的目的就是借助计算…

C++ day37 贪心算法 单调递增的数字 监控二叉树

题目1&#xff1a;738 单调递增的数字 题目链接&#xff1a;单调递增的数字 对题目的理解 返回小于或等于n的最大数字&#xff0c;且数字是单调递增&#xff08;单调递增数字的定义&#xff1a;每个相邻位上的数字满足x<y&#xff09; 贪心算法 注意本题的遍历顺序是从…

写 SVG 动画必看!SVG系列文章1-简介

1、SVG是什么 SVG 是一种 XML 语言&#xff0c;类似 XHTML&#xff0c;可以用来绘制矢量图形&#xff0c;例如下面展示的图形。SVG 可以通过定义必要的线和形状来创建一个图形&#xff0c;也可以修改已有的位图&#xff0c;或者将这两种方式结合起来创建图形。图形和其组成部分…

Vatee万腾科技的未来探险:Vatee数字创新的独特发现

在科技的浩瀚海洋中&#xff0c;Vatee万腾科技如一艘探险船般&#xff0c;勇敢地驶向未知的数字化领域。这次未来的探险&#xff0c;不仅是一场科技创新的冒险&#xff0c;更是对数字化时代的独特发现和深刻探讨。 Vatee万腾科技视科技创新为一座高峰&#xff0c;而他们的未来探…

探究Kafka原理-5.Kafka设计原理和生产者原理解析

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱吃芝士的土豆倪&#xff0c;24届校招生Java选手&#xff0c;很高兴认识大家&#x1f4d5;系列专栏&#xff1a;Spring源码、JUC源码、Kafka原理&#x1f525;如果感觉博主的文章还不错的话&#xff0c;请&#x1f44…

SATA信息传输FIS结构总结通过实例代码(快速)掌握(二)

目录 一、简介二、总体介绍2.1 详细FIS传输过程2.2 FIS内容详解 三、FIS实例3.1 构造一个write FIS3.1.1 WRITE FIS内容3.1.2 协议分析仪trace分析3.1.3 write过程总trace 3.2 构造一个read FIS3.2.1 READ FIS内容3.2.2 协议分析仪 read trace3.2.3 read过程总trace 3.3 FIS和C…

【自制】我与GPT3.5合作,一天制作一个生成手写文字图片的软件

视频讲解地址&#xff1a;https://www.bilibili.com/video/BV1uC4y127ME/ bgm我与GPT3.5合作&#xff0c;一天制作一个生成手写文字图片的软件 请给出一个程序&#xff0c;左边显示一个图片&#xff0c;将图片放入进去&#xff0c;可以在里面画出一个框&#xff0c; 右边窗口…

MQTT客户端MQTT.fx 1.7.1下载、安装和界面介绍

MQTT.fx是一款基于Eclipse Paho&#xff0c;使用Java语言编写的MQTT客户端工具。支持通过Topic订阅和发布消息&#xff0c;用来前期和物理云平台调试非常方便。 1.下载 1.1.访问官方下载地址下载&#xff0c;但是下载不到1.7.1版本 1.2.在连接网页末尾点击立即下载&#xff0c;…

Linux(9):正规表示法与文件格式化处理

简单的说&#xff0c;正规表示法就是处理字符串的方法&#xff0c;他是以行为单位来进行字符串的处理行为&#xff0c;正规表示法透过一些特殊符号的辅助&#xff0c;可以让使用者轻易的达到【搜寻/删除/取代】某特定字符串的处理程序。 正规表示法基本上是一种【表示法】&…

【从入门到起飞】JavaSE—多线程(2)(lock锁,死锁,等待唤醒机制)

文章目录 &#x1f33a;lock锁⭐获得锁⭐释放锁✨注意&#x1f3f3;️‍&#x1f308;代码实现&#x1f388;细节 &#x1f33a;死锁⭐解决方法 &#x1f384;等待唤醒机制⭐代码实现&#x1f388;注意 &#x1f6f8;使用阻塞队列实现等待唤醒机制 &#x1f354;线程的六种状态…

AOP + 自定义注解实现日志打印

1. 先定义个注解&#xff0c;让它作用于方法上 Target({ElementType.METHOD}) Retention(RetentionPolicy.RUNTIME) public interface Loggable {}2. 定义切面 Aspect Component Slf4j public class LogMethodCallAspect {Pointcut("annotation(com.wy.spring_demo.aop.…

1panel在应用商店里面安装jenkins

文章目录 目录 文章目录 前言 一、使用步骤 1.1 填写安装参数 1.2 在界面中进入容器拿到自动生成的jenkins密码 前言 一、使用步骤 1.1 填写安装参数 在应用商店里面搜索jenkins,然后点击安装 填写参数 1.2 在界面中进入容器拿到自动生成的jenkins密码 命令 cat /var/jenki…

【数据结构与算法篇】一文详解数据结构之二叉树

树的介绍及二叉树的C实现 树的概念相关术语树的表示 树的概念 树是一种非线性的数据结构&#xff0c;它是由n&#xff08;n>0&#xff09;个有限结点组成一 个具有层次关系的集合。把它叫做树是因为它看起来像一棵倒挂的树&#xff0c; 也就是说它是根朝上&#xff0c;而叶朝…

上海站报名启动! 2023年开源产业生态大会OpenHarmony生态分论坛

作为年内开源领域不容错过的科技盛宴&#xff0c;2023年开源产业生态大会将于12月19日在上海盛大开幕。本次活动由上海市经济和信息化委员会、上海市科学技术协会和"科创中国"开源创新联合体共同指导&#xff0c;上海开源信息技术协会统筹主办。 届时&#xff0c;大会…

WEB渗透—反序列化(六)

Web渗透—反序列化 课程学习分享&#xff08;课程非本人制作&#xff0c;仅提供学习分享&#xff09; 靶场下载地址&#xff1a;GitHub - mcc0624/php_ser_Class: php反序列化靶场课程&#xff0c;基于课程制作的靶场 课程地址&#xff1a;PHP反序列化漏洞学习_哔哩哔_…

黑马程序员索引学习笔记

文章目录 索引的分类从索引字段特性从物理存储从数据结构组成索引的字段个数 InnoDB主键索的Btree高度为多高呢?explain执行计划最左匹配原则索引失效情况SQL提示覆盖索引、回表查询前缀索引索引设计原则 索引的分类 从索引字段特性 主键索引、唯一索引、常规索引、全文索引…