Android相机实时自动对焦的完美实现

github,由于android碎片化严重,而且各大厂商极有可能去修改相关API的实现,其中遇到了不少坑,包括实时相机高斯模糊,自动对焦的兼容问题,以及一系列性能问题。换过很多搜索引擎,访问过很多网站,访问过很多网站,拜读过很多代码,没有发现对于相机实时自动对焦特别完美的实现方式。现对相机的自动对焦问题单独做一个记录,算是对这部分的一个总结。也希望后人在这部分能够快速地解决问题,不必浪费过多的时间。测试手机包括:MX4 pro,小米4,华为荣耀3C等等。

参考过@yanzi1225627 大神的一些做法,测试结果不是特别满意。

一,一些对焦方案的尝试

1,autoFocus()的尝试:

private Camera.AutoFocusCallback mAutoFocusCallback;

mAutoFocusCallback = new Camera.AutoFocusCallback() {

public void onAutoFocus(boolean success, Camera camera) {

// TODO Auto-generated method stub

if(success){

myCamera.setOneShotPreviewCallback(null);

Toast.makeText(TestPhotoActivity.this,

"自动聚焦成功" , Toast.LENGTH_SHORT).show();

}

}

};

myCamera.autoFocus(mAutoFocusCallback);

在一部分手机上,始终只对焦一次,也就是说根本不能实现。即使是在按下拍照的时候去调用一次对焦,等对焦成功后再进行拍照,实现的效果也不是很完美。

还见部分博客把autoFocus()方法放在Camera预览SurfaceView的surfaceChanged()中的一些实现,发现也只对焦了一次。

2,设置对焦模式FOCUS_MODE_CONTINUOUS_PICTURE

/**

* Continuous auto focus mode intended for taking pictures. The camera

* continuously tries to focus. The speed of focus change is more

* aggressive than {@link #FOCUS_MODE_CONTINUOUS_VIDEO}. Auto focus

* starts when the parameter is set.

*

*

Applications can call {@link #autoFocus(AutoFocusCallback)} in

* this mode. If the autofocus is in the middle of scanning, the focus

* callback will return when it completes. If the autofocus is not

* scanning, the focus callback will immediately return with a boolean

* that indicates whether the focus is sharp or not. The apps can then

* decide if they want to take a picture immediately or to change the

* focus mode to auto, and run a full autofocus cycle. The focus

* position is locked after autoFocus call. If applications want to

* resume the continuous focus, cancelAutoFocus must be called.

* Restarting the preview will not resume the continuous autofocus. To

* stop continuous focus, applications should change the focus mode to

* other modes.

*

* @see #FOCUS_MODE_CONTINUOUS_VIDEO

*/

public static final String FOCUS_MODE_CONTINUOUS_PICTURE = "continuous-picture";

setFocusMode(Camera.Parameters.FOCUS_MODE_CONTINUOUS_PICTURE);1

在大部分手机上实现了自动对焦,而且效果还不错,而且不需要去额外调用方法。但是测试后发现小米4机型不会自动对焦。

3,设置对焦模式为FOCUS_MODE_CONTINUOUS_VIDEO

/**

* Continuous auto focus mode intended for video recording. The camera

* continuously tries to focus. This is the best choice for video

* recording because the focus changes smoothly . Applications still can

* call {@link #takePicture(Camera.ShutterCallback,

* Camera.PictureCallback, Camera.PictureCallback)} in this mode but the

* subject may not be in focus. Auto focus starts when the parameter is

* set.

*

*

Since API level 14, applications can call {@link

* #autoFocus(AutoFocusCallback)} in this mode. The focus callback will

* immediately return with a boolean that indicates whether the focus is

* sharp or not. The focus position is locked after autoFocus call. If

* applications want to resume the continuous focus, cancelAutoFocus

* must be called. Restarting the preview will not resume the continuous

* autofocus. To stop continuous focus, applications should change the

* focus mode to other modes.

*

* @see #FOCUS_MODE_CONTINUOUS_PICTURE

*/

public static final String FOCUS_MODE_CONTINUOUS_VIDEO = "continuous-video";

setFocusMode(Camera.Parameters.FOCUS_MODE_CONTINUOUS_VIDEO);1

经过测试,发现大部分手机可以连续对焦,但是在对焦过程中屏幕会连续闪烁,而且体验极其不好。魅族MX4不支持此种方式的对焦。也就是说第二,第三种方案都要放弃。

4,触摸对焦

本来在一番焦头烂额后准备妥协,先把触摸对焦实现吧。基本思路是支持定点对焦,就调用定点对焦,否则调用autoFocus()。

/**

* 手动聚焦

*

* @param point 触屏坐标

*/

protected boolean onFocus(Point point, Camera.AutoFocusCallback callback) {

if (mCamera == null) {

return false;

}

Camera.Parameters parameters = null;

try {

parameters = mCamera.getParameters();

} catch (Exception e) {

e.printStackTrace();

return false;

}

//不支持设置自定义聚焦,则使用自动聚焦,返回

if(Build.VERSION.SDK_INT >= 14) {

if (parameters.getMaxNumFocusAreas() <= 0) {

return focus(callback);

}

Log.i(TAG, "onCameraFocus:" + point.x + "," + point.y);

//定点对焦

List areas = new ArrayList();

int left = point.x - 300;

int top = point.y - 300;

int right = point.x + 300;

int bottom = point.y + 300;

left = left < -1000 ? -1000 : left;

top = top < -1000 ? -1000 : top;

right = right > 1000 ? 1000 : right;

bottom = bottom > 1000 ? 1000 : bottom;

areas.add(new Camera.Area(new Rect(left, top, right, bottom), 100));

parameters.setFocusAreas(areas);

try {

//本人使用的小米手机在设置聚焦区域的时候经常会出异常,看日志发现是框架层的字符串转int的时候出错了,

//目测是小米修改了框架层代码导致,在此try掉,对实际聚焦效果没影响

mCamera.setParameters(parameters);

} catch (Exception e) {

// TODO: handle exception

e.printStackTrace();

return false;

}

}

return focus(callback);

}

private boolean focus(Camera.AutoFocusCallback callback) {

try {

mCamera.autoFocus(callback);

} catch (Exception e) {

e.printStackTrace();

return false;

}

return true;

}

二,自动对焦方案

在实现无力的情况下,打开了其他已经实现自定义相机而且能够完美对焦的app,一番操作后,发现很多app都是在我移动手机或者有轻微晃动才进行了第二次对焦,等等,这不就是基于传感器实现的吗?

我们完全可以判断手机的运动状态啊,比如静止和移动。在移动一定时间后去对焦。

核心代码:

package com.jerry.sweetcamera;

import android.app.Activity;

import android.hardware.Sensor;

import android.hardware.SensorEvent;

import android.hardware.SensorEventListener;

import android.hardware.SensorManager;

import android.util.Log;

import java.util.Calendar;

/**

* 加速度控制器 用来控制对焦

*

* @author jerry

* @date 2015-09-25

*/

public class SensorControler implements IActivityLifiCycle, SensorEventListener {

public static final String TAG = "SensorControler";

private SensorManager mSensorManager;

private Sensor mSensor;

private int mX, mY, mZ;

private long lastStaticStamp = 0;

Calendar mCalendar;

boolean isFocusing = false;

boolean canFocusIn = false; //内部是否能够对焦控制机制

boolean canFocus = false;

public static final int DELEY_DURATION = 500;

public static final int STATUS_NONE = 0;

public static final int STATUS_STATIC = 1;

public static final int STATUS_MOVE = 2;

private int STATUE = STATUS_NONE;

private CameraFocusListener mCameraFocusListener;

private static SensorControler mInstance;

private int foucsing = 1; //1 表示没有被锁定 0表示被锁定

private SensorControler() {

mSensorManager = (SensorManager) SweetApplication.CONTEXT.getSystemService(Activity.SENSOR_SERVICE);

mSensor = mSensorManager.getDefaultSensor(Sensor.TYPE_ACCELEROMETER);// TYPE_GRAVITY

}

public static SensorControler getInstance() {

if (mInstance == null) {

mInstance = new SensorControler();

}

return mInstance;

}

public void setCameraFocusListener(CameraFocusListener mCameraFocusListener) {

this.mCameraFocusListener = mCameraFocusListener;

}

@Override

public void onStart() {

restParams();

canFocus = true;

mSensorManager.registerListener(this, mSensor,

SensorManager.SENSOR_DELAY_NORMAL);

}

@Override

public void onStop() {

mSensorManager.unregisterListener(this, mSensor);

canFocus = false;

}

@Override

public void onAccuracyChanged(Sensor sensor, int accuracy) {

}

@Override

public void onSensorChanged(SensorEvent event) {

if (event.sensor == null) {

return;

}

if (isFocusing) {

restParams();

return;

}

if (event.sensor.getType() == Sensor.TYPE_ACCELEROMETER) {

int x = (int) event.values[0];

int y = (int) event.values[1];

int z = (int) event.values[2];

mCalendar = Calendar.getInstance();

long stamp = mCalendar.getTimeInMillis();// 1393844912

int second = mCalendar.get(Calendar.SECOND);// 53

if (STATUE != STATUS_NONE) {

int px = Math.abs(mX - x);

int py = Math.abs(mY - y);

int pz = Math.abs(mZ - z);

// Log.d(TAG, "pX:" + px + " pY:" + py + " pZ:" + pz + " stamp:"

// + stamp + " second:" + second);

double value = Math.sqrt(px * px + py * py + pz * pz);

if (value > 1.4) {

// textviewF.setText("检测手机在移动..");

// Log.i(TAG,"mobile moving");

STATUE = STATUS_MOVE;

} else {

// textviewF.setText("检测手机静止..");

// Log.i(TAG,"mobile static");

//上一次状态是move,记录静态时间点

if (STATUE == STATUS_MOVE) {

lastStaticStamp = stamp;

canFocusIn = true;

}

if (canFocusIn) {

if (stamp - lastStaticStamp > DELEY_DURATION) {

//移动后静止一段时间,可以发生对焦行为

if (!isFocusing) {

canFocusIn = false;

// onCameraFocus();

if (mCameraFocusListener != null) {

mCameraFocusListener.onFocus();

}

// Log.i(TAG,"mobile focusing");

}

}

}

STATUE = STATUS_STATIC;

}

} else {

lastStaticStamp = stamp;

STATUE = STATUS_STATIC;

}

mX = x;

mY = y;

mZ = z;

}

}

private void restParams() {

STATUE = STATUS_NONE;

canFocusIn = false;

mX = 0;

mY = 0;

mZ = 0;

}

/**

* 对焦是否被锁定

*

* @return

*/

public boolean isFocusLocked() {

if(canFocus) {

return foucsing <= 0;

}

return false;

}

/**

* 锁定对焦

*/

public void lockFocus() {

isFocusing = true;

foucsing--;

Log.i(TAG, "lockFocus");

}

/**

* 解锁对焦

*/

public void unlockFocus() {

isFocusing = false;

foucsing++;

Log.i(TAG, "unlockFocus");

}

public void restFoucs() {

foucsing = 1;

}

public interface CameraFocusListener {

void onFocus();

}

}

Start检测手机是否移动对焦几秒后yes

onSensorChanged()来判断手机的运动状态,自动去调用mCameraFocusListener.onFocus();

在mCameraFocusListener中可以调用触摸对焦的方法,这样基本上可以兼容大部分手机的自动对焦功能,而且可以比较好的控制对焦的显示以及对焦区域。foucsing用于对焦的计数,用来锁定对焦。

android自动对焦代码,Android相机实时自动对焦的完美实现相关推荐

  1. Android相机实时自动对焦的完美实现

    Android相机实时自动对焦的完美实现 想写这篇文章很久了,去年十月份接触了相机这一块的知识,由于android碎片化严重,而且各大厂商极有可能去修改相关API的实现,其中遇到了不少坑,包括实时相机 ...

  2. android 共享代码,Android团队成员间的代码共享!(Git+Github版)

    原创,转载请说明出处. 一. 这篇文章要解决什么问题,能学习到什么? 在android团队的开发中,少不了与队员之间的合作.为了避免我们傻瓜式地复制粘贴对方的负责的那部分功能代码,我们可以用svn或者 ...

  3. 名片夹android布局代码,Android自定义布局实现仿qq侧滑部分代码

    自定义布局实现仿qq侧滑部分Android代码,供大家参考,具体内容如下 实现说明: 通过自定义布局实现: SlidingLayout继承于 HorizontalScrollView /** * Cr ...

  4. 获取android型号代码,Android应用开发之Android获取手机品牌、手机型号、手机唯一序列号的代码教程...

    本文将带你了解Android应用开发Android获取手机品牌.手机型号.手机唯一序列号的代码教程,希望本文对大家学Android有所帮助. Android获取手机品牌.手机型号.手机唯一序列号的代码 ...

  5. android gravity 代码,android:layout_gravity和android:gravity属性的区别

    android:gravity / android:layout_gravity区别: android:gravity 是设置该view类里面的内容相对于该view的位置,例如设置button里面的t ...

  6. android扔骰子代码,Android实现掷骰子效果

    本文实例为大家分享了android实现掷骰子效果的具体代码,供大家参考,具体内容如下 利用handler接受子线程的消息完成骰子点数的不断更替 演示 start:开始游戏 stop:停止游戏 reco ...

  7. android滴滴打车代码,Android 端滴滴打车接口的开发

    前两天做了一个对Android端'滴滴打车'的接口开发吧! 今天来做个简单的流程, 滴滴会提供这样一个接口, http://webapp.diditaxi.com.cn/? maptype=wgs&a ...

  8. android linearlayout 代码,Android如何从代码向另一个LinearLayout添加Linearlayout

    我在main.xml中有一个LinearLayout: android:layout_width="match_parent" android:layout_height=&quo ...

  9. android摄像头代码,Android摄像头

    [android]代码库import java.io.BufferedOutputStream; import java.io.File; import java.io.FileOutputStrea ...

  10. android内存代码,Android内存优化(五) Lint代码扫描工具

    1.使用 工具栏 -> Analyze -> Inspect Code- 点击 Inspect Code 后会弹出检查范围的对话框: 默认是检查整个项目,我们可以点击 Custom sco ...

最新文章

  1. Swift使用通知Notification
  2. 操作系统:生产者与消费者问题
  3. Nhibernate 对view 查询的几种方法
  4. hprofile教程
  5. [头脑风暴] 解读Docker Bridge网络模型
  6. Google大佬手把手教你从数据中挖掘价值:好产品是怎样炼成的
  7. java兔子问题 递归_兔子问题 —— 递归的应用
  8. 设计灵感|色彩与形状巧妙运用在创意时尚海报设计中
  9. JPA mappedBy、JoinColumn、FetchType.LAZY和FetchType.EAGER介绍
  10. php解密方法,php加密解密的几种方法的使用教程
  11. Mac PhotoShop CS6破解
  12. 记录一下iter()的用法
  13. Notepad++的64位HexEditor免费下载
  14. js 详解es6 let TDZ(暂时性死区)
  15. [前端之旅] - 01 开端 (持续更新各种资料)(夜·猫之使徒·哮喘征服者·被光选中的人·逐梦)
  16. 做单:第3章 骂人的客户
  17. golang 撤回_golang 连接mongoDB的方法示例
  18. 使用torch.autograd.function解决dist.all_gather不能反向传播问题
  19. 计算机网络个人简历优秀范文,计算机网络个人简历范文
  20. 宝宝如何喝水最科学,做家长的一定要知道

热门文章

  1. pdf 天线理论与技术 钟顺时_钟顺时. 天线理论与技术[M]. 北京:电子工业出版社, 2011: 290-296....
  2. Short But Scary 解题报告
  3. .chm文件打开时显示找不到网页内容
  4. 求有限区间内素数个数
  5. 网络在线播放ASF格式流媒体文件的制作(转)
  6. Openstack日常运维
  7. 牛刀小试-99乘法表
  8. operator开发流程
  9. Win11如何查看硬盘型号?Win11查看硬盘型号步骤
  10. 海湾gst5000协议号_海湾GST5000主机基本操作