查看: 11191|回复: 0
收起左侧

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

[复制链接]
发表于 2015-4-15 15:42:30 | 显示全部楼层 |阅读模式
     Android手机关于Camera的使用,一是拍照,二是摄像,由于Android提供了强大的组件功能,为此对于在Android手机系统上进行Camera的开发,我们可以使用两类方法:一是借助Intent和MediaStroe调用系统Camera App程序来实现拍照和摄像功能,二是根据Camera API自写Camera程序。由于自写Camera需要对Camera API了解很充分,而且对于通用的拍照和摄像应用只需要借助系统Camera App程序就能满足要求了,为此先从调用系统Camera App应用开始来对Android Camera做个简单的使用小结。4 }; Y( L0 T) y+ K  g/ R/ u. M+ O! m' P
8 y1 M& T7 ?/ d$ x( z
调用系统Camera App实现拍照和摄像功能
2 B0 m; }4 y/ G1 Q: o
    不是专门的Camera应用,一般用到Camera的需求就是获取照片或者视频,比如微博分享、随手记等,对于在Symbian系统上通过简单地调用系统自带的Camera APP来实现该功能是做不到的,但是Android系统强大的组件特性,使得应用开发者只需通过Intent就可以方便的打开系统自带的Camera APP,并通过MediaStroe方便地获取照片和视频的文件路径。具体我们还是用代码来说话吧: * e  E: \8 B4 a% P3 l
例1、 实现拍照
7 I: v+ S3 T# ~2 |5 l在菜单或按钮的选择操作中调用如下代码,开启系统自带Camera APP,并传递一个拍照存储的路径给系统应用程序,具体如下:8 f% C  h- x; O2 g. i( E
imgPath = "/sdcard/test/img.jpg";
; W/ M% p2 |, D% i/ ^//必须确保文件夹路径存在,否则拍照后无法完成回调
3 T# F& q1 v8 |9 J9 w; t3 f; d4 dFile vFile = new File(imgPath);
5 L0 e9 D2 b" Q+ e; fif(!vFile.exists()) , G) S1 w7 i, J: B4 w) V
{ " I; X7 t, P2 ?- z1 y
File vDirPath = vFile.getParentFile(); //new File(vFile.getParent());
5 G- E. K* T& h% K1 MvDirPath.mkdirs(); ! N3 U( I" S5 c, F. E8 `3 C
}
$ c$ Z% R0 [2 F9 L6 w& Y, O1 U! hUri uri = Uri.fromFile(vFile);
) t8 C: H, |2 @) t, IIntent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); 4 }$ _1 Y) X0 r9 W8 [
intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);//
8 u2 c6 ?/ |3 j2 s4 v  QstartActivityForResult(intent, SystemCapture); 4 E; K  Z: x+ O! {4 @$ _! @
上面我们使用的是startActivityForResult,所以最好需要重载void onActivityResult(int requestCode, int resultCode, Intent data)函数,不过因为当传入文件路径的的情况下,data返回参数是null值,只要resultCode为RESULT_OK,则上述代码中/sdcard/test/img.jpg的图片文件就是最新的照片文件。所以我们在这里只需给出如下简单的代码,将其显示到ImageView中 ) W2 @! G, T: E+ V7 Z$ M
if (resultCode == RESULT_OK)  8 H6 {+ I$ _: x
{ % ~. b9 ?  U9 p
iViewPic.setImageURI(Uri.fromFile(new File(imgPath)));
) {8 T0 R! t, D}
; u' s0 u; y- D* B/ J9 Z假设不传参数MediaStore.EXTRA_OUTPUT的情况下,onActivityResult函数在resultCode为RESULT_OK的情况下,data返回的参数是经过实际拍摄照片经过缩放的图像数据,可以通过类似如下方法来打印缩放图像的尺寸 + Q1 h% g, Y! l' l
if (resultCode == RESULT_OK)  ( b( ^& I4 ]! E& Y+ B
{ # C- M+ H* m% f+ g. q8 r' w! z
Bitmap bmp = (Bitmap)data.getExtras().get("data");
( q# V* a% w8 L" x* cLog.d("Test", "bmp width:" + bmp.getWidth() + ", height:" + bmp.getHeight());
& G9 k) m  n3 x4 D5 r5 V1 p$ Q} $ r8 Y/ T! W. G+ |" E* q# u
另外假如仅仅是调用系统照相机拍照,不关心拍照结果,则可以简单使用如下代码 & s0 E9 _# P: i* w$ W, p
Intent intent = new Intent(); //调用照相机
" ]$ q" @5 J/ iintent.setAction("android.media.action.STILL_IMAGE_CAMERA");
- ?6 F1 ^/ ?4 m" estartActivity(intent);
& m3 N0 }0 I& p& P3 l5 k$ b4 N4 Y/ _, y备注:上面设置MediaStore.EXTRA_OUTPUT的方法,经过手机实测除了我们设定的路径下有照片外,在手机存储卡上也会保存一份照片,默认目录为sdcard/dcim/camera下面,我曾经尝试着想如果每次返回可以取得sdcard/dcim/camera下面的路径就好了,但是目前看来没办法直接获得,可以借助MediaStroe每次去查询最后一条照片记录,应该也是可行的。 1 x- x( P: R  Y3 z
例2、 实现摄像 : ]- M6 e. U+ {7 i' G
在摄像功能时,尝试着设置MediaStore.EXTRA_OUTPUT以传入类似拍照时的文件路径,结果在我的测试真机上,那个视频文件居然是一个0k的空文件,最后通过类似如下代码实现 9 {! M: Q4 _/ R7 r
Intent intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE);
! K4 J8 E! K5 \% S7 iintent.putExtra(MediaStore.EXTRA_VIDEO_QUALITY, 1);//参数设置可以省略
: A) k0 p: H, x, h" WstartActivityForResult(intent, SystemVideoRecord); 8 Q9 |3 l) n% X  C5 I* @, O
在onActivityResult函数中进行如下代码调用 % L+ L4 K9 Y. _
Uri videoUri = data.getData(); 9 J4 O* P& `+ P8 Q- P% c( L; P
//String[] projection = { MediaStore.Video.Media.DATA, MediaStore.Video.Media.SIZE }; 8 u- z  d3 o- l5 a3 Q9 ~
Cursor cursor = managedQuery(videoUri, null, null, null, null);
0 K% H; Y7 K$ Q' u  \cursor.moveToFirst();//这个必须加,否则下面读取会报错
( i* m# }/ Z6 E- [& c% `( o6 Iint num = cursor.getCount();
0 D( q+ M/ W; {8 e) iString recordedVideoFilePath = cursor.getString(cursor.getColumnIndex(MediaStore.Video.Media.DATA)); $ N( `/ V5 b/ d. e: G' c* x
int recordedVideoFileSize = cursor.getInt(cursor.getColumnIndex(MediaStore.Video.Media.SIZE)); * c1 `1 p; W" k& d
iResultText.setText(recordedVideoFilePath); 9 t3 r2 M/ C; o$ o# k8 ~
Log.i("videoFilePath", recordedVideoFilePath);
  m6 z# F$ U2 p( Q5 C0 ]! m) \+ DLog.i("videoSize", ""+recordedVideoFileSize);
& c' g5 A% X7 t+ h* f; g上面的返回参数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,但是该文件居然是空白内容(不知道是不是跟手机有关,也没有在其它手机上验证过)。 . D0 ?' D' {0 a/ }8 y
根据Camera API实现自己的拍照和摄像程序 通过上面对调用系统Camera App实现拍照和摄像功能的例子,我们发现虽然能够满足我们的需求,但是毕竟自由度降低了,而且拍照的界面就是系统的样子,现在很多拍照程序,比如火爆的Camera 360软件等,就需要根据SDK提供的Camera API来编写自己的程序。 9 \, W4 B9 Q  s: X
准备工作 上面调用系统Camera App,我们压根不需要任何权限,但是这里用Camera API,就必须在manifest内声明使用权限,通常由以下三项 9 v8 p( T" R5 ~' C( d) w. e
<uses-permission android:name = "android.permission.CAMERA" />
4 t% T1 y! B, d; H7 p, u<uses-feature android:name = "android.hardware.camera" />
  m! \* P. @/ x( g9 |  p2 v$ B. ]<uses-feature android:name = "android.hardware.camera.autofocus" /> ! y' g# S6 a8 b0 n4 o+ T
一般拍照和摄像的时候需要写到sd卡上,所以还有一向权限声明如下
; t! e. Z! `/ @4 H0 A  C" {<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
( m2 k* Y4 j4 R& w  l真做摄像功能时,需要音频录制和视频录制功能,所以又需要下面两项权限声明 3 p- P/ I7 A: f1 y! U( u
<uses-permission android:name="android.permission.RECORD_VIDEO"/> + h9 U& F' P0 Z4 L* t2 \$ A: e
<uses-permission android:name="android.permission.RECORD_AUDIO"/>
2 \& ?. R. V7 x  e% O8 V& _另外使用Camera API拍照或摄像,都需要用到预览,预览就要用到SurfaceView,为此Activity的布局中必须有SurfaceView。
3 l  R! }4 Q$ f, C! R/ _
# ]2 ^6 i8 H8 d: c" r

/ E' g8 Y% J: A2 k) F拍照流程 上面简单介绍了下准备工作,下面结合拍照过程中的需要用到的API对拍照流程做下简单描述
" B/ }8 k. L' M. \1、在Activity的OnCreate函数中设置好SurfaceView,包括设置SurfaceHolder.Callback对象和SurfaceHolder对象的类型,具体如下   C! [: D% l$ v( t
SurfaceView mpreview = (SurfaceView) this.findViewById(R.id.camera_preview); / E: W, M2 M2 i; \
SurfaceHolder mSurfaceHolder = mpreview.getHolder();
( u% X0 @9 o& [2 E2 {mSurfaceHolder.addCallback(this); ! T$ a% I+ ~5 `$ ~+ S
mSurfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS); - y$ o" }/ K5 B4 V, W
2、在SurfaceHolder.Callback的surfaceCreated函数中,使用Camera的Open函数开机摄像头硬件,这个API在SDK 2.3之前,是没有参数的,2.3以后支持多摄像头,所以开启前可以通过getNumberOfCameras先获取摄像头数目,再通过getCameraInfo得到需要开启的摄像头id,然后传入Open函数开启摄像头,假如摄像头开启成功则返回一个Camera对象,否则就抛出异常; ! G% j/ x/ d! |: R# k( V) J7 J
3、开启成功的情况下,在SurfaceHolder.Callback的surfaceChanged函数中调用getParameters函数得到已打开的摄像头的配置参数Parameters对象,如果有需要就修改对象的参数,然后调用setParameters函数设置进去(SDK2.2以后,还可以通过Camera::setDisplayOrientation设置方向); ; O8 Y8 }5 O, I0 D4 y
4、同样在surfaceChanged函数中,通过Camera::setPreviewDisplay为摄像头设置SurfaceHolder对象,设置成功后调用Camera::startPreview函数开启预览功能,上面3,4两步的代码可以如下所示
0 Y) l  t; g5 j4 j3 x+ R2 rpublic void surfaceChanged(SurfaceHolder holder, int format, int w, int h) ! Q/ I9 r- p8 q: X2 Z3 s" O
{ : i! i6 G+ g! u8 Z/ f
//已经获得Surface的width和height,设置Camera的参数
5 B3 R0 ~3 k1 j; L/ H1 O1 gCamera.Parameters parameters = camera.getParameters(); % H- Z, c( }( k/ F5 z) o  B. Q
parameters.setPreviewSize(w, h); / ~" v6 f* |& ~: @4 g
List<Size> vSizeList = parameters.getSupportedPictureSizes(); ; V! [5 P& w  X6 h5 P" w2 b) a. C6 }
for(int num = 0; num < vSizeList.size(); num++)
0 k4 e- ~( Z+ r{ 6 Q" k! _; ~, u5 N4 l! z
Size vSize = vSizeList.get(num);
" ~+ d& x7 w9 H2 _} ! ^7 t8 |: Y' s
if(this.getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE)
. j# K$ T1 B+ P* ~{ ( a) [; `1 Q+ L2 h9 [
//如果是竖屏
7 s* o  |" }5 E2 V9 I: d$ Iparameters.set("orientation", "portrait"); ' s8 c+ O2 I$ o) }0 J7 l% G9 }4 l
//在2.2以上可以使用 * l7 _& X/ \( ]
//camera.setDisplayOrientation(90);
2 E: x% ^0 W2 [7 m} ! C% S/ P" P' @) m7 H& b/ Q( E
else
4 n+ `! W' W5 W" V9 }{ , \2 t3 B' p; m/ n
parameters.set("orientation", "landscape"); 3 j% O1 w) o- q
//在2.2以上可以使用 2 N. K4 \% {5 t3 a  `: b# `
//camera.setDisplayOrientation(0);   K) a( }! w9 m' B) O
}
( l& r% B# F+ A7 E. m3 mcamera.setParameters(parameters);
* H) C4 \. b& ?4 etry {
7 _$ S" n& b  Z5 S4 p. `  k& X//设置显示
  {% }9 d. N$ \7 P& C2 }camera.setPreviewDisplay(holder);
2 u1 v( V7 k; h$ K0 j} catch (IOException exception) {
+ O, x: |& G) ncamera.release();
1 }/ q! k3 n9 O2 _camera = null;
0 T  b2 D% I# i} 5 P- Q9 M! f$ E8 ]
//开始预览
1 {* V% g  f$ Z2 ]7 c8 h9 Bcamera.startPreview();
4 i! U# V6 J( f5 R5 }}   h8 S# G/ z/ c6 p2 X' Q
5、假设要支持自动对焦功能,则在需要的情况下,或者在上述surfaceChanged调用完startPreview函数后,可以调用Camera::autoFocus函数来设置自动对焦回调函数,该步是可选操作,有些设备可能不支持,可以通过Camera::getFocusMode函数查询。代码可以参考如下:
1 y2 K: E9 S& u+ R, I// 自动对焦
" n1 v& g+ ]) X6 ^, N. A* icamera.autoFocus(new AutoFocusCallback() 6 r) g. j/ a+ o8 y& u9 V$ w3 A
{
3 @8 V$ o0 f1 b# m% H/ p@Override 3 ?6 R. }% l/ L$ ^
public void onAutoFocus(boolean success, Camera camera)
. a% ^3 }" `* I1 v# G' ]: k{
  P$ L  l3 E0 N1 C7 c( @if (success) : p; M* c& J+ f" W, k
{ 8 _' |0 i9 _, C( o
// success为true表示对焦成功,改变对焦状态图像 0 i- S9 N$ d2 A4 _, A/ b
ivFocus.setImageResource(R.drawable.focus2);
4 s- o2 D) _' Q& c1 r: X! |6 Z} " V' `8 {5 Y! C8 X" A" G
}
( y/ x8 O  c* n. o- w0 W});
6 {4 X# M1 f0 Y% @/ V" \' Q6、在需要拍照的时候,调用takePicture(Camera.ShutterCallback, Camera.PictureCallback, Camera.PictureCallback, Camera.PictureCallback)函数来完成拍照,这个函数中可以四个回调接口,ShutterCallback是快门按下的回调,在这里我们可以设置播放“咔嚓”声之类的操作,后面有三个PictureCallback接口,分别对应三份图像数据,分别是原始图像、缩放和压缩图像和JPG图像,图像数据可以在PictureCallback接口的void onPictureTaken(byte[] data, Camera camera)中获得,三份数据相应的三个回调正好按照参数顺序调用,通常我们只关心JPG图像数据,此时前面两个PictureCallback接口参数可以直接传null; 5 n! C/ H& a( H5 h3 d5 C
7、每次调用takePicture获取图像后,摄像头会停止预览,假如需要继续拍照,则我们需要在上面的PictureCallback的onPictureTaken函数末尾,再次掉哟更Camera::startPreview函数; ) m* D( C  i; s' J1 d  ]  W7 m
8、在不需要拍照的时候,我们需要主动调用Camera::stopPreview函数停止预览功能,并且调用Camera::release函数释放Camera,以便其他应用程序调用。SDK中建议放在Activity的Pause函数中,但是我觉得放在surfaceDestroyed函数中更好,示例代码如下
, g1 D' d0 K  r# W9 ]: @// 停止拍照时调用该方法 ' D/ W% y* |* J. H/ q* L
public void surfaceDestroyed(SurfaceHolder holder)
: {* H: S$ B0 z$ F+ B* o7 U& j{ ) f+ E# e4 G5 K- _, N# {
// 释放手机摄像头 # M9 R: d6 z( y
camera.release();
+ _# }$ l) m4 h9 [: s0 [6 A}
0 M' S% V4 L1 Q$ I) l/ l

2 R( P: v' Q, Y. G6 q3 _9 e2 K! A3 X, W& e- ]' L0 f$ P  z2 Z4 S! ^
' p1 {4 O' z% G- N
以上就是自己实现拍照程序的的流程,一般还可以还可以获取预览帧的图像数据,可以分别通过Camera::setPreviewCallback和Camera::setOneShotPreviewCallback来设置每帧或下一帧图像数据的回调,这里就不做展开了。 6 _/ J6 T" ?5 G& G  X8 r

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

' b3 v9 B. Z7 j# k  ]
7 R( E4 f3 c- I- p6、开启视频录制,需要创建一个MediaRecorder对象,并调用Camera::unLock操作解锁摄像头,因为默认Camera都是锁定的,只有解锁后MediaRecorder等多媒体进程调用,并设置一些参数,然后调用MediaRecorder:: start开启录制具体可以参阅如下代码:
0 I! p: O! Q: R  m# Q3 xMediaRecorder mMediaRecorder = new MediaRecorder();
" K- b! M' U1 N1 V// Unlock the camera object before passing it to media recorder. ' C: {! H1 O7 F
camera.unlock();
' Q+ `1 v$ t* p- Z3 ZmMediaRecorder.setCamera(camera); 2 c; R* k% E. n; m' r0 ~2 t
mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.CAMCORDER);
6 J* P' R4 N( G. cmMediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA); & S: b! D" n/ X: U
mMediaRecorder.setProfile(mProfile);
! s' X- h$ e% c8 QmMediaRecorder.setMaxDuration(100000);//ms为单位
% ]7 w6 G: x# A5 B1 @0 N& Glong dateTaken = System.currentTimeMillis(); 0 @) O$ G/ f" ]& A# I
Date date = new Date(dateTaken);
( S' ^# M; u) Y* TSimpleDateFormat dateFormat = new SimpleDateFormat(getString(R.string.video_file_name_format)); 1 w: x3 p  j& P+ |
String title = dateFormat.format(date); 2 w( }4 s1 f) @% U9 G
String filename = title + ".3gp"; // Used when emailing. . h% A2 R, |* x/ i7 w
String cameraDirPath = ImageManager.CAMERA_IMAGE_BUCKET_NAME;
3 o- U" ~# Q; ?7 U, Z/ e+ ?String filePath = cameraDirPath + "/" + filename; 5 r/ I/ E- J4 ^1 S) b
File cameraDir = new File(cameraDirPath); ) {7 g- j* @3 e9 q" s
cameraDir.mkdirs();
( Y: c2 f4 K( ~6 a" w" KmMediaRecorder.setOutputFile(filePath); : n2 M& V7 `+ s# `
try { & H5 C! G! a% N1 W
mMediaRecorder.prepare();
  E4 S! M' i- i1 ^5 cmMediaRecorder.start(); // Recording is now started   V- U* a! p5 Z2 c
} catch (RuntimeException e) {
( c0 x# n' e' O% s: O' QLog.e(TAG, "Could not start media recorder. ", e); , i- r# I* w2 v' u/ h. S% `: z5 q
return; 6 P! v; g$ `# l. o0 o8 J
}
) |% V1 g3 N. ?' Y7、上面设置了最大间隔为100s,当100是视频录制结束,录制就会被停止,如果没有设时长和文件大小限制,那么通常需要调用MediaRecorder:: stop函数主动停止视频的录制,并将Camera对象通过lock函数继续加锁,示例代码如下
  @! [0 F- M/ n! `+ h2 y/ ?" i7 J. AmMediaRecorder.stop();
# j. u6 L" q3 P1 ]mMediaRecorder.reset(); % ~) t5 f# ~& D: r; u
mMediaRecorder.release();
0 F" @! ~3 v3 [  |! Z8 s- F) F' rmMediaRecorder = null; : A: z, q% T  [: v
if(camera != null)  $ ^. k+ J; |% |  m9 A
camera.lock();
: B( g& \+ ~2 Y! f3 l+ s3 S

4 y/ H% S. }/ r5 L0 l6 W- I; i6 ?4 V之后的操作根据交互要么重新录制要么就释放Camera对象回到拍照流程的8步骤了。在这里就不做赘述了。

" x0 Z0 x( u( T- e
  g4 s! M. h  G4 d* p2 P) r) l& ~8 D, ?1 o* P3 A

  ?: z  g5 L* s/ e摄像头模组论坛网,行业交流,分享,学习...
6 \/ h& u) P* _0 S: `) Owww.ccm99.com
1 f& P" u9 U- a" x$ ~+ i* z
高级模式
B Color Image Link Quote Code Smilies @朋友 |上传

本版积分规则

在线客服

客服电话

欢迎来电咨询

188-9985 8350

微信关注

手机APP程序:
扫码下载访问

微信公众平台:
摄像头之家公众号

微信小程序:
摄像头小程序

返回顶部

QQ|站点统计|小黑屋|手机版|Archiver|摄像头模组论坛网-摄像头方案网CCM99 ( 粤ICP备18155214号 )

Powered by Discuz! X3.4 Licensed© 2001-2013 Comsenz Inc.