Camera.java

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

FocusManager.java

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

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

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

我是天王盖地虎的分割线

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

  1. Android — Camera聚焦流程

    原文  http://www.cnphp6.com/archives/65098 主题 Android Camera.java autoFocus()聚焦回调函数 @Override public v ...

  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. 使用gdb调试

    启用gdb进行调试二进制程序,必须在二进制程序在采用gcc或g++编译时加入-g参数 启动gdb进行调试的几种形式: 直接启动gdb程序进行调试program程序 gdb program 启动gdb挂 ...

  2. 机器学习之路:python 特征降维 主成分分析 PCA

    主成分分析: 降低特征维度的方法. 不会抛弃某一列特征, 而是利用线性代数的计算,将某一维度特征投影到其他维度上去, 尽量小的损失被投影的维度特征 api使用: estimator = PCA(n_c ...

  3. Linux驱动程序中的并发控制

    <临界区> a:对共享资源进行访问的代码称为临界区.   <原子操作>     a:原子操作用于执行轻量级,仅仅执行一次的的操作比如修改计数器,有条件的增加值,设置某一位.所谓 ...

  4. 数据预处理:标准化(Standardization)

    注:本文是人工智能研究网的学习笔记 常用的数据预处理方式 Standardization, or mean removal and variance scaling Normalization: sc ...

  5. JavaScript 继承和数组

    前言 因为篇幅比较短,所以将JavaScript中的继承和数组进行统一写. 继承 当一个函数对象被创建的时候,Function构造器产生的函数对象会运行类似这样的代码: this.prototype ...

  6. PyQt QString 与 Python str&unicode

    昨日,将许久以前做的模拟网页登录脚本用PyQt封装了一下,结果出大问题了, 登录无数次都提示登录失败!!而不用PyQt实现的GUI登录直接脚本登录无数次都提示登录成功!!心中甚是伤痛,于是探究起来,解 ...

  7. Git_配置别名

    有没有经常敲错命令?比如git status?status这个单词真心不好记. 如果敲git st就表示git status那就简单多了,当然这种偷懒的办法我们是极力赞成的. 我们只需要敲一行命令,告 ...

  8. JSON在PHP中的基本应用

    从5.2版本开始,PHP原生提供json_encode()和json_decode()函数,前者用于编码,后者用于解码. 一.json_encode() 该函数主要用来将数组和对象,转换为json格式 ...

  9. iOS学习之WebView的使用 (主要是下面的全屏半透明实现)

    1.使用UIWebView加载网页 运行XCode 4.3,新建一个Single View Application,命名为WebViewDemo. 2.加载WebView 在ViewControlle ...

  10. MySQLAdmin的用法

    mysqladmin 适合于linux和windows系统 linux下:mysqladmin -u[username] -p[password] status windows下:先在安装目录找到my ...