Android Camera2开启电子防抖(EIS)和光学防抖(OIS)

2023-11-30 21:12

本文主要是介绍Android Camera2开启电子防抖(EIS)和光学防抖(OIS),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

刚好当前项目有录像功能,使用了第三方框架是基于Camera2引擎开发,当使用 Camera2 API 开发相机应用时,启用和关闭 EIS(电子防抖)是一个重要的功能。EIS 可以帮助减少相机拍摄时的抖动,从而提高图像和视频的稳定性;OIS是光学图片想稳定器,功能是通过镜头的浮动透镜来纠正“光轴偏移”,需要硬件的支持,两者同时开启在录像时效果更佳。

EIS原理:

使用CCD偏移来实现防抖,通过降低图像失真和抖动,从而提高图像质量,首先把CCD安置在一个可以上下左右移动的支架上,然后当陀螺传感器检测到抖动的时候,就会把抖动的方向、速度和移动量等参数经过处理,计算出足以抵消抖动的CCD移动量,简单说就是通过放大录像焦距,根据陀螺仪旋转时预留周边的视角通过进行补帧移动画面给人感觉没有那么抖动,但效果没有光学防抖的好。

OIS原理:

其原理是通过镜头内的陀螺仪侦测到微小的移动,然后将信号传至微处理器,处理器立即计算需要补偿的位移量,然后通过补偿镜片组,根据镜头的抖动方向及位移量加以补偿。

对比开启EIS和OIS的摄像对比

eis开

eis关

1. 检查设备支持的特性

在使用 Camera2 API 开启 EIS 之前,我们需要首先检查设备是否支持 EIS 特性。我们可以通过 CameraCharacteristics 类来获取相机设备的特性信息。

 CameraManager manager = (CameraManager) getSystemService(Context.CAMERA_SERVICE);CameraCharacteristics characteristics = manager.getCameraCharacteristics(mCameraDevice.getId());boolean isModeHaveStabilization = false; // 支持电子防抖final int[] availableVideoStabilization = characteristics.get(CameraCharacteristics.CONTROL_AVAILABLE_VIDEO_STABILIZATION_MODES);if (availableVideoStabilization == null || availableVideoStabilization.length == 0) {Log.e(TAG, "Optical stabilization not available");return;}for (int mode : availableVideoStabilization) {if (mode == CaptureRequest.CONTROL_VIDEO_STABILIZATION_MODE_ON) {isModeHaveStabilization = true;Log.d(TAG, "Using video stabilization.");break;}}int[] stabilizationModes = characteristics.get(CameraCharacteristics.LENS_INFO_AVAILABLE_OPTICAL_STABILIZATION);if (stabilizationModes == null || stabilizationModes.length == 0) {Log.e(TAG, "Optical stabilization not available");return;}for (int mode : stabilizationModes) {if (mode == CaptureRequest.LENS_OPTICAL_STABILIZATION_MODE_ON) {isModeSupported = true;   // 支持光学防抖break;}}
2. 启用 EIS和关闭EIS

记得判断是否支持EIS

// 开启EIS
mPreviewBuilder.set(CaptureRequest.CONTROL_VIDEO_STABILIZATION_MODE, CaptureRequest.CONTROL_VIDEO_STABILIZATION_MODE_ON);//关闭EIS
mPreviewBuilder.set(CaptureRequest.CONTROL_VIDEO_STABILIZATION_MODE, CaptureRequest.CONTROL_VIDEO_STABILIZATION_MODE_OFF);
3. 启用OIS和关闭OIS
 captureRequestBuilder.set(CaptureRequest.LENS_OPTICAL_STABILIZATION_MODE,mEisEnabled ? CaptureRequest.LENS_OPTICAL_STABILIZATION_MODE_ON :CaptureRequest.LENS_OPTICAL_STABILIZATION_MODE_OFF);

在创建预览请求的时候注意一定要设置成录制模式,预览模式开启不了EIS,设置无效

mPreviewBuilder = mCameraDevice.createCaptureRequest(CameraDevice.TEMPLATE_PREVIEW);

下面时完整代码片段,希望可以帮助到有需要的人

public class Camera2Activity extends AppCompatActivity {private static final String TAG = "Camera2Example";private TextureView mTextureView;private CameraDevice mCameraDevice;private CaptureRequest.Builder mPreviewBuilder;private CameraCaptureSession mCaptureSession;// private SurfaceTexture mSurfaceTexture;private Button captureModel;//private Camera2Helper mCamera2Helper;boolean mEisEnabled = true;private Handler backgroundHandler;private HandlerThread backgroundThread;@SuppressLint("MissingInflatedId")@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_camera);initView();}private void initView() {captureModel = (Button) findViewById(R.id.captureModel);mTextureView = findViewById(R.id.textureView);mCaptureButton = findViewById(R.id.captureButton);if (ContextCompat.checkSelfPermission(this, Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.CAMERA, Manifest.permission.RECORD_AUDIO, Manifest.permission.CAMERA}, 200);return;}findViewById(R.id.eisopen).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {enableEis();}});findViewById(R.id.eisclose).setOnClickListener(v -> {disableEis();});captureModel.setOnClickListener(v -> {switchMode();});}private TextureView.SurfaceTextureListener textureListener = new TextureView.SurfaceTextureListener() {@Overridepublic void onSurfaceTextureAvailable(SurfaceTexture surface, int width, int height) {openCamera();}@Overridepublic void onSurfaceTextureSizeChanged(SurfaceTexture surface, int width, int height) {}@Overridepublic boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {return false;}@Overridepublic void onSurfaceTextureUpdated(SurfaceTexture surface) {}};//打开相机public void openCamera() {CameraManager cameraManager = (CameraManager) getSystemService(Context.CAMERA_SERVICE);try {String cameraId = cameraManager.getCameraIdList()[0];if (ActivityCompat.checkSelfPermission(this, Manifest.permission.CAMERA) != PackageManager.PERMISSION_GRANTED) {// 请求权限ActivityCompat.requestPermissions(this, new String[]{Manifest.permission.CAMERA}, 200);return;}cameraManager.openCamera(cameraId, new CameraDevice.StateCallback() {@Overridepublic void onOpened(@NonNull CameraDevice camera) {mCameraDevice = camera;createCameraPreviewSession();}@Overridepublic void onDisconnected(@NonNull CameraDevice camera) {mCameraDevice.close();mCameraDevice = null;}@Overridepublic void onError(@NonNull CameraDevice camera, int error) {mCameraDevice.close();mCameraDevice = null;}}, null);} catch (CameraAccessException e) {e.printStackTrace();}}// 创建CameraCaptureSessionprivate void createCameraPreviewSession() {try {SurfaceTexture texture = mTextureView.getSurfaceTexture();// 设置预览尺寸texture.setDefaultBufferSize(mTextureView.getWidth(), mTextureView.getHeight());assert texture != null;// 创建预览SurfaceSurface surface = new Surface(texture);// 创建CaptureRequest.BuildermPreviewBuilder = mCameraDevice.createCaptureRequest(mCurrentMode);// 设置自动对焦模式mPreviewBuilder.set(CaptureRequest.CONTROL_AF_MODE, CaptureRequest.CONTROL_AF_MODE_CONTINUOUS_PICTURE);// 设置自动曝光模式mPreviewBuilder.set(CaptureRequest.CONTROL_AE_MODE, CaptureRequest.CONTROL_AE_MODE_ON_AUTO_FLASH);mPreviewBuilder.addTarget(surface);toggleStabilization(mPreviewBuilder);// 创建CameraCaptureSessionmCameraDevice.createCaptureSession(Collections.singletonList(surface),new CameraCaptureSession.StateCallback() {@Overridepublic void onConfigured(@NonNull CameraCaptureSession cameraCaptureSession) {// 相机已经关闭if (null == mCameraDevice) {return;}// 会话准备好后,我们开始显示预览mCaptureSession = cameraCaptureSession;updatePreview();}@Overridepublic void onConfigureFailed(@NonNull CameraCaptureSession cameraCaptureSession) {//   showToast("Failed");}}, backgroundHandler);} catch (CameraAccessException e) {e.printStackTrace();}}int mCurrentMode = CameraDevice.TEMPLATE_PREVIEW;// 在点击按钮时切换模式private void switchMode() {if (mCurrentMode == CameraDevice.TEMPLATE_PREVIEW) {// 切换到录制模式mCurrentMode = CameraDevice.TEMPLATE_RECORD;createCameraPreviewSession();captureModel.setText("切换到预览模式");} else {// 切换到预览模式mPreviewBuilder.set(CaptureRequest.CONTROL_AF_TRIGGER, CameraMetadata.CONTROL_AF_TRIGGER_CANCEL);mCurrentMode = CameraDevice.TEMPLATE_PREVIEW;createCameraPreviewSession();captureModel.setText("切换到录制模式");}}// 电子防抖和光学防抖private void toggleStabilization(CaptureRequest.Builder captureRequestBuilder) {if (mPreviewBuilder == null) {return;}try {CameraManager manager = (CameraManager) getSystemService(Context.CAMERA_SERVICE);CameraCharacteristics characteristics = manager.getCameraCharacteristics(mCameraDevice.getId());boolean isModeHaveStabilization = false; // 支持电子防抖boolean isModeSupported = false; // 支持光学防抖final int[] availableVideoStabilization = characteristics.get(CameraCharacteristics.CONTROL_AVAILABLE_VIDEO_STABILIZATION_MODES);if (availableVideoStabilization == null || availableVideoStabilization.length == 0) {Log.e(TAG, "Optical stabilization not available");return;}for (int mode : availableVideoStabilization) {if (mode == CaptureRequest.CONTROL_VIDEO_STABILIZATION_MODE_ON) {isModeHaveStabilization = true;Log.d(TAG, "Using video stabilization.");break;}}// 开启EIS电子防抖功能if (mEisEnabled && isModeHaveStabilization) {mPreviewBuilder.set(CaptureRequest.CONTROL_VIDEO_STABILIZATION_MODE, CaptureRequest.CONTROL_VIDEO_STABILIZATION_MODE_ON);mPreviewBuilder.set(CaptureRequest.CONTROL_MODE, CameraMetadata.CONTROL_MODE_USE_SCENE_MODE);mPreviewBuilder.set(CaptureRequest.CONTROL_SCENE_MODE, CameraMetadata.CONTROL_SCENE_MODE_ACTION); // hrd 动作场景模式} else {mPreviewBuilder.set(CaptureRequest.CONTROL_VIDEO_STABILIZATION_MODE, CaptureRequest.CONTROL_VIDEO_STABILIZATION_MODE_OFF);mPreviewBuilder.set(CaptureRequest.CONTROL_MODE, CameraMetadata.CONTROL_MODE_USE_SCENE_MODE);mPreviewBuilder.set(CaptureRequest.CONTROL_SCENE_MODE, CameraMetadata.CONTROL_SCENE_MODE_ACTION); // hrd  动作场景模式}int[] stabilizationModes = characteristics.get(CameraCharacteristics.LENS_INFO_AVAILABLE_OPTICAL_STABILIZATION);if (stabilizationModes == null || stabilizationModes.length == 0) {Log.e(TAG, "Optical stabilization not available");return;}for (int mode : stabilizationModes) {if (mode == CaptureRequest.LENS_OPTICAL_STABILIZATION_MODE_ON) {isModeSupported = true;break;}}// 开启OIS光学防抖功能if (isModeSupported) {captureRequestBuilder.set(CaptureRequest.LENS_OPTICAL_STABILIZATION_MODE,mEisEnabled ? CaptureRequest.LENS_OPTICAL_STABILIZATION_MODE_ON :CaptureRequest.LENS_OPTICAL_STABILIZATION_MODE_OFF);}} catch (CameraAccessException e) {e.printStackTrace();}}// 更新预览private void updatePreview() {if (null == mCameraDevice) {return;}try {mCaptureSession.setRepeatingRequest(mPreviewBuilder.build(), null, backgroundHandler);} catch (CameraAccessException e) {e.printStackTrace();}}// 开启EIS电子防抖功能private void enableEis() {mEisEnabled = true;toggleStabilization(mPreviewBuilder);updatePreview();}// 关闭EIS电子防抖功能private void disableEis() {mEisEnabled = false;toggleStabilization(mPreviewBuilder);updatePreview();}@Overrideprotected void onResume() {super.onResume();startBackgroundThread();if (mTextureView.isAvailable()) {openCamera();} else {mTextureView.setSurfaceTextureListener(textureListener);}}@Overrideprotected void onPause() {super.onPause();stopBackgroundThread();closeCamera();}private void closeCamera() {if (mCaptureSession != null) {mCaptureSession.close();mCaptureSession = null;}if (mCameraDevice != null) {mCameraDevice.close();mCameraDevice = null;}}private void startBackgroundThread() {backgroundThread = new HandlerThread("CameraBackground");backgroundThread.start();backgroundHandler = new Handler(backgroundThread.getLooper());}private void stopBackgroundThread() {backgroundThread.quitSafely();try {backgroundThread.join();backgroundThread = null;backgroundHandler = null;} catch (InterruptedException e) {e.printStackTrace();}}
}

这篇关于Android Camera2开启电子防抖(EIS)和光学防抖(OIS)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/438571

相关文章

Android与iOS设备MAC地址生成原理及Java实现详解

《Android与iOS设备MAC地址生成原理及Java实现详解》在无线网络通信中,MAC(MediaAccessControl)地址是设备的唯一网络标识符,本文主要介绍了Android与iOS设备M... 目录引言1. MAC地址基础1.1 MAC地址的组成1.2 MAC地址的分类2. android与I

Android 实现一个隐私弹窗功能

《Android实现一个隐私弹窗功能》:本文主要介绍Android实现一个隐私弹窗功能,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 效果图如下:1. 设置同意、退出、点击用户协议、点击隐私协议的函数参数2. 《用户协议》、《隐私政策》设置成可点击的,且颜色要区分出来res/l

Android实现一键录屏功能(附源码)

《Android实现一键录屏功能(附源码)》在Android5.0及以上版本,系统提供了MediaProjectionAPI,允许应用在用户授权下录制屏幕内容并输出到视频文件,所以本文将基于此实现一个... 目录一、项目介绍二、相关技术与原理三、系统权限与用户授权四、项目架构与流程五、环境配置与依赖六、完整

Android 12解决push framework.jar无法开机的方法小结

《Android12解决pushframework.jar无法开机的方法小结》:本文主要介绍在Android12中解决pushframework.jar无法开机的方法,包括编译指令、框架层和s... 目录1. android 编译指令1.1 framework层的编译指令1.2 替换framework.ja

Android开发环境配置避坑指南

《Android开发环境配置避坑指南》本文主要介绍了Android开发环境配置过程中遇到的问题及解决方案,包括VPN注意事项、工具版本统一、Gerrit邮箱配置、Git拉取和提交代码、MergevsR... 目录网络环境:VPN 注意事项工具版本统一:android Studio & JDKGerrit的邮

Android实现定时任务的几种方式汇总(附源码)

《Android实现定时任务的几种方式汇总(附源码)》在Android应用中,定时任务(ScheduledTask)的需求几乎无处不在:从定时刷新数据、定时备份、定时推送通知,到夜间静默下载、循环执行... 目录一、项目介绍1. 背景与意义二、相关基础知识与系统约束三、方案一:Handler.postDel

Android使用ImageView.ScaleType实现图片的缩放与裁剪功能

《Android使用ImageView.ScaleType实现图片的缩放与裁剪功能》ImageView是最常用的控件之一,它用于展示各种类型的图片,为了能够根据需求调整图片的显示效果,Android提... 目录什么是 ImageView.ScaleType?FIT_XYFIT_STARTFIT_CENTE

Android实现在线预览office文档的示例详解

《Android实现在线预览office文档的示例详解》在移动端展示在线Office文档(如Word、Excel、PPT)是一项常见需求,这篇文章为大家重点介绍了两种方案的实现方法,希望对大家有一定的... 目录一、项目概述二、相关技术知识三、实现思路3.1 方案一:WebView + Office Onl

springboot项目如何开启https服务

《springboot项目如何开启https服务》:本文主要介绍springboot项目如何开启https服务方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录springboot项目开启https服务1. 生成SSL证书密钥库使用keytool生成自签名证书将

Android实现两台手机屏幕共享和远程控制功能

《Android实现两台手机屏幕共享和远程控制功能》在远程协助、在线教学、技术支持等多种场景下,实时获得另一部移动设备的屏幕画面,并对其进行操作,具有极高的应用价值,本项目旨在实现两台Android手... 目录一、项目概述二、相关知识2.1 MediaProjection API2.2 Socket 网络