`

Android 自定义Camera

阅读更多

Android 自定义Camera

 

mCamera.setDisplayOrientation(90);是一个假像,当存储之后所看到的图片其实还是倾斜90度

源码中已解决图片旋转的问题

此案例还有一问题,希望高手看到后能够解决:在部分手机使用中会出现花屏的现象(例:华为U8860)。



 

 

Android Camera 使用小结

Android手机关于Camera的使用,一是拍照,二是摄像,由于Android提供了强大的组件功能,为此对于在Android手机系统上进行Camera的开发,我们可以使用两类方法:一是借助Intent和MediaStroe调用系统Camera App程序来实现拍照和摄像功能,二是根据Camera API自写Camera程序。由于自写Camera需要对Camera API了解很充分,而且对于通用的拍照和摄像应用只需要借助系统Camera App程序就能满足要求了,为此先从调用系统Camera App应用开始来对Android Camera做个简单的使用小结。

调用系统Camera App实现拍照和摄像功能

不是专门的Camera应用,一般用到Camera的需求就是获取照片或者视频,比如微博分享、随手记等,对于在Symbian系统上通过简单地调用系统自带的Camera APP来实现该功能是做不到的,但是Android系统强大的组件特性,使得应用开发者只需通过Intent就可以方便的打开系统自带的Camera APP,并通过MediaStroe方便地获取照片和视频的文件路径。具体我们还是用代码来说话吧:

例1、 实现拍照

在菜单或按钮的选择操作中调用如下代码,开启系统自带Camera APP,并传递一个拍照存储的路径给系统应用程序,具体如下:

imgPath = "/sdcard/test/img.jpg";

//必须确保文件夹路径存在,否则拍照后无法完成回调

File vFile = new File(imgPath);

if(!vFile.exists())

{

File vDirPath = vFile.getParentFile(); //new File(vFile.getParent());

vDirPath.mkdirs();

}

Uri uri = Uri.fromFile(vFile);

Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE);

intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);//

startActivityForResult(intent, SystemCapture);

上面我们使用的是startActivityForResult,所以最好需要重载void onActivityResult(int requestCode, int resultCode, Intent data)函数,不过因为当传入文件路径的的情况下,data返回参数是null值,只要resultCode为RESULT_OK,则上述代码中/sdcard/test/img.jpg的图片文件就是最新的照片文件。所以我们在这里只需给出如下简单的代码,将其显示到ImageView中

if (resultCode == RESULT_OK)

{

iViewPic.setImageURI(Uri.fromFile(new File(imgPath)));

}

假设不传参数MediaStore.EXTRA_OUTPUT的情况下,onActivityResult函数在resultCode为RESULT_OK的情况下,data返回的参数是经过实际拍摄照片经过缩放的图像数据,可以通过类似如下方法来打印缩放图像的尺寸

if (resultCode == RESULT_OK)

{

Bitmap bmp = (Bitmap)data.getExtras().get("data");

Log.d("Test", "bmp width:" + bmp.getWidth() + ", height:" + bmp.getHeight());

}

另外假如仅仅是调用系统照相机拍照,不关心拍照结果,则可以简单使用如下代码

Intent intent = new Intent(); //调用照相机

intent.setAction("android.media.action.STILL_IMAGE_CAMERA");

startActivity(intent);

备注:上面设置MediaStore.EXTRA_OUTPUT的方法,经过手机实测除了我们设定的路径下有照片外,在手机存储卡上也会保存一份照片,默认目录为sdcard/dcim/camera下面,我曾经尝试着想如果每次返回可以取得sdcard/dcim/camera下面的路径就好了,但是目前看来没办法直接获得,可以借助MediaStroe每次去查询最后一条照片记录,应该也是可行的。

例2、 实现摄像

在摄像功能时,尝试着设置MediaStore.EXTRA_OUTPUT以传入类似拍照时的文件路径,结果在我的测试真机上,那个视频文件居然是一个0k的空文件,最后通过类似如下代码实现

Intent intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE);

intent.putExtra(MediaStore.EXTRA_VIDEO_QUALITY, 1);//参数设置可以省略

startActivityForResult(intent, SystemVideoRecord);

在onActivityResult函数中进行如下代码调用

Uri videoUri = data.getData();

//String[] projection = { MediaStore.Video.Media.DATA, MediaStore.Video.Media.SIZE };

Cursor cursor = managedQuery(videoUri, null, null, null, null);

cursor.moveToFirst();//这个必须加,否则下面读取会报错

int num = cursor.getCount();

String recordedVideoFilePath = cursor.getString(cursor.getColumnIndex(MediaStore.Video.Media.DATA));

int recordedVideoFileSize = cursor.getInt(cursor.getColumnIndex(MediaStore.Video.Media.SIZE));

iResultText.setText(recordedVideoFilePath);

Log.i("videoFilePath", recordedVideoFilePath);

Log.i("videoSize", ""+recordedVideoFileSize);

上面的返回参数data,也会因为用户是否设置MediaStore.EXTRA_OUTPUT参数而改变,假设没有通过EXTRA_OUTPUT设置路径,data.getData返回的Uri为content://media/external/video/media/*,*个数字,代表具体的记录号,通过managedQuery可以获取到路径,假如设置了EXTRA_OUTPUT的话(比如/sdcard/test.3gp),则data.getData返回的Uri则为file:///sdcard/test.3gp,但是该文件居然是空白内容(不知道是不是跟手机有关,也没有在其它手机上验证过)。

根据Camera API实现自己的拍照和摄像程序

通过上面对调用系统Camera App实现拍照和摄像功能的例子,我们发现虽然能够满足我们的需求,但是毕竟自由度降低了,而且拍照的界面就是系统的样子,现在很多拍照程序,比如火爆的Camera 360软件等,就需要根据SDK提供的Camera API来编写自己的程序。

准备工作

上面调用系统Camera App,我们压根不需要任何权限,但是这里用Camera API,就必须在manifest内声明使用权限,通常由以下三项

<uses-permission android:name = "android.permission.CAMERA" />

<uses-feature android:name = "android.hardware.camera" />

<uses-feature android:name = "android.hardware.camera.autofocus" />

一般拍照和摄像的时候需要写到sd卡上,所以还有一向权限声明如下

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>

真做摄像功能时,需要音频录制和视频录制功能,所以又需要下面两项权限声明

<uses-permission android:name="android.permission.RECORD_VIDEO"/>

<uses-permission android:name="android.permission.RECORD_AUDIO"/>

另外使用Camera API拍照或摄像,都需要用到预览,预览就要用到SurfaceView,为此Activity的布局中必须有SurfaceView。

拍照流程

上面简单介绍了下准备工作,下面结合拍照过程中的需要用到的API对拍照流程做下简单描述

1、在Activity的OnCreate函数中设置好SurfaceView,包括设置SurfaceHolder.Callback对象和SurfaceHolder对象的类型,具体如下

SurfaceView mpreview = (SurfaceView) this.findViewById(R.id.camera_preview);

SurfaceHolder mSurfaceHolder = mpreview.getHolder();

mSurfaceHolder.addCallback(this);

mSurfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);

2、在SurfaceHolder.Callback的surfaceCreated函数中,使用Camera的Open函数开机摄像头硬件,这个API在SDK 2.3之前,是没有参数的,2.3以后支持多摄像头,所以开启前可以通过getNumberOfCameras先获取摄像头数目,再通过getCameraInfo得到需要开启的摄像头id,然后传入Open函数开启摄像头,假如摄像头开启成功则返回一个Camera对象,否则就抛出异常;

3、开启成功的情况下,在SurfaceHolder.Callback的surfaceChanged函数中调用getParameters函数得到已打开的摄像头的配置参数Parameters对象,如果有需要就修改对象的参数,然后调用setParameters函数设置进去(SDK2.2以后,还可以通过Camera::setDisplayOrientation设置方向);

4、同样在surfaceChanged函数中,通过Camera::setPreviewDisplay为摄像头设置SurfaceHolder对象,设置成功后调用Camera::startPreview函数开启预览功能,上面3,4两步的代码可以如下所示

public void surfaceChanged(SurfaceHolder holder, int format, int w, int h)

{

//已经获得Surface的width和height,设置Camera的参数

Camera.Parameters parameters = camera.getParameters();

parameters.setPreviewSize(w, h);

List<Size> vSizeList = parameters.getSupportedPictureSizes();

for(int num = 0; num < vSizeList.size(); num++)

{

Size vSize = vSizeList.get(num);

}

if(this.getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE)

{

//如果是竖屏

parameters.set("orientation", "portrait");

//在2.2以上可以使用

//camera.setDisplayOrientation(90);

}

else

{

parameters.set("orientation", "landscape");

//在2.2以上可以使用

//camera.setDisplayOrientation(0);

}

camera.setParameters(parameters);

try {

//设置显示

camera.setPreviewDisplay(holder);

} catch (IOException exception) {

camera.release();

camera = null;

}

//开始预览

camera.startPreview();

}

5、假设要支持自动对焦功能,则在需要的情况下,或者在上述surfaceChanged调用完startPreview函数后,可以调用Camera::autoFocus函数来设置自动对焦回调函数,该步是可选操作,有些设备可能不支持,可以通过Camera::getFocusMode函数查询。代码可以参考如下:

// 自动对焦

camera.autoFocus(new AutoFocusCallback()

{

@Override

public void onAutoFocus(boolean success, Camera camera)

{

if (success)

{

// success为true表示对焦成功,改变对焦状态图像

ivFocus.setImageResource(R.drawable.focus2);

}

}

});

6、在需要拍照的时候,调用takePicture(Camera.ShutterCallback, Camera.PictureCallback, Camera.PictureCallback, Camera.PictureCallback)函数来完成拍照,这个函数中可以四个回调接口,ShutterCallback是快门按下的回调,在这里我们可以设置播放“咔嚓”声之类的操作,后面有三个PictureCallback接口,分别对应三份图像数据,分别是原始图像、缩放和压缩图像和JPG图像,图像数据可以在PictureCallback接口的void onPictureTaken(byte[] data, Camera camera)中获得,三份数据相应的三个回调正好按照参数顺序调用,通常我们只关心JPG图像数据,此时前面两个PictureCallback接口参数可以直接传null;

7、每次调用takePicture获取图像后,摄像头会停止预览,假如需要继续拍照,则我们需要在上面的PictureCallback的onPictureTaken函数末尾,再次掉哟更Camera::startPreview函数;

8、在不需要拍照的时候,我们需要主动调用Camera::stopPreview函数停止预览功能,并且调用Camera::release函数释放Camera,以便其他应用程序调用。SDK中建议放在Activity的Pause函数中,但是我觉得放在surfaceDestroyed函数中更好,示例代码如下

// 停止拍照时调用该方法

public void surfaceDestroyed(SurfaceHolder holder)

{

// 释放手机摄像头

camera.release();

}

以上就是自己实现拍照程序的的流程,一般还可以还可以获取预览帧的图像数据,可以分别通过Camera::setPreviewCallback和Camera::setOneShotPreviewCallback来设置每帧或下一帧图像数据的回调,这里就不做展开了。

摄像流程

摄像流程也是需要预览的,而且流程上与拍照流程在起始的1~4步流程和结束的8流程是一样的,唯一不同的是6和7两个步骤,至于5自动对焦本身就是可选的,在摄像流程也没必要。

6、开启视频录制,需要创建一个MediaRecorder对象,并调用Camera::unLock操作解锁摄像头,因为默认Camera都是锁定的,只有解锁后MediaRecorder等多媒体进程调用,并设置一些参数,然后调用MediaRecorder:: start开启录制具体可以参阅如下代码:

MediaRecorder mMediaRecorder = new MediaRecorder();

// Unlock the camera object before passing it to media recorder.

camera.unlock();

mMediaRecorder.setCamera(camera);

mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.CAMCORDER);

mMediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);

mMediaRecorder.setProfile(mProfile);

mMediaRecorder.setMaxDuration(100000);//ms为单位

long dateTaken = System.currentTimeMillis();

Date date = new Date(dateTaken);

SimpleDateFormat dateFormat = new SimpleDateFormat(getString(R.string.video_file_name_format));

String title = dateFormat.format(date);

String filename = title + ".3gp"; // Used when emailing.

String cameraDirPath = ImageManager.CAMERA_IMAGE_BUCKET_NAME;

String filePath = cameraDirPath + "/" + filename;

File cameraDir = new File(cameraDirPath);

cameraDir.mkdirs();

mMediaRecorder.setOutputFile(filePath);

try {

mMediaRecorder.prepare();

mMediaRecorder.start(); // Recording is now started

} catch (RuntimeException e) {

Log.e(TAG, "Could not start media recorder. ", e);

return;

}

7、上面设置了最大间隔为100s,当100是视频录制结束,录制就会被停止,如果没有设时长和文件大小限制,那么通常需要调用MediaRecorder:: stop函数主动停止视频的录制,并将Camera对象通过lock函数继续加锁,示例代码如下

mMediaRecorder.stop();

mMediaRecorder.reset();

mMediaRecorder.release();

mMediaRecorder = null;

if(camera != null)

camera.lock();

之后的操作根据交互要么重新录制要么就释放Camera对象回到拍照流程的8步骤了。在这里就不做赘述了。

 java.lang.RuntimeException: setParameters failed的解决办法

public void surfaceChanged(SurfaceHolder holder, int format, int w,
int h) {
Camera.Parameters parameters = mCamera.getParameters();
Size size = parameters.getPreviewSize();
Log.e(VIEW_LOG_TAG, "size.width="+size.width+"size.height="+size.height);
// setPreviewSize 宽高不能大于 parameters.getPreviewSize(),否者会出现java.lang.RuntimeException: setParameters failed错误
//parameters.setPreviewSize(800, 600);
mCamera.setParameters(parameters);
mCamera.startPreview();
}

可参考Camera.Parameters 参数的使用

 示例:

	/**
	* 往图片上加经纬度等信息
	 * @param path 图片路径
	 */
	public void bitmapAddExif(String path){
		//往图片上加经纬度等信息
		ExifInterface exif;
		try {
			exif = new ExifInterface(path);
			if (mlocation != null) {
				//纬度
				exif.setAttribute(ExifInterface.TAG_GPS_LATITUDE, mlocation.getLatitude()+"");
				//经度
				exif.setAttribute(ExifInterface.TAG_GPS_LONGITUDE, mlocation.getLongitude()+"");
				//角度
				exif.setAttribute("GPSImgDirection", mlocation.getBearing() + "");
				//海拔高度
				exif.setAttribute(ExifInterface.TAG_GPS_ALTITUDE, mlocation.getAltitude()+"");
				//速度
				exif.setAttribute("GPSSpeed", mlocation.getSpeed() + "");
				//手机方向
				exif.setAttribute("SensorOrientation", sensorOrientation + "");
				//
				exif.setAttribute(ExifInterface.TAG_GPS_TIMESTAMP,dateFormat.format(mlocation.getTime()));
			}else {
				
				exif.setAttribute(ExifInterface.TAG_GPS_LONGITUDE, "0");
				exif.setAttribute(ExifInterface.TAG_GPS_LATITUDE, "0");
				exif.setAttribute(ExifInterface.TAG_GPS_TIMESTAMP, dateFormat.format(System.currentTimeMillis()));
			}
			exif.saveAttributes();
		} catch (IOException e) {
			e.printStackTrace();
		}
		
		
	}

  

 android.hardware.Camera.CameraInfo的使用

 

        /**
	 * 保持照片与实景一致
	 * 处理照相机的旋转角度
	 * @param orientation
	 * @return
	 */
	@TargetApi(Build.VERSION_CODES.GINGERBREAD)
	public int onOrientationChanged(int orientation) {
		
	     android.hardware.Camera.CameraInfo info = new android.hardware.Camera.CameraInfo();
	     if (info.facing == CameraInfo.CAMERA_FACING_FRONT) {
			
	    	 android.hardware.Camera.getCameraInfo(1, info);
		 }else {
			 android.hardware.Camera.getCameraInfo(0, info);
		}
	     int rotation = 0;
	     orientation = (orientation + 45) / 90 * 90;
	     //如果为前置摄像头
	     if (info.facing == CameraInfo.CAMERA_FACING_FRONT) {
	         rotation = (info.orientation - orientation + 360) % 360;
	     } else {  // back-facing camera
	         rotation = (info.orientation + orientation) % 360;
	     }
	     
	     return rotation;
	 }

 

        /**
	 * 保持预览界面与实景一致
	 * @param camera
	 */
	public  void setCameraDisplayOrientation(android.hardware.Camera camera) {
		 
		android.hardware.Camera.CameraInfo info = new android.hardware.Camera.CameraInfo();
		if (info.facing == CameraInfo.CAMERA_FACING_FRONT) {
			
	    	 android.hardware.Camera.getCameraInfo(1, info);
		 }else {
			 android.hardware.Camera.getCameraInfo(0, info);
		}
	    int rotation = this.getWindowManager().getDefaultDisplay().getRotation();
	 
	    int degrees = 0;
	 
	    switch (rotation) {
	 
	        case Surface.ROTATION_0: degrees = 0; break;
	 
	        case Surface.ROTATION_90: degrees = 90; break;
	 
	        case Surface.ROTATION_180: degrees = 180; break;
	 
	        case Surface.ROTATION_270: degrees = 270; break;
	 
	    }
	 
	    int result;
	 
	    if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {
	 
	        result = (info.orientation + degrees) % 360;
	 
	        result = (360 - result) % 360;  // compensate the mirror
	 
	    } else {  // back-facing
	 
	        result = (info.orientation - degrees + 360) % 360;
	 
	    }
			
	 	camera.setDisplayOrientation(result);
	}

 

  • 大小: 110.3 KB
分享到:
评论

相关推荐

    Android自定义Camera实现拍照功能

    Android自定义Camera实现拍照功能 Android操作系统中,Camera是一个非常重要的组件,通过Camera,我们可以实现拍照、录像等功能。在Android系统中,Camera组件可以分为两种:前置摄像头和后置摄像头。前置摄像头...

    android自定义Camera实现

    总之,Android自定义Camera的实现涉及许多细节,从初始化、配置参数到拍照和处理结果,每个环节都需要仔细处理。对于初学者来说,从简单的Camera API1开始,逐步熟悉后再尝试更复杂的Camera2 API,是一个合理的学习...

    Android自定义Camera

    在Android开发中,自定义Camera是一项常见的需求,...以上只是Android自定义Camera开发的基础内容,实际应用中可能涉及更多细节,如硬件加速、多摄像头支持、图像处理算法等。不断学习和实践,才能更好地掌握这个领域。

    Android 实例实现自定义Camera和前后置摄像头切换以及图片缩小放大预览

    在具体实现代码之前,我们先来了解一下Android api对实现自定义Camera的介绍。 根据api的介绍,对于Camera应用可以简单总结以下几个步骤。 1.检查Camera是否存在,并在AndroidManifest.xml中赋予相关的权限; 2....

    android 自定义相机,Camera和camera2.TextureView,相机遮罩层demo

    在Android开发中,自定义相机是一项常见的需求,它允许开发者根据特定的应用场景或者设计风格定制相机界面和功能。本文将详细讲解如何使用Camera API和Camera2 API结合TextureView来实现一个自定义相机,并添加相机...

    android自定义Camera实现录像和拍照

    本文实例为大家分享了android自定义Camera实现录像和拍照的具体代码,供大家参考,具体内容如下 源码: package com.example.myvideocamera; import java.io.BufferedOutputStream; import java.io.File; import ...

    Android自定义拍照解决部分手机拍完之后图片不清楚的问题

    综上所述,解决Android自定义拍照后图片不清楚的问题需要从多个方面着手,包括相机配置、图片质量和硬件兼容性的优化。开发者需要具备扎实的Android相机API知识,并且要有耐心去测试和调试,确保在各种设备上都能...

    Android自定义相机Camera

    总结一下,创建一个完善的Android自定义相机应用,需要理解Camera API,正确设置预览界面和尺寸,处理图像旋转问题,提供拍照控制以及释放资源。通过不断的优化和调试,可以打造出符合用户需求的高质量相机应用。在...

    android 自定义camera

    项目发展需要, 自定义camera,很显然android 2.3之前适配工作还是很麻烦的, 苦逼的android 开发们呐 源码不断更新中, 目前设计功能 聚焦 闪光 摄像头切换 放大缩小 色效支持

    android Camera 系统,自定义相机

    在Android平台上,自定义相机(Custom Camera)是一个常见的需求,涉及到对系统相机API的深入理解和应用。本主题将详述如何在Android中构建一个自定义的相机应用,涵盖关键概念、步骤以及可能遇到的问题。 一、...

    Android 自定义相机Camera2 apk

    android 自定义相机apk 是的安装包,请结合我的博客:http://blog.csdn.net/qq_38416326/article/details/71124946

    Android 自定义拍照实例(解决竖拍照片横向问题)

    在Android平台上,自定义拍照功能是一项常见的...通过以上方法,我们可以成功解决Android自定义拍照时竖拍照片横向显示的问题,提供良好的用户体验。在实际开发中,还要注意优化性能,确保拍照功能的流畅性和稳定性。

    android自定义camera

    以上就是创建一个具备自动对焦、触摸对焦及对焦框动画的Android自定义相机的基本步骤。实际开发中,你可能还需要处理相机权限、横竖屏切换、拍照保存等问题,以及考虑兼容不同设备和Android版本的差异。参考提供的...

    Android 自定义相机Camera2源码下载

    在Android平台上,自定义相机开发是一项常见的任务,Camera2 API是Google为了提供更高级、更灵活的相机控制而引入的接口。本教程将深入探讨如何使用Camera2 API创建一个自定义相机应用,通过分析提供的OneSelfCamera...

    Android 自定义相机Camera类

    本文将深入探讨如何利用`android.hardware.camera` API来构建一个自定义相机应用,并结合SurfaceView进行预览显示。 首先,我们需要了解`Camera`类是Android系统提供的核心组件,用于与设备的物理相机硬件进行交互...

    android自定义相机带取景框

    "android自定义相机带取景框"这个话题就是关于如何在Android应用中创建一个具有用户界面元素(如取景框)的自定义相机组件。下面将详细介绍这个过程中的关键知识点。 一、开启相机权限 在AndroidManifest.xml文件中...

Global site tag (gtag.js) - Google Analytics