google 在Android 5.0推出 Camera2 这个类,用于替换 Camera,但是Camera2要求android sdk 最低版本为 minSdkVersion = 21 (5.0系统),所以Camera2 还不能完全替换 Camera,在兼容低版本的时候,还是需要两者一起协同开发。下面我来说一下 Camera 的拍摄例子:
首先需要在xml 上布局一个 SurfaceView 设置全屏
<SurfaceView
android:id="@+id/surfaceView"
android:layout_width="match_parent"
android:layout_height="match_parent"/>

同时也把状态栏和titleBar隐藏了:
requestWindowFeature(Window.FEATURE_NO_TITLE);// 去掉标题栏
getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN,
WindowManager.LayoutParams.FLAG_FULLSCREEN);// 设置全屏

然后获取SurfacView 实例以及其持有者SurfaceHolder,接入SurfaceHolder.Callback回调,
mSurfaceView = (SurfaceView)findViewById(R.id.surfaceView);
mSurfaceHolder = mSurfaceView.getHolder();// 取得holder
mSurfaceHolder.addCallback(this); // holder加入回调接口
mSurfaceHolder.setKeepScreenOn(true);

SurfaceHolder.Callback会在页面Actvity 初始化完毕后调用,则在回调的surfaceChanged初始化Camera,也就是打开预览页面:
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
if (mCamera != null) {
freeCameraResource();
}try {
mCamera = Camera.open();
if (mCamera == null)
return;
mCamera.setDisplayOrientation(90);
mCamera.setPreviewDisplay(mSurfaceHolder);
parameters = mCamera.getParameters();// 获得相机参数

List<Camera.Size> mSupportedPreviewSizes = parameters.getSupportedPreviewSizes();
List<Camera.Size> mSupportedVideoSizes = parameters.getSupportedVideoSizes();
optimalSize = CameraHelper.getOptimalVideoSize(mSupportedVideoSizes,
mSupportedPreviewSizes, height, width);parameters.setPreviewSize(optimalSize.width, optimalSize.height); // 设置预览图像大小

parameters.set("orientation", "portrait");
List<String> focusModes = parameters.getSupportedFocusModes();
if (focusModes.contains("continuous-video")) {
parameters.setFocusMode(Camera.Parameters.FOCUS_MODE_CONTINUOUS_VIDEO);
}
mFpsRange = parameters.getSupportedPreviewFpsRange();mCamera.setParameters(parameters);// 设置相机参数
mCamera.startPreview();// 开始预览
 }catch (Exception io){
io.printStackTrace();
}
}

该方法返回了SurfaceView的宽与高,根据给出的尺寸与宽高比例,获取一个最适配的预览尺寸,你先看下面有两个参数:
 List<Camera.Size> mSupportedPreviewSizes = parameters.getSupportedPreviewSizes();
List<Camera.Size> mSupportedVideoSizes = parameters.getSupportedVideoSizes();

这两个队列分别是 该相机支持的 预览大小(一般就是拍照时照片的大小),另外一个就是支持适配的大小,因为都是队列,说明相机支持很多组尺寸,而且,照片的尺寸与视频的尺寸是不一样的。我debug看了几款手机,通常摄像支持的尺寸少一点,照片会多一些。这样,我们就要通过刚刚方法给出的宽高,获取一个最佳匹配的预览尺寸:
public static Camera.Size getOptimalVideoSize(List<Camera.Size> supportedVideoSizes,
List<Camera.Size> previewSizes, int w, int h) {
// Use a very small tolerance because we want an exact match.
final double ASPECT_TOLERANCE = 0.1;
double targetRatio = (double) w / h;// Supported video sizes list might be null, it means that we are allowed to use the preview
// sizes
List<Camera.Size> videoSizes;
if (supportedVideoSizes != null) {
videoSizes = supportedVideoSizes;
} else {
videoSizes = previewSizes;
}
Camera.Size optimalSize = null;// Start with max value and refine as we iterate over available video sizes. This is the
// minimum difference between view and camera height.
double minDiff = Double.MAX_VALUE;// Target view height
int targetHeight = h;// Try to find a video size that matches aspect ratio and the target view size.
// Iterate over all available sizes and pick the largest size that can fit in the view and
// still maintain the aspect ratio.
for (Camera.Size size : videoSizes) {
double ratio = (double) size.width / size.height;
if (Math.abs(ratio - targetRatio) > ASPECT_TOLERANCE)
continue;
if (Math.abs(size.height - targetHeight) < minDiff && previewSizes.contains(size)) {
optimalSize = size;
minDiff = Math.abs(size.height - targetHeight);
}
}// Cannot find video size that matches the aspect ratio, ignore the requirement
if (optimalSize == null) {
minDiff = Double.MAX_VALUE;
for (Camera.Size size : videoSizes) {
if (Math.abs(size.height - targetHeight) < minDiff && previewSizes.contains(size)) {
optimalSize = size;
minDiff = Math.abs(size.height - targetHeight);
}
}
}
return optimalSize;
}

该方法是获取最佳的预览与摄像尺寸。然后设置预览图像大小:
parameters.setPreviewSize(optimalSize.width, optimalSize.height);

可以通过获取相机的参数实例,设置里面各种效果,包括刚刚的预览图,前置摄像头,闪光灯等。
parameters = mCamera.getParameters();// 获得相机参数

设置好预览已经相机参数,则打开:
mCamera.setParameters(parameters);// 设置相机参数
mCamera.startPreview();// 开始预览

那么就进入一个预览的拍摄页面了,该页面其实也可以用来做拍照。要想做拍摄,还要实例化MediaRecorder,然后传入camera并初始化相应的参数:
mMediaRecorder.setCamera(mCamera);
mMediaRecorder.setOnErrorListener(this);
mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.DEFAULT );
mMediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);// 视频源// Use the same size for recording profile.
CamcorderProfile mProfile = CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH);
mProfile.videoFrameWidth = optimalSize.width;
mProfile.videoFrameHeight = optimalSize.height;mMediaRecorder.setProfile(mProfile);
//该设置是为了抽取视频的某些帧,真正录视频的时候,不要设置该参数
// mMediaRecorder.setCaptureRate(mFpsRange.get(0)[0]);//获取最小的每一秒录制的帧数

mMediaRecorder.setOutputFile(mVecordFile.getAbsolutePath());mMediaRecorder.prepare();
mMediaRecorder.start();

录完的时候停止,需要重置才能再次使用。
try {
mMediaRecorder.stop();
mMediaRecorder.reset();
} catch (Exception e) {
e.printStackTrace();
}

当页面destroy的时候,要记得释放它们:
 private void releaseRecord() {
if (mMediaRecorder != null) {
mMediaRecorder.setPreviewDisplay(null);
mMediaRecorder.setOnErrorListener(null);
try {
mMediaRecorder.release();
} catch (IllegalStateException e) {
e.printStackTrace();
} catch (Exception e) {
e.printStackTrace();
}
}
mMediaRecorder = null;
}

闪光灯关闭与开启:
private void flashLightToggle(){
try {
if(isFlashLightOn){
parameters.setFlashMode(Camera.Parameters.FLASH_MODE_OFF);
mCamera.setParameters(parameters);
isFlashLightOn = false;
}else {
parameters.setFlashMode(Camera.Parameters.FLASH_MODE_TORCH);
mCamera.setParameters(parameters);
isFlashLightOn = true;
}
} catch (Exception e) {
e.printStackTrace();
}
}

前后摄像头切换,就要重新初始化 camera实例:
private void switchCamera(){
Camera.CameraInfo cameraInfo = new Camera.CameraInfo();
int cameraCount = Camera.getNumberOfCameras();//得到摄像头的个数for(int i = 0; i < cameraCount; i++ ) {
Camera.getCameraInfo(i, cameraInfo);//得到每一个摄像头的信息
if(cameraPosition == 1) {
//现在是后置,变更为前置
if(cameraInfo.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {//代表摄像头的方位,CAMERA_FACING_FRONT前置 CAMERA_FACING_BACK后置
mCamera.stopPreview();//停掉原来摄像头的预览
mCamera.release();//释放资源
mCamera = null;//取消原来摄像头
mCamera = Camera.open(i);//打开当前选中的摄像头
try {
mCamera.setDisplayOrientation(90);
mCamera.setPreviewDisplay(mSurfaceHolder);//通过surfaceview显示取景画面
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
mCamera.setParameters(parameters);// 设置相机参数
mCamera.startPreview();//开始预览
cameraPosition = 0;
break;
}
} else {
//现在是前置, 变更为后置
if(cameraInfo.facing == Camera.CameraInfo.CAMERA_FACING_BACK) {//代表摄像头的方位,CAMERA_FACING_FRONT前置 CAMERA_FACING_BACK后置
mCamera.stopPreview();//停掉原来摄像头的预览
mCamera.release();//释放资源
mCamera = null;//取消原来摄像头
mCamera = Camera.open(i);//打开当前选中的摄像头
try {
mCamera.setDisplayOrientation(90);
mCamera.setPreviewDisplay(mSurfaceHolder);//通过surfaceview显示取景画面
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
mCamera.setParameters(parameters);// 设置相机参数
mCamera.startPreview();//开始预览
cameraPosition = 1;
break;
}
}}
}

录制时的页面:
这就是使用 camera 来摄像的主要步骤,demo:https://github.com/xiaoxiaoqingyi/android-CameraVideo
如果你想了解 Camera2,你也可以看看google 的 Camera2 官方例子:
https://github.com/googlesamples/android-Camera2Basic

转载于:https://www.cnblogs.com/xiaoxiaoqingyi/p/7016312.html

Android Camera 摄像 demo相关推荐

  1. android自定义camera预览区域,android camera摄像surfaceview预览界面特定区域(该区域可移动)...

    1.自定义一个imageview用来设定surfaceview上的特定区域. public class DrawImageView extends ImageView { private Paint ...

  2. android 人脸识别demo,Android Camera 内置人脸识别的Demo

    CameraFace Android Camera 内置人脸识别的Demo 通过Android源生API支持的人脸识别FaceDetection,获取到脸部矩形坐标,左右眼坐标,嘴坐标通过View动态 ...

  3. Android Camera开发(一)之基础知识

    概述 Android手机关于Camera的使用,一是拍照,二是摄像,由于Android提供了强大的组件功能,为此对于在Android手机系统上进行Camera的开发,我们可以使用两类方法:一是借助In ...

  4. Android camera(4)---Android Camera开发之基础知识篇

    Android Camera开发之基础知识篇 转自:https://blog.csdn.net/feiduclear_up/article/details/51968975#jump5 概述 Andr ...

  5. Android Camera 编程从入门到精通

    一.前言 想通过一篇文章就让我们精通 Android 的 Camera 那肯定是不可能的事情.但通过对 Android 中相机拍照的所有的方式的梳理和理解,包括直接调起相机拍照,Camera API ...

  6. Android学习小Demo(10)ToDoList的改进版之ViewPager显示多个图片

    在TodoList增强版的增加界面上,为了显示图片,我是挖了两块地方,放了两个ImageButton,来显示图片,而且限制了最多只能放两张图片.当两个View都放置图片之后,我就会把"Gal ...

  7. Android Camera架构浅析

    原址 1.Camera成像原理介绍 Camera工作流程图 Camera的成像原理可以简单概括如下: 景物(SCENE)通过镜头(LENS)生成的光学图像投射到图像传感器(Sensor)表面上,然后转 ...

  8. Android 自定义相机Demo 入门学习

    Android 自定义相机Demo 本文是参考网上一些自定义相机示例,再结合自己对相机的功能需求写的,基本上包含了很多基本功能,比如相机对焦.闪光灯,以及在手机预览界面上绘制自己想要绘制的图案. 话不 ...

  9. Android USB摄像头插拔监听

    Android USB摄像头插拔监听 一.简要介绍 二.使用广播监听USB摄像头插拔情况 1.注册监听USB设备插拔广播 2.判断UsbDevice是否是Camera设备 (重点) 二.通过UsbMa ...

最新文章

  1. WebGIS空间数据请求访问机制
  2. 这三篇论文开源了!何恺明等人的PointRend,Hinton组的SimCLR和谷歌大脑的EfficientDet...
  3. 原理_JS引擎对未声明变量的处理
  4. Flutter之Align
  5. 枚举对象注释_如何以及何时使用枚举和注释
  6. 通过Maven构建打包Spring boot,并将config配置文件提取到jar文件外
  7. 彭荣新:喜马拉雅自研网关架构演进过程
  8. 使用JAVA实现语音朗读一段文本
  9. Retinex图像增强算法
  10. Unity实现多语言切换
  11. 手把手教你调整电脑磁盘的分区大小
  12. 疑犯追踪第一季/全集Person Of Interest迅雷下载
  13. 使用 yarn 安装 marked
  14. 汉诺塔游戏c语言作业,C语言实现汉诺塔游戏.pdf
  15. 滚轮JAVA_java滚轮
  16. Swift中键盘的弹出隐藏,页面抬高,Return键等的配置
  17. 清华大学出版社与SAP签署战略合作协议
  18. 一个人窝在摇椅里乘凉 我承认这样真的很安详 和楼下老爷爷一样
  19. 转专业2017武汉大学计算机学,转专业申请计算机,武大学子圆梦伯克利
  20. i3-1125G4 怎么样 相当于什么水平

热门文章

  1. Jenkins配置MSBuild时使用环境变量
  2. Kafka Broker常用配置详解
  3. python3 slice
  4. 我自己关于C语言,编译器,标准库,GUN glibc,CRT ,API之类的理解。
  5. butter滤波器是iir吗_如何快速设计应用一个IIR滤波器
  6. 同一工作组无法访问_工作组、AD、域、DC...
  7. 项目管理知识体系指南_PMP考前冲刺丨项目管理知识体系指南(PMBOK)串讲(4)...
  8. spark 把一列数据合并_Spark DataFrame列的合并与拆分
  9. java和python的优势_Java和Python哪个更好?解读Python对比其他语言的优势
  10. Kubernetes 证书过期