Camera.java

  • autoFocus()聚焦回调函数
@Override
public void autoFocus() {
//记录当前聚焦开始时间
mFocusStartTime = System.currentTimeMillis();
//设置Camera的回调聚焦
mCameraDevice.autoFocus(mAutoFocusCallback);
//设置Camera的状态为Focusing
setCameraState(FOCUSING);
}
  • 设置相机状态
private void setCameraState(int state) {
mCameraState = state;
switch (state) {
case SNAPSHOT_IN_PROGRESS:
case FOCUSING:
enableCameraControls(false);
break;
case IDLE:
case PREVIEW_STOPPED:
enableCameraControls(true);
break;
}
}
  • enableCameraControls,设置enable,是否可以点击
/**
* 设置几个button或者view不可点击
* @param enable
*/
private void enableCameraControls(boolean enable) {
if (mIndicatorControlContainer != null) {
mIndicatorControlContainer.setEnabled(enable);
}
if (mModePicker != null) mModePicker.setEnabled(enable);
if (mZoomControl != null) mZoomControl.setEnabled(enable);
if (mThumbnailView != null) mThumbnailView.setEnabled(enable);
}
  • mIndicatorControlContainer的enable的设置
Override
public void setEnabled(boolean enabled) {
super.setEnabled(enabled);
final int count = getChildCount();
for (int i = 0; i < count; i++) {
View v = getChildAt(i);
// Zoom buttons and shutter button are controlled by the activity.
if (v instanceof AbstractIndicatorButton) {
v.setEnabled(enabled);
// Show or hide the indicator buttons during recording.
if (mCurrentMode == MODE_VIDEO) {
v.setVisibility(enabled ? View.VISIBLE : View.INVISIBLE);
}
}
}
if (mCameraPicker != null) {
mCameraPicker.setEnabled(enabled);
if (mCurrentMode == MODE_VIDEO) {
mCameraPicker.setVisibility(enabled ? View.VISIBLE : View.INVISIBLE);
}
}
}
  • mAutoFocusCallback中的处理
private final class AutoFocusCallback
implements android.hardware.Camera.AutoFocusCallback {
public void onAutoFocus(
boolean focused, android.hardware.Camera camera) {
//如果是暂停状态,不聚焦
if (mPausing) return;
//算出当前到聚焦开始的时间差
mAutoFocusTime = System.currentTimeMillis() - mFocusStartTime;
Log.v(TAG, "mAutoFocusTime = " + mAutoFocusTime + "ms");
//设置Camera状态为
setCameraState(IDLE);
//调用FocusManager中的AutoFocus
mFocusManager.onAutoFocus(focused);
}
}

FocusManager.java

/**
* 聚焦
* @param focused
*/
public void onAutoFocus(boolean focused) {
Log.i(TAG, "focus used : " + (System.currentTimeMillis() - focusStart));
//正在进行聚焦,拍照动作必须再聚焦完之后
if (mState == STATE_FOCUSING_SNAP_ON_FINISH) {
// 无论聚焦成功还是失败,都会拍照。如果要进行拍照发声,就无需AF发声了
if (focused) {
//聚焦成功
mState = STATE_SUCCESS;
} else {
//聚焦失败
mState = STATE_FAIL;
}
//更新聚焦框UI&&设置人脸识别UI已经各种状态的设置
updateFocusUI();
//拍照,mState的状态变为STATE_IDLE
capture();
} else if (mState == STATE_FOCUSING) {//如果是聚焦中的状态
//此状态的发生分为两种,half-pressing按压聚焦或者触摸聚焦被触发,这个时候不要发生拍照动作
if (focused) {
//聚焦成功
mState = STATE_SUCCESS;
//在连续聚焦状态不要发声,聚焦回调会在拍照前完成,所有状态一直为STATE_FOCUSING
if (!Parameters.FOCUS_MODE_CONTINUOUS_PICTURE.
equals(mFocusMode)) {
mListener.playSound(CameraSound.FOCUS_COMPLETE);
}
} else {
//聚焦失败
mState = STATE_FAIL;
}
//更新聚焦&&人脸UI
updateFocusUI();
// If this is triggered by touch focus, cancel focus after a
// while.
//如果是触摸聚焦,需要延迟一下取消掉聚焦
if (mFocusArea != null) {
mHandler.sendEmptyMessageDelayed(RESET_TOUCH_FOCUS, RESET_TOUCH_FOCUS_DELAY);
}
} else if (mState == STATE_IDLE) {//空闲
// User has released the focus key before focus completes.
// Do nothing.
}
}
  • 处理消息
private class MainHandler extends Handler {
@Override
public void handleMessage(Message msg) {
switch (msg.what) {
case RESET_TOUCH_FOCUS: {
//取消掉聚焦
cancelAutoFocus();
//开始人脸识别
mListener.startFaceDetection();
break;
}
}
}
}
/**
* 再重置tap area之前调用mListener.cancelAutofocus,否则,聚焦模式将一直是自动&tap聚焦,并且驱动也不会重置
*/
private void cancelAutoFocus() {
//放置聚焦框到屏幕中间
resetTouchFocus();
mListener.cancelAutoFocus();
if (mFaceView != null) mFaceView.resume();
mState = STATE_IDLE;
updateFocusUI();
mHandler.removeMessages(RESET_TOUCH_FOCUS);
}
  • 触摸聚焦
/**
* 触摸,,这里会发生触摸聚焦
* @param e
* @return
*/
public boolean onTouch(MotionEvent e) {
//没有初始化或者拍照前的聚焦的状态,直接返回
if (!mInitialized || mState == STATE_FOCUSING_SNAP_ON_FINISH) return false; //让用户可以取消掉之前未消失的触摸聚焦
if ((mFocusArea != null) && (mState == STATE_FOCUSING ||
mState == STATE_SUCCESS || mState == STATE_FAIL)) {
cancelAutoFocus();
} // Initialize variables.
int x = Math.round(e.getX());
int y = Math.round(e.getY());
int focusWidth = mFocusIndicatorRotateLayout.getWidth();
int focusHeight = mFocusIndicatorRotateLayout.getHeight();
int previewWidth = mPreviewFrame.getWidth();
int previewHeight = mPreviewFrame.getHeight();
if (mFocusArea == null) {
mFocusArea = new ArrayList<Area>();
mFocusArea.add(new Area(new Rect(), 1));
mMeteringArea = new ArrayList<Area>();
mMeteringArea.add(new Area(new Rect(), 1));
} //将坐标转换为驱动的格式。AE面积更大,因为曝光会敏感和容易,或者说曝光不足,如果面积太小了。
calculateTapArea(focusWidth, focusHeight, 1f, x, y, previewWidth, previewHeight,
mFocusArea.get(0).rect);
calculateTapArea(focusWidth, focusHeight, 1.5f, x, y, previewWidth, previewHeight,
mMeteringArea.get(0).rect); // Use margin to set the focus indicator to the touched area.
RelativeLayout.LayoutParams p =
(RelativeLayout.LayoutParams) mFocusIndicatorRotateLayout.getLayoutParams();
int left = Util.clamp(x - focusWidth / 2, 0, previewWidth - focusWidth);
int top = Util.clamp(y - focusHeight / 2, 0, previewHeight - focusHeight);
p.setMargins(left, top, 0, 0);
// Disable "center" rule because we no longer want to put it in the center.
int[] rules = p.getRules();
rules[RelativeLayout.CENTER_IN_PARENT] = 0;
mFocusIndicatorRotateLayout.requestLayout(); //停止人脸识别,因为要进行识别聚焦和测量area
mListener.stopFaceDetection(); //设置聚焦区域&测量区域
mListener.setFocusParameters();
//如果支持触摸聚焦&&手指抬起
if (mFocusAreaSupported && (e.getAction() == MotionEvent.ACTION_UP)) {
autoFocus();
} else { // Just show the indicator in all other cases.
updateFocusUI();
// Reset the metering area in 3 seconds.
mHandler.removeMessages(RESET_TOUCH_FOCUS);
mHandler.sendEmptyMessageDelayed(RESET_TOUCH_FOCUS, RESET_TOUCH_FOCUS_DELAY);
} return true;
}

聚焦各个状态能做什么事不能做什么事&切换

  • STATE_IDLE
doSnap():空闲状态可以执行拍照
onAutoFocus(boolean focused):用户再聚焦完成前放掉了聚焦按钮,所以不做任何事。
onPreviewStarted(),onPreviewStopped():状态都变为空闲
cancelAutoFocus():取消聚焦,状态变为空闲
updateFocusUI():如果空闲&有聚焦区域,显示聚焦框框
  • STATE_FOCUSING
onShutterUp():如果是focusmode为自动聚焦,状态为正在聚焦,则取消掉聚焦
doSnap():如果是正在聚焦状态,将状态改为聚焦完拍照状态
onAutoFocus(boolean focused):判断focused,为true变为聚焦成功状态,为false变为聚焦失败状态
onTouch(MotionEvent e):如果正在聚焦&之前有手动聚焦了,则取消掉之前的聚焦
autoFocus():状态变为聚焦状态
updateFocusUI():显示聚焦的框框
  • STATE_FOCUSING_SNAP_ON_FINISH
onShutterUp():状态不是聚焦完拍照,可以设置FocusParameters
onAutoFocus(boolean focused):判断focused,为true变为聚焦成功状态,为false变为聚焦失败状态,更新聚焦框UI
onTouch(MotionEvent e):直接不继续操作下去
updateFocusUI():显示聚焦的框框
  • STATE_SUCCESS
onShutterDown():如果是自动聚焦mode,并且不为聚焦成功状态,执行聚焦
onShutterUp(): 如果是自动聚焦mode,并且为聚焦成功状态,执行取消聚焦
doSnap():拍照
onTouch(MotionEvent e):让用户可以取消掉之前未消失的触摸聚焦
updateFocusUI():聚焦框框显示成功
  • STATE_FAIL
onShutterDown():如果是自动聚焦mode,并且不为聚焦失败状态,执行聚焦
onShutterUp(): 如果是自动聚焦mode,并且为聚焦失败状态,执行取消聚焦
doSnap():拍照
onTouch(MotionEvent e):让用户可以取消掉之前未消失的触摸聚焦
updateFocusUI():聚焦框框显示失败

Android — Camera聚焦流程的更多相关文章

  1. Android -- Camera聚焦流程

    Camera.java autoFocus()聚焦回调函数 @Override public void autoFocus() { //记录当前聚焦开始时间 mFocusStartTime = Sys ...

  2. MTK Android Camera运行流程

    Android Camera 运行流程 总体架构1.CameraService服务的注册2.Client端的应用层到JNI层Camera App-JNI3.Client到Service的连接4.HAL ...

  3. 高通Android camera运行流程【转】

    本文转载自:http://blog.csdn.net/unicornkylin/article/details/13293295 1.总体架构 Android Camera 框架从整体上看是一个 cl ...

  4. Android Camera 调用流程总结

    1.总体介绍  Android Camera框架从整体上看是一个client/service架构.有两个进程,一个是client进程,可以看成AP端,主要包括Java代码和一些native层的c/c+ ...

  5. android Camera 数据流程分析

    这篇文章主要针对其数据流程进行分析.Camera一般用于图像浏览.拍照和视频录制.这里先对图像浏览和拍照的数据流进行分析,后面再对视频电话部分进行分析. 1.针对HAL层对摄像头数据处理补充一下 Li ...

  6. Android Camera 流程梳理

    毕业已经快两年了,一直没有写博客的习惯,这是第一篇,以后要慢慢养成这个习惯.毕业之后一直在做相机,先简单的梳理下Android Camera的流程. Android Camera 是一个client/ ...

  7. Android -- Camera源码简析,启动流程

    com.android.camera.Camera.java,主要的实现Activity,继承于ActivityBase. ActivityBase 在ActivityBase中执行流程: onCre ...

  8. 【Android】Android Camera原始帧格式转换 —— 获取Camera图像(一)

     概述: 做过Android Camera图像采集和处理的朋友们应该都知道,Android手机相机采集的原始帧(RawFrame)默认是横屏格式的,而官方API有没有提供一个设置Camera采集图像的 ...

  9. 玩转Android Camera开发(一):Surfaceview预览Camera,基础拍照功能完整demo

    杂家前文是在2012年的除夕之夜仓促完成,后来很多人指出了一些问题,琐事缠身一直没有进行升级.后来随着我自己的使用,越来越发现不出个升级版的demo是不行了.有时候就连我自己用这个demo测一些性能. ...

随机推荐

  1. 安卓开发第一步:Android Studio安装配置

    虽然本人是JAVA开发工程师平时主要开发Web App,但因为项目需求需要开发对应的移动端.一时又找不到合适的安卓开发人员,兄弟我只好被项目经理"抓来当壮丁了".俗话说好" ...

  2. 示例详解:UIScrollview 与 Autolayout 的那点事

    前言 自从写了介绍Masonry那篇文章以后 就一直有人对UIScrollView的那个例子不是很理解 UIView *container = [UIView new]; [scrollView ad ...

  3. My97DatePicker时间控件使用

    刚刚工作中遇到一个修改时间空间的bug,顺带学习了My97DatePicker时间空间 网上查到的资料已经很详细: http://www.360doc.com/content/14/0606/11/1 ...

  4. Symantec Backup Exec备份作业服务器盘符变更

    Symantec Backup Exec的备份作业中,如果某个服务器的磁盘更改了盘符,如果不修改备份作业里面的相关配置,就会出现类似下面的错误信息,如下截图所示 因为这台服务器上我们将原先的G盘的盘符 ...

  5. hibernate 实现分页查询语句、单条查询语句、多条查询语句、修改、删除语句

    package com.hanqi.test; import java.util.Date; import java.util.List; import org.hibernate.Query; im ...

  6. ARM 汇编寻址方式

    ARM支持9种寻址方式:立即数寻址,寄存器寻址,寄存器偏移寻址,寄存器间接寻址,基址变址寻址,多寄存器寻址,相对寻址,堆栈寻址,块拷贝寻址. 立即数寻址 将数据直接存放的指令中发给CPU,首先由于AR ...

  7. LINUX运维实战案例之文件已删除但空间不释放问题的分析与解决办法

    1.错误现象 运维的监控系统发来通知,报告一台服务器空间满了,登陆服务器查看,根分区确实没有空间了,如下图所示: 这里首先说明一下服务器的一些删除策略,由于Linux没有回收站功能,我们的线上服务器所 ...

  8. 构建 ARM Linux 4.7.3 嵌入式开发环境 —— BusyBox 构建 RootFS

    上一篇我们已经成功将 ARM Linux 4.7.3 的内核利用 U-BOOT 引导了起来.但是细心的你会发现,引导到后面,系统无法启动,出现内核恐慌 (Kernel Panic). 原因是找不到文件 ...

  9. [译] OpenStack Kilo 版本中 Neutron 的新变化

    OpenStack Kilo 版本,OpenStack 这个开源项目的第11个版本,已经于2015年4月正式发布了.现在是个合适的时间来看看这个版本中Neutron到底发生了哪些变化了,以及引入了哪些 ...

  10. ListView的基础应用

    在写完基础的布局之后,下一课我们会学习一下如何使用Android中一个非常重要,但是对于新手略有困难的ListView,甚至很久以前都有人说过,会不会写ListView是Android能否入门的第一步 ...