Android -- Camera聚焦流程

简介:

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():聚焦框框显示失败

我是天王盖地虎的分割线       




本文转自我爱物联网博客园博客,原文链接:http://www.cnblogs.com/yydcdut/p/4133862.html,如需转载请自行联系原作者

相关文章
|
1天前
|
缓存 前端开发 Android开发
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
【04】flutter补打包流程的签名过程-APP安卓调试配置-结构化项目目录-完善注册相关页面-开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程
|
5天前
|
Dart 前端开发 Android开发
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
【02】写一个注册页面以及配置打包选项打包安卓apk测试—开发完整的社交APP-前端客户端开发+数据联调|以优雅草商业项目为例做开发-flutter开发-全流程-商业应用级实战开发-优雅草央千澈
|
14天前
|
测试技术 Android开发 开发者
【03】优雅草央千澈详解关于APP签名以及分发-上架完整流程-第三篇安卓APP上架华为商店后面的步骤-华为应用商店相对比较麻烦一些-华为商店安卓上架
【03】优雅草央千澈详解关于APP签名以及分发-上架完整流程-第三篇安卓APP上架华为商店后面的步骤-华为应用商店相对比较麻烦一些-华为商店安卓上架
|
3月前
|
Android开发
Android面试之Activity启动流程简述
Android面试之Activity启动流程简述
105 6
|
3月前
|
XML 前端开发 Android开发
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
|
3月前
|
消息中间件 Android开发 索引
Android面试高频知识点(4) 详解Activity的启动流程
Android面试高频知识点(4) 详解Activity的启动流程
37 3
|
3月前
|
XML 前端开发 Android开发
Android面试高频知识点(3) 详解Android View的绘制流程
Android面试高频知识点(3) 详解Android View的绘制流程
34 2
|
3月前
|
XML 前端开发 Android开发
Android View的绘制流程和原理详细解说
Android View的绘制流程和原理详细解说
64 3
|
3月前
|
Android开发
Android面试之Activity启动流程简述
Android面试之Activity启动流程简述
26 0
|
4月前
|
Android开发 开发者
Android面试之Activity启动流程简述
每个Android开发者都熟悉的Activity,但你是否了解它的启动流程呢?本文将带你深入了解。启动流程涉及四个关键角色:Launcher进程、SystemServer的AMS、应用程序的ActivityThread及Zygote进程。核心在于AMS与ActivityThread间的通信。文章详细解析了从Launcher启动Activity的过程,包括通过AIDL获取AMS、Zygote进程启动以及ActivityThread与AMS的通信机制。接着介绍了如何创建Application及Activity的具体步骤。整体流程清晰明了,帮助你更深入理解Activity的工作原理。
77 0