(一) 前言
最近看Camera的api,觉得写的真的不错。现在翻译过来,给大家分享分享,译文可能不太好,大家将就着看哈。
(二) 正文
1. Camera
Camera是Android framework里面支持的,允许你拍照和拍摄视频的设备,那么,本文在接下来就会讨论如何为你的用户提供快速,简单的图片和视频拍摄方法。
2. 注意事项
a. 在确定你的程序使用Camera之前,你应该问自己几个问题:
i. Camera Requirement -- 使用照相机是不是对你的程序真的如此重要?如果确定如此,那么请在AndroidMainfest.xml文件中申明要使用照相机。
ii. Quick Picture or Customized Camera -- 快速拍照还是自定义照相机。你只是简单对拍照或者拍摄视频感兴趣还是你的程序要定义一套全新的照相机使用方法?如果只是简单的拍照或者录制视频,我们可以用一种非常简单的方法,如果想自定义照相机,那么请阅读自定义照相机片段。
iii. Storage -- 你的照片或者视频只想让你自己的程序看见,还是想和应用程序共享?比如Gallery,或者其他的多媒体和社交应用。 你是想让你的照片或者视频,即使你的应用已经被用户卸载,还仍然保持可见吗?那么请参考Saving Media Files章节。
ok.. 那么接下来,我们一个个讲解。
3. 在AndroidMainfest.xml中声明使用照相机.
a. Camera权限, 你必须在AndroidMainfest.xml中声明使用照相机的权限,例如
<uses-permission android:name="android.permission.CAMERA" />
注:如果你是调用系统Camera程序的话,就不必声明
b. Camera特性,你必须在AndroidMainfest.xml中声明照相机特性,例如:
<uses-feature android:name="android.hardware.camera" />
如果你的程序可能需要使用照相机,但并不是一定的,那么可以设置android:required属性,比如:
<uses-feature android:name="android.hardware.camera" android:required="false" />
c. 存储权限,如果你的程序想在扩展存储设备上(如sd卡)存储你的照片或者拍摄的视频,那么必须声明如下权限:
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
d. 音频录制权限, 为了录制音频或者视频,你必须在AndroidMainfest.xml文件中设置如下权限:
<uses-permission android:name="android.permission.RECORD_AUDIO" />
4. 使用系统Camera程序
a. 这是一种可以让你的程序以最少的编码,然后可以快速使用照相机的方法,它通过一个Intent来调用系统的Camera程序。一个Camera的intent可以发起一个请求,来让系统的Camera应用程序去拍摄照片或者录制视频,而且可以返回拍摄或者录制的结果给你自己的应用程序,那么接下来,我们将讨论如何使用这种方法。
i. 构造一个Camera Intent -- 创建一个拍摄照片或者视频的Intent, 我们可以使用如下两种方法:
MediaStore.ACTION_IMAGE_CAPTURE -- 一个intent action,用来向系统Camera程序请求拍摄图片。
MediaStore.ACTION_VIDEO_CAPTURE -- 一个intent action, 用来向系统Camera程序请求录制视频。
ii. 开启Camera intent -- 通过调用startActivityForResult()来执行Camera intent, 在你调用这个方法之后,系统Camera程序就是出现在用户界面,然后用户就可以拍摄照片或者视频了。
iii. 接收Intent 结果 -- 在你的应用程序里面建立一个onActivityResult()方法来接收来自系统Camera程序的执行结果。当用户拍摄了照片或者视频(也许取消了拍摄操作),系统就会调用这个方法。
b. 图片拍摄Intent
使用系统Camera来拍摄图片,是一种以最快的速度,最少的代码,让你的程序能够拍照的方法。一个图片拍摄的intent可以包含如下扩展信息:
i. MediaStore.EXTRA_OUTPUT -- 指定一个Uri对象,系统Camera程序会把拍摄的图片存储到指定位置。这个设置一般是强烈推荐的。如果你不指定这个Uri路径,那么系统Camera程序会把图片以一个默认的名字,存储在一个默认的位置。我们可以通过返回的intent,然后用Intent.getData()方法来获取这个值。
ii. 下面这个例子可以告诉大家怎么构造一个拍摄图片的intent,并且执行它。
01 |
private static final int CAPTURE_IMAGE_ACTIVITY_REQUEST_CODE = 100 ; |
05 |
public void onCreate(Bundle savedInstanceState) { |
06 |
super .onCreate(savedInstanceState); |
07 |
setContentView(R.layout.main); |
09 |
// create Intent to take a picture and return control to the calling application |
10 |
Intent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); |
12 |
fileUri = getOutputMediaFileUri(MEDIA_TYPE_IMAGE); // create a file to save the image |
13 |
intent.putExtra(MediaStore.EXTRA_OUTPUT, fileUri); // set the image file name |
15 |
// start the image capture Intent |
16 |
startActivityForResult(intent, CAPTURE_IMAGE_ACTIVITY_REQUEST_CODE); |
当startActivityForResult()方法执行之后,用户就会看到系统的Camera界面。当用户完成拍摄照片(或者取消拍摄)之后,用户界面就会返回你自己的应用程序。那么你必须拦截onActivityResult方法,以便接收执行结果和继续你自己代码的逻辑。
c. 视频录制intent
使用系统Camera来录制视频,是一种以最快的速度,最少的代码,让你的程序能够录制视频的方法。一个视频录制的intent可以包含如下扩展信息:
MediaStore.EXTRA_OUTPUT -- 指定一个Uri对象,系统Camera程序会把录制的视频存储到指定位置。这个设置一般是强烈推荐的。如果你不指定这个Uri路径,那么系统Camera程序会把视频以一个默认的名字,存储在一个默认的位置。我们可以通过返回的intent,然后用Intent.getData()方法来获取这个值。
MediaStore.EXTRA_VIDEO_QUALITY -- 视频质量,0(低质量,比较小的文件来存储视频), 1(高质量,比较大的文件来存储视频)
MediaStore.EXTRA_DURATION_LIMIT -- 设置一个值来限制视频时间长度,秒为单位。
MediaStore.EXTRA_SIZE_LIMIT -- 设置一个值来限制视频大小,byte为单位。
下面这个例子可以告诉大家怎么构造一个录制视频的intent,并且执行它。
01 |
private static final int CAPTURE_VIDEO_ACTIVITY_REQUEST_CODE = 200 ; |
05 |
public void onCreate(Bundle savedInstanceState) { |
06 |
super .onCreate(savedInstanceState); |
07 |
setContentView(R.layout.main); |
10 |
Intent intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE); |
12 |
fileUri = getOutputMediaFileUri(MEDIA_TYPE_VIDEO); // create a file to save the video |
13 |
intent.putExtra(MediaStore.EXTRA_OUTPUT, fileUri); // set the image file name |
15 |
intent.putExtra(MediaStore.EXTRA_VIDEO_QUALITY, 1 ); // set the video image quality to high |
17 |
// start the Video Capture Intent |
18 |
startActivityForResult(intent, CAPTURE_VIDEO_ACTIVITY_REQUEST_CODE); |
当startActivityForResult()方法执行之后,用户就会看到系统的Camera界面。当用户完成录制视频(或者取消拍摄)之后,用户界面就会返回你自己的应用程序。那么你必须拦截onActivityResult方法,以便接收执行结果和继续你自己代码的逻辑。
d. 接受Camera返回的结果
一旦你执行了拍摄图片或者视频的intent之后,那么你必须去接收执行结果。那么,下面的内容就是告诉你怎么去接收执行结果,以及进一步处理Camera拍摄的图片或者视频。
为了接受Camera拍摄的结果,你必须重写onActivityResult()方法,如下:
01 |
private static final int CAPTURE_IMAGE_ACTIVITY_REQUEST_CODE = 100 ; |
02 |
private static final int CAPTURE_VIDEO_ACTIVITY_REQUEST_CODE = 200 ; |
05 |
protected void onActivityResult( int requestCode, int resultCode, Intent data) { |
06 |
if (requestCode == CAPTURE_IMAGE_ACTIVITY_REQUEST_CODE) { |
07 |
if (resultCode == RESULT_OK) { |
08 |
// Image captured and saved to fileUri specified in the Intent |
09 |
Toast.makeText( this , "Image saved to:\n" + |
10 |
data.getData(), Toast.LENGTH_LONG).show(); |
11 |
} else if (resultCode == RESULT_CANCELED) { |
12 |
// User cancelled the image capture |
14 |
// Image capture failed, advise user |
18 |
if (requestCode == CAPTURE_VIDEO_ACTIVITY_REQUEST_CODE) { |
19 |
if (resultCode == RESULT_OK) { |
20 |
// Video captured and saved to fileUri specified in the Intent |
21 |
Toast.makeText( this , "Video saved to:\n" + |
22 |
data.getData(), Toast.LENGTH_LONG).show(); |
23 |
} else if (resultCode == RESULT_CANCELED) { |
24 |
// User cancelled the video capture |
26 |
// Video capture failed, advise user |
一旦你的Activity接收到成功执行拍摄的结果,那么存储在指定位置的图片或者视频,就能被我们的程序使用了。
5. 构造自己的Camera程序
一些开发者可能需要自己定制Camera应用程序,定制自定义的Camera程序可能比调用系统的Camera需要更多的代码,但是这真的能给你的用户带来不一样的用户体验。。
下面是开发自定义Camera的一般步骤:
a. 检测Camera设备的可访问性 -- 书写代码确认Camera设备是否存在并且请求访问。
b. 创建一个预览窗口 -- 创建一个预览窗口,一般继承SurfaceView并且实现SurfaceHolder接口,这个预览窗口可以通过Camera动态预览图片。
c. 创建一个预览布局 -- 一旦你拥有了预览窗口,那么创建一个视频Layout来封装预览窗口。
d. 设置拍摄监听器 -- 设置图片或者视频拍摄的监听器,比如按下一个Button
e. 拍摄和保存文件
d. 释放Camera设备 -- 拍摄完毕之后,你的程序必须释放Camera设备,以便其他程序使用Camera。
Camera是一个共享的硬件资源,所以你的程序必须非常小心的管理它,以便不能和其他程序发生冲突。下面的内容就是告诉大家怎么去检测Camera设备,怎么去请求访问Camera,以及当你的程序关闭的时候怎么释放Camera。
6. 检测Camera硬件设备
如果你的程序没有在AndroidMainfest.xml中申明请求使用Camera,那么你应该在运行期去检查Camera是否可用。使用PackageManager.hasSystemFeature() 方法,比如:
01 |
/** Check if this device has a camera */ |
02 |
private boolean checkCameraHardware(Context context) { |
03 |
if (context.getPackageManager().hasSystemFeature(PackageManager.FEATURE_CAMERA)){ |
04 |
// this device has a camera |
07 |
// no camera on this device |
注: 一个Android设备可能含有多个Camera设备,比如背部摄像头用来拍照,前部摄像头用来拨打视频电话,那么,Android2.3(API Level 9) 以及以后的版本将允许你检查设备上可以用的Camera设备,通过调用Camera.getNumberOfCameras()方法。
7. 访问Camera设备
为了打开主Camera设备,我们可以调用Camera.open() 方法,如下:
01 |
/** A safe way to get an instance of the Camera object. */ |
02 |
public static Camera getCameraInstance(){ |
05 |
c = Camera.open(); // attempt to get a Camera instance |
08 |
// Camera is not available (in use or does not exist) |
10 |
return c; // returns null if camera is unavailable |
注: 当使用Camera.open()方法的时候,无论如何都需要检查是否发生了异常。如果忘记检查异常的话,有可能会因为Camera设备不存在而导致你的应用程序崩溃。
在Android2.3(API Level 9)或者更高版本,你可以通过Camera.open(int)方法来访问指定的Camera设备,
8. 检查Camera属性
一旦你可以访问Camera设备之后,你可以调用Camera.getParameters()方法来获取更多的Camera信息,当使用API Level 9或者更高版本时,可以使用Camera.getCameraInfo()方法来检测Camera设备是前端设备还是背部设备。
9. 创建预览窗口
01 |
/** A basic Camera preview class */ |
02 |
public class CameraPreview extends SurfaceView implements SurfaceHolder.Callback { |
03 |
private SurfaceHolder mHolder; |
04 |
private Camera mCamera; |
06 |
public CameraPreview(Context context, Camera camera) { |
10 |
// Install a SurfaceHolder.Callback so we get notified when the |
11 |
// underlying surface is created and destroyed. |
12 |
mHolder = getHolder(); |
13 |
mHolder.addCallback( this ); |
14 |
// deprecated setting, but required on Android versions prior to 3.0 |
15 |
mHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS); |
18 |
public void surfaceCreated(SurfaceHolder holder) { |
19 |
// The Surface has been created, now tell the camera where to draw the preview. |
21 |
mCamera.setPreviewDisplay(holder); |
22 |
mCamera.startPreview(); |
23 |
} catch (IOException e) { |
24 |
Log.d(TAG, "Error setting camera preview: " + e.getMessage()); |
28 |
public void surfaceDestroyed(SurfaceHolder holder) { |
29 |
// empty. Take care of releasing the Camera preview in your activity. |
32 |
public void surfaceChanged(SurfaceHolder holder, int format, int w, int h) { |
33 |
// If your preview can change or rotate, take care of those events here. |
34 |
// Make sure to stop the preview before resizing or reformatting it. |
36 |
if (mHolder.getSurface() == null ){ |
37 |
// preview surface does not exist |
41 |
// stop preview before making changes |
43 |
mCamera.stopPreview(); |
44 |
} catch (Exception e){ |
45 |
// ignore: tried to stop a non-existent preview |
48 |
// make any resize, rotate or reformatting changes here |
50 |
// start preview with new settings |
52 |
mCamera.setPreviewDisplay(mHolder); |
53 |
mCamera.startPreview(); |
55 |
} catch (Exception e){ |
56 |
Log.d(TAG, "Error starting camera preview: " + e.getMessage()); |
10. 把预览窗口设置到一个Layout里面
定义Activity配置文件,
01 |
<?xml version= "1.0" encoding= "utf-8" ?> |
03 |
android:orientation= "horizontal" |
04 |
android:layout_width= "fill_parent" |
05 |
android:layout_height= "fill_parent" |
08 |
android:id= "@+id/camera_preview" |
09 |
android:layout_width= "fill_parent" |
10 |
android:layout_height= "fill_parent" |
11 |
android:layout_weight= "1" |
15 |
android:id= "@+id/button_capture" |
16 |
android:text= "Capture" |
17 |
android:layout_width= "wrap_content" |
18 |
android:layout_height= "wrap_content" |
19 |
android:layout_gravity= "center" |
在大多数设备上,拍摄照片一般是横屏的,所以,你需要添加如下代码:
01 |
< activity android:name = ".CameraActivity" |
02 |
android:label = "@string/app_name" |
04 |
android:screenOrientation = "landscape" > |
05 |
<!-- configure this activity to use landscape orientation --> |
08 |
< action android:name = "android.intent.action.MAIN" /> |
09 |
< category android:name = "android.intent.category.LAUNCHER" /> |
在Activity中,把预览窗口添加到Layout里面去
01 |
public class CameraActivity extends Activity { |
03 |
private Camera mCamera; |
04 |
private CameraPreview mPreview; |
07 |
public void onCreate(Bundle savedInstanceState) { |
08 |
super .onCreate(savedInstanceState); |
09 |
setContentView(R.layout.main); |
11 |
// Create an instance of Camera |
12 |
mCamera = getCameraInstance(); |
14 |
// Create our Preview view and set it as the content of our activity. |
15 |
mPreview = new CameraPreview( this , mCamera); |
16 |
FrameLayout preview = (FrameLayout) findViewById(id.camera_preview); |
17 |
preview.addView(mPreview); |
11. 拍照
a. 首先定义一个接口,用来处理拍摄完毕之后的动作。
01 |
private PictureCallback mPicture = new PictureCallback() { |
04 |
public void onPictureTaken( byte [] data, Camera camera) { |
06 |
File pictureFile = getOutputMediaFile(MEDIA_TYPE_IMAGE); |
07 |
if (pictureFile == null ){ |
08 |
Log.d(TAG, "Error creating media file, check storage permissions: " + |
14 |
FileOutputStream fos = new FileOutputStream(pictureFile); |
17 |
} catch (FileNotFoundException e) { |
18 |
Log.d(TAG, "File not found: " + e.getMessage()); |
19 |
} catch (IOException e) { |
20 |
Log.d(TAG, "Error accessing file: " + e.getMessage()); |
getOutputMediaFile(int) 方法的实现
01 |
public static final int MEDIA_TYPE_IMAGE = 1 ; |
02 |
public static final int MEDIA_TYPE_VIDEO = 2 ; |
04 |
/** Create a file Uri for saving an image or video */ |
05 |
private static Uri getOutputMediaFileUri( int type){ |
06 |
return Uri.fromFile(getOutputMediaFile(type)); |
09 |
/** Create a File for saving an image or video */ |
10 |
private static Uri getOutputMediaFile( int type){ |
11 |
// To be safe, you should check that the SDCard is mounted |
12 |
// using Environment.getExternalStorageState() before doing this. |
14 |
File mediaStorageDir = new File(Environment.getExternalStoragePublicDirectory( |
15 |
Environment.DIRECTORY_PICTURES), "MyCameraApp" ); |
16 |
// This location works best if you want the created images to be shared |
17 |
// between applications and persist after your app has been uninstalled. |
19 |
// Create the storage directory if it does not exist |
20 |
if (! mediaStorageDir.exists()){ |
21 |
if (! mediaStorageDir.mkdirs()){ |
22 |
Log.d( "MyCameraApp" , "failed to create directory" ); |
27 |
// Create a media file name |
28 |
String timeStamp = new SimpleDateFormat( "yyyyMMdd_HHmmss" ).format( new Date()); |
30 |
if (type == MEDIA_TYPE_IMAGE){ |
31 |
mediaFile = new File(mediaStorageDir.getPath() + File.separator + |
32 |
"IMG_" + timeStamp + ".jpg" ); |
33 |
} else if (type == MEDIA_TYPE_VIDEO) { |
34 |
mediaFile = new File(mediaStorageDir.getPath() + File.separator + |
35 |
"VID_" + timeStamp + ".mp4" ); |
设置我们刚才的监听器
01 |
// Add a listener to the Capture button |
02 |
Button captureButton = (Button) findViewById(id.button_capture); |
03 |
captureButton.setOnClickListener( |
04 |
new View.OnClickListener() { |
06 |
public void onClick(View v) { |
07 |
// get an image from the camera |
08 |
mCamera.takePicture( null , null , mPicture); |
好多。。 翻译的我头都晕了,真心觉得api是不错的阅读文档~~
- Android Camera API/Camera2 API 相机预览及滤镜、贴纸等处理
Android Lollipop 添加了Camera2 API,并将原来的Camera API标记为废弃了.相对原来的Camera API来说.Camera2是又一次定义的相机 API,也重构了相机 ...
- Android Camera API ISO Setting
https://stackoverflow.com/questions/2978095/android-camera-api-iso-setting exif this.mCameraParamete ...
- Android 音视频开发(四):使用 Camera API 采集视频数据
本文主要将的是:使用 Camera API 采集视频数据并保存到文件,分别使用 SurfaceView.TextureView 来预览 Camera 数据,取到 NV21 的数据回调. 注: 需要权限 ...
- Android 新老两代 Camera API 大起底
https://blog.csdn.net/Byeweiyang/article/details/80515192 0.背景简介 最近有一部分相机相关的需求,专注于对拍摄的照片.视频的噪点.色温.明暗 ...
- 【Android】Android Camera原始帧格式转换 —— 获取Camera图像(一)
概述: 做过Android Camera图像采集和处理的朋友们应该都知道,Android手机相机采集的原始帧(RawFrame)默认是横屏格式的,而官方API有没有提供一个设置Camera采集图像的 ...
- Using the Cordova Camera API
使用ionic开发一款android或ios应用,估计少不了使用到Camera API,这里记录下使用过程. 创建空的ionic应用 ionic start myTabs tabs 通过cd demo ...
- Android Camera 相机程序编写
Android Camera 相机程序编写 要自己写一个相机应用直接使用相机硬件,首先应用需要一个权限设置,在AndroidManifest.xml中加上使用设备相机的权限: <uses-per ...
- Android开发-API指南-应用程序开发基础
Application Fundamentals 英文原文:http://developer.android.com/guide/components/fundamentals.html 采集(更新) ...
- Android Camera 使用小结
Android手机关于Camera的使用,一是拍照,二是摄像,由于Android提供了强大的组件功能,为此对于在Android手机系统上进行Camera的开发,我们可以使用两类方法:一是借助Inten ...
随机推荐
- 股票K线图-JfreeChart版
http://blog.csdn.net/ami121/article/details/3953272 股票K线图-JfreeChart版 标签: jfreechartpropertiesapplet ...
- scala与java的==的比较
如果你想比较一下看看两个对象是否相等,可以使用或者==,或它的反义 !=.(对所有对象都适用,而不仅仅是基本数据类型) ? 1 2 3 4 scala> 1 == 2 res24: Boole ...
- hdu_3966_Aragorn's Story(树链剖分裸题)
题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=3966 题意:给你一棵树,然后给定点之间的路径权值修改,最后单点查询 题解:树链剖分裸题,这里我用树状数 ...
- hdu_1728_逃离迷宫(bfs)
题目连接:http://acm.hdu.edu.cn/showproblem.php?pid=1728 题意:走迷宫,找最小的拐角 题解:对BFS有了新的理解,DFS+剪枝应该也能过,用BFS就要以拐 ...
- dfs.replication 参数 动态修改
首先 dfs.replication这个参数是个client参数,即node level参数.需要在每台datanode上设置.其实默认为3个副本已经够用了,设置太多也没什么用. 一个文件,上传到hd ...
- 贾扬清分享_深度学习框架caffe
Caffe是一个清晰而高效的深度学习框架,其作者是博士毕业于UC Berkeley的 贾扬清,目前在Google工作.本文是根据机器学习研究会组织的online分享的交流内容,简单的整理了一下. 目录 ...
- 《Head.First设计模式》的学习笔记(9)--外观模式
意图:为子系统中的一组接口提供一个一致的界面,Facade 模式定义了一个高层接口,这个接口使得这一子系统更加容易使用. 结构: 例子: 假设你有一套杀手级的家庭影院系统,内含DVD播放器.投影仪.自 ...
- QML Flipable、Flickable和状态与动画 下篇
本文介绍的是Flickable和状态与动画,我们以前接触过QML相关的内容,那么本文介绍的内容就很明了了.先来看内容. AD: Flickable和状态与动画 下篇是本节要介绍的内容,Flickabl ...
- label不换行的问题
除了numberOfLines属性label有一个preferredMaXLayoutWidth属性.设置试试
- wl18xx编译的时候出现WARNING: "simple_open" WARNING: "wl12xx_get_platform_data"
................................................................................................... ...