查看: 10605|回复: 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做个简单的使用小结。( K7 [& Y5 H7 M1 r6 f+ s! H' d
, m% {+ C% j) H5 ?
调用系统Camera App实现拍照和摄像功能

' J! r/ P+ r. m/ ?    不是专门的Camera应用,一般用到Camera的需求就是获取照片或者视频,比如微博分享、随手记等,对于在Symbian系统上通过简单地调用系统自带的Camera APP来实现该功能是做不到的,但是Android系统强大的组件特性,使得应用开发者只需通过Intent就可以方便的打开系统自带的Camera APP,并通过MediaStroe方便地获取照片和视频的文件路径。具体我们还是用代码来说话吧:
: a- Y4 p- @. X& j例1、 实现拍照
# a# Z& n7 K8 W( }在菜单或按钮的选择操作中调用如下代码,开启系统自带Camera APP,并传递一个拍照存储的路径给系统应用程序,具体如下:
+ e2 h; l$ R! [; Z& ^; `$ cimgPath = "/sdcard/test/img.jpg"; 1 K4 m9 H; @4 b7 A; X& k( {
//必须确保文件夹路径存在,否则拍照后无法完成回调
7 ?* u/ b1 K* l: `% B3 k: X7 _File vFile = new File(imgPath); + o$ x) |$ f9 t( U& z( y
if(!vFile.exists()) , N/ |: S" z/ n* J* z4 r/ d. ]
{
( X. }& }5 G" H0 nFile vDirPath = vFile.getParentFile(); //new File(vFile.getParent()); 3 v" g. d2 t# `# [( ?
vDirPath.mkdirs();
5 j5 Z! V' ^6 c: `2 P) C}
* h1 J3 S( V# o3 i# r1 ~Uri uri = Uri.fromFile(vFile);
2 x( b! ?( h! PIntent intent = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); 9 b7 A! B3 l+ O# K
intent.putExtra(MediaStore.EXTRA_OUTPUT, uri);// 0 Q7 x8 K/ s- p+ J$ u5 s8 q+ t( v
startActivityForResult(intent, SystemCapture); " u% z1 V8 i+ z( d9 _. k
上面我们使用的是startActivityForResult,所以最好需要重载void onActivityResult(int requestCode, int resultCode, Intent data)函数,不过因为当传入文件路径的的情况下,data返回参数是null值,只要resultCode为RESULT_OK,则上述代码中/sdcard/test/img.jpg的图片文件就是最新的照片文件。所以我们在这里只需给出如下简单的代码,将其显示到ImageView中
/ N. p5 A' V! Bif (resultCode == RESULT_OK)  
: D8 t6 P8 K7 [" x0 R7 I6 z{ / ?% ?. ?) ?# ^8 U
iViewPic.setImageURI(Uri.fromFile(new File(imgPath))); - |. N) [6 E# I# j: S6 W
} # S5 r; [6 g1 U4 N3 p4 m
假设不传参数MediaStore.EXTRA_OUTPUT的情况下,onActivityResult函数在resultCode为RESULT_OK的情况下,data返回的参数是经过实际拍摄照片经过缩放的图像数据,可以通过类似如下方法来打印缩放图像的尺寸
1 I: `/ _/ f6 b/ j9 ]if (resultCode == RESULT_OK)  ! `& n9 D/ j7 @8 N
{ 9 B( v4 y. k1 K$ w
Bitmap bmp = (Bitmap)data.getExtras().get("data");
( P& o6 }1 t* s; p0 YLog.d("Test", "bmp width:" + bmp.getWidth() + ", height:" + bmp.getHeight());
; w5 z( |4 ]; ]; P# b}
0 c+ S" ?4 G' `. \$ T6 U另外假如仅仅是调用系统照相机拍照,不关心拍照结果,则可以简单使用如下代码 ; \  @- h' o; w% X* {; Q( M5 z, ?
Intent intent = new Intent(); //调用照相机 " {+ Z; \) R. O; A
intent.setAction("android.media.action.STILL_IMAGE_CAMERA");
/ ]4 |: p1 L, R5 o; v" u* N+ ]startActivity(intent); * s3 ^& _. M$ ?8 B( F% W" q
备注:上面设置MediaStore.EXTRA_OUTPUT的方法,经过手机实测除了我们设定的路径下有照片外,在手机存储卡上也会保存一份照片,默认目录为sdcard/dcim/camera下面,我曾经尝试着想如果每次返回可以取得sdcard/dcim/camera下面的路径就好了,但是目前看来没办法直接获得,可以借助MediaStroe每次去查询最后一条照片记录,应该也是可行的。
+ r  ^. |0 t# `; i& I, n" }2 j例2、 实现摄像
- _0 H, X# X. ?) s在摄像功能时,尝试着设置MediaStore.EXTRA_OUTPUT以传入类似拍照时的文件路径,结果在我的测试真机上,那个视频文件居然是一个0k的空文件,最后通过类似如下代码实现
+ P- ?( m4 O% C5 pIntent intent = new Intent(MediaStore.ACTION_VIDEO_CAPTURE);
. P% s- e; T6 \" Z% P) W) W! Zintent.putExtra(MediaStore.EXTRA_VIDEO_QUALITY, 1);//参数设置可以省略
) w7 F( z- J1 I! sstartActivityForResult(intent, SystemVideoRecord);
( p; \7 c8 ^$ s# N9 ~; a在onActivityResult函数中进行如下代码调用
" R, L' m. \- g# u/ P7 bUri videoUri = data.getData(); % j5 @+ ]: [* c2 x- n* X
//String[] projection = { MediaStore.Video.Media.DATA, MediaStore.Video.Media.SIZE }; 2 a: Z: C) F- N) Y
Cursor cursor = managedQuery(videoUri, null, null, null, null); & Q- ^( [) d* B8 y1 P
cursor.moveToFirst();//这个必须加,否则下面读取会报错
" B$ i# J( U0 _; e$ i& Iint num = cursor.getCount();
) ~' V8 G- I9 T! f, cString recordedVideoFilePath = cursor.getString(cursor.getColumnIndex(MediaStore.Video.Media.DATA));
, G: I& X' w+ c: R+ |int recordedVideoFileSize = cursor.getInt(cursor.getColumnIndex(MediaStore.Video.Media.SIZE));
& j0 V' Z. w2 P/ xiResultText.setText(recordedVideoFilePath);
5 T0 ?! F: h4 j7 B- P) P  g3 aLog.i("videoFilePath", recordedVideoFilePath);
2 w$ r0 y1 f8 K% T( A" bLog.i("videoSize", ""+recordedVideoFileSize); : I( ?% @1 v2 {& a
上面的返回参数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,但是该文件居然是空白内容(不知道是不是跟手机有关,也没有在其它手机上验证过)。
, ~8 F+ {/ s! W6 S/ P* V根据Camera API实现自己的拍照和摄像程序 通过上面对调用系统Camera App实现拍照和摄像功能的例子,我们发现虽然能够满足我们的需求,但是毕竟自由度降低了,而且拍照的界面就是系统的样子,现在很多拍照程序,比如火爆的Camera 360软件等,就需要根据SDK提供的Camera API来编写自己的程序。 . X# w7 U  _- E( W% f* `- K* k+ \
准备工作 上面调用系统Camera App,我们压根不需要任何权限,但是这里用Camera API,就必须在manifest内声明使用权限,通常由以下三项
6 e  \$ o* M0 w<uses-permission android:name = "android.permission.CAMERA" /> " n& u1 J+ r4 ?- L
<uses-feature android:name = "android.hardware.camera" />
0 L, w; s/ n$ }/ j7 z4 m<uses-feature android:name = "android.hardware.camera.autofocus" />
+ s9 P) w# b  a/ e. J3 y. d/ ?一般拍照和摄像的时候需要写到sd卡上,所以还有一向权限声明如下
! t& d9 y% ^, Y; [3 p* E) y<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
0 c6 T' g6 t. h% T/ q& ^1 B! Y真做摄像功能时,需要音频录制和视频录制功能,所以又需要下面两项权限声明 ) N8 `( S  }: [* [$ T
<uses-permission android:name="android.permission.RECORD_VIDEO"/> $ @3 a; k% a) i& Z  A9 x
<uses-permission android:name="android.permission.RECORD_AUDIO"/>
4 l8 A9 ^9 K, b/ \& o另外使用Camera API拍照或摄像,都需要用到预览,预览就要用到SurfaceView,为此Activity的布局中必须有SurfaceView。

% [5 F( O* w6 e2 l
5 x' Z" `3 i8 g( V" E
8 f! K4 N& J8 n3 [0 ]: ?+ a. M. K  L拍照流程 上面简单介绍了下准备工作,下面结合拍照过程中的需要用到的API对拍照流程做下简单描述
8 }" q! T" f1 H. {1、在Activity的OnCreate函数中设置好SurfaceView,包括设置SurfaceHolder.Callback对象和SurfaceHolder对象的类型,具体如下 9 D; j3 X8 _! E' D; [
SurfaceView mpreview = (SurfaceView) this.findViewById(R.id.camera_preview); + |* t- Y4 _& V5 w0 U% ~/ g' p  j& S4 K/ y
SurfaceHolder mSurfaceHolder = mpreview.getHolder();
0 L: b7 n" X0 vmSurfaceHolder.addCallback(this);
7 D  z5 s* d3 o" a5 r) cmSurfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS); 0 D6 s) D! ?8 S0 W- `, Z) D9 l
2、在SurfaceHolder.Callback的surfaceCreated函数中,使用Camera的Open函数开机摄像头硬件,这个API在SDK 2.3之前,是没有参数的,2.3以后支持多摄像头,所以开启前可以通过getNumberOfCameras先获取摄像头数目,再通过getCameraInfo得到需要开启的摄像头id,然后传入Open函数开启摄像头,假如摄像头开启成功则返回一个Camera对象,否则就抛出异常; ) s) a# w- N/ \& g
3、开启成功的情况下,在SurfaceHolder.Callback的surfaceChanged函数中调用getParameters函数得到已打开的摄像头的配置参数Parameters对象,如果有需要就修改对象的参数,然后调用setParameters函数设置进去(SDK2.2以后,还可以通过Camera::setDisplayOrientation设置方向); $ z* h3 t1 P7 Q, j6 S* k% H. E
4、同样在surfaceChanged函数中,通过Camera::setPreviewDisplay为摄像头设置SurfaceHolder对象,设置成功后调用Camera::startPreview函数开启预览功能,上面3,4两步的代码可以如下所示 . I7 W6 E! s0 i- R+ z
public void surfaceChanged(SurfaceHolder holder, int format, int w, int h)
4 |  }5 z- A$ I( k, b% e: M{
+ T, {5 K9 b7 \8 m# }//已经获得Surface的width和height,设置Camera的参数 2 k8 t2 u* e+ j, G5 m4 u
Camera.Parameters parameters = camera.getParameters();
; T9 l" E, @; J, ^0 `$ ?parameters.setPreviewSize(w, h); % x! p  z# [; z& o+ g
List<Size> vSizeList = parameters.getSupportedPictureSizes();
& U6 G- W2 F2 ^( rfor(int num = 0; num < vSizeList.size(); num++) % S' d4 l9 z! ~7 M# G  X
{
; z3 M& M! X* V) wSize vSize = vSizeList.get(num); & N% z1 Z* _" \% T& Q9 j7 G  C
}
& f9 d7 d! g, G" i1 w4 mif(this.getResources().getConfiguration().orientation != Configuration.ORIENTATION_LANDSCAPE) ! }5 U& I, E0 c6 S, o9 J' `6 ~
{
2 @* [" W3 K; {) {//如果是竖屏
2 Q8 P. C% w! r7 wparameters.set("orientation", "portrait");
. Y( g  M! I1 e1 X, I; d4 M//在2.2以上可以使用 & j& W8 U$ ?1 L1 A
//camera.setDisplayOrientation(90);
- c! o$ F( e% {5 Z! N. `}
6 m9 d  G6 t% M7 d* Y0 eelse , A* [3 R3 |$ I
{
0 q4 s' l2 g# L+ B7 G/ P/ S/ i' kparameters.set("orientation", "landscape");
) u( Y5 i: T  s//在2.2以上可以使用
' r5 E* _8 h: w7 S- z% E, w//camera.setDisplayOrientation(0); ) `, f# M5 Y2 c& @, r9 n  F6 o4 W
} 5 a5 ?1 Z9 F4 l) \+ T; X
camera.setParameters(parameters);
5 I) o* Z" K, ztry { % z" w6 O0 I1 ~! ^
//设置显示
8 ^" y# x: R# w1 c5 ccamera.setPreviewDisplay(holder);
) r, B# ^! _, {' n. f& j( S} catch (IOException exception) { 1 G/ i- t1 `6 U9 _) h" D
camera.release(); ( I' U0 D  v: d1 j# c
camera = null; ; G- {! t: |; l2 w/ a+ F. u9 U* C: ?
}
% L* l# G7 u4 }" V7 W- G2 {* |* v6 x//开始预览
* w$ G" H1 X2 |* A9 S  @camera.startPreview();
( f; e* @% a  a# y+ Q}
+ B: _1 Y  i+ k* e" @* J5、假设要支持自动对焦功能,则在需要的情况下,或者在上述surfaceChanged调用完startPreview函数后,可以调用Camera::autoFocus函数来设置自动对焦回调函数,该步是可选操作,有些设备可能不支持,可以通过Camera::getFocusMode函数查询。代码可以参考如下: ! P( F9 Q5 W& w4 u3 d" \
// 自动对焦
  k% a1 }9 a* j9 ~- _1 |camera.autoFocus(new AutoFocusCallback() 7 m% h. R; \- T3 N, I
{
% l/ _8 @# q2 ]; o- ~1 C! O@Override
0 j) Y+ N8 u& [9 e8 ]public void onAutoFocus(boolean success, Camera camera)
9 y3 o: H" W; A( s6 ]! E* ^{ " M+ k/ R6 s; W2 k  s3 M! {9 N
if (success)
  c  _8 [) v# q  D; {8 E, [0 ^{
( t  Q1 q- D6 a* G// success为true表示对焦成功,改变对焦状态图像
/ h! B3 j, e  }* g2 b' iivFocus.setImageResource(R.drawable.focus2);
; F) _1 u" B3 y  k# L8 _} 0 B5 M+ {  _7 X. ?- j, x
} 2 B/ C( v9 w% k
}); * v" p0 [6 P0 B2 D2 E
6、在需要拍照的时候,调用takePicture(Camera.ShutterCallback, Camera.PictureCallback, Camera.PictureCallback, Camera.PictureCallback)函数来完成拍照,这个函数中可以四个回调接口,ShutterCallback是快门按下的回调,在这里我们可以设置播放“咔嚓”声之类的操作,后面有三个PictureCallback接口,分别对应三份图像数据,分别是原始图像、缩放和压缩图像和JPG图像,图像数据可以在PictureCallback接口的void onPictureTaken(byte[] data, Camera camera)中获得,三份数据相应的三个回调正好按照参数顺序调用,通常我们只关心JPG图像数据,此时前面两个PictureCallback接口参数可以直接传null;
, P1 U0 Q9 t6 D) G7、每次调用takePicture获取图像后,摄像头会停止预览,假如需要继续拍照,则我们需要在上面的PictureCallback的onPictureTaken函数末尾,再次掉哟更Camera::startPreview函数;
4 M; q, ?- L  y/ [0 G4 }: T+ A& c* d8、在不需要拍照的时候,我们需要主动调用Camera::stopPreview函数停止预览功能,并且调用Camera::release函数释放Camera,以便其他应用程序调用。SDK中建议放在Activity的Pause函数中,但是我觉得放在surfaceDestroyed函数中更好,示例代码如下 ' X3 |2 t' k  q8 w/ Z* B
// 停止拍照时调用该方法
; ~: n$ J5 q0 A1 q/ npublic void surfaceDestroyed(SurfaceHolder holder)
' u4 A; n  p- I7 U{ 0 `1 ~0 B( I; W7 W7 a
// 释放手机摄像头
0 w! J$ y; v2 o1 F; b6 M  i3 Fcamera.release();
- L* X4 f% p2 X3 [0 s}
$ f  }% x$ b: t( Q9 X7 `; [( U7 u

* z9 G& [9 e# u) p+ c+ H9 Y* A  a, T% z( [$ b; u
  ]1 q5 G8 W; p) F0 P/ F1 l& o* v. f
以上就是自己实现拍照程序的的流程,一般还可以还可以获取预览帧的图像数据,可以分别通过Camera::setPreviewCallback和Camera::setOneShotPreviewCallback来设置每帧或下一帧图像数据的回调,这里就不做展开了。
. |7 `! s  @1 r7 N5 Q6 b" k' k* l! j8 r: g8 _6 D/ y9 M, t
摄像流程 摄像流程也是需要预览的,而且流程上与拍照流程在起始的1~4步流程和结束的8流程是一样的,唯一不同的是6和7两个步骤,至于5自动对焦本身就是可选的,在摄像流程也没必要。
- E6 R& b8 u) M

4 W1 M" l" L2 b  J6、开启视频录制,需要创建一个MediaRecorder对象,并调用Camera::unLock操作解锁摄像头,因为默认Camera都是锁定的,只有解锁后MediaRecorder等多媒体进程调用,并设置一些参数,然后调用MediaRecorder:: start开启录制具体可以参阅如下代码:
: V8 h. G8 d  D7 g7 f/ ^MediaRecorder mMediaRecorder = new MediaRecorder();
% [* f* ]* `- Z5 u// Unlock the camera object before passing it to media recorder. 4 o* d, T1 i- N: {3 `: R
camera.unlock();
6 D6 W+ n* {8 _' z6 U4 l' \mMediaRecorder.setCamera(camera);
( \6 o% A  G7 i7 t( dmMediaRecorder.setAudioSource(MediaRecorder.AudioSource.CAMCORDER); 8 l) j, c1 G$ |+ D
mMediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);
+ n9 {+ q. a- u  g' K" L& q5 u6 HmMediaRecorder.setProfile(mProfile); * s+ h' {1 d' a0 t8 Z* |. J  I
mMediaRecorder.setMaxDuration(100000);//ms为单位
5 `7 ]1 `7 _) E. Plong dateTaken = System.currentTimeMillis();
& r2 A# r$ |# Y3 o% |+ X8 j' ^Date date = new Date(dateTaken); + @* V7 a3 I* N
SimpleDateFormat dateFormat = new SimpleDateFormat(getString(R.string.video_file_name_format)); - D8 T) B7 f9 h8 F' i, d6 ~
String title = dateFormat.format(date);
8 q, Z3 N7 Y- F7 k) U# K0 Z4 pString filename = title + ".3gp"; // Used when emailing.
! ]( i4 \, A9 NString cameraDirPath = ImageManager.CAMERA_IMAGE_BUCKET_NAME; 7 ?; M$ o8 `1 v3 _# B4 l* N4 i4 n
String filePath = cameraDirPath + "/" + filename;
: H9 w: T. v" gFile cameraDir = new File(cameraDirPath);
- W1 l3 V8 W# gcameraDir.mkdirs();
& n  s- i- Y/ {$ y' k) NmMediaRecorder.setOutputFile(filePath);
) I' ]7 Q% n* _try {
# o2 [) R& f" i+ W. h$ UmMediaRecorder.prepare(); 4 u, i6 v' {8 v5 ^& }  M
mMediaRecorder.start(); // Recording is now started : i- K7 W/ S: e) m
} catch (RuntimeException e) {
1 ?' G5 H# @: {. c, QLog.e(TAG, "Could not start media recorder. ", e);
- G5 @4 }9 k! \return;
. K0 q* _2 b0 C. w8 G9 w! b! E# F}
1 A, z7 q4 X% v0 P0 J7、上面设置了最大间隔为100s,当100是视频录制结束,录制就会被停止,如果没有设时长和文件大小限制,那么通常需要调用MediaRecorder:: stop函数主动停止视频的录制,并将Camera对象通过lock函数继续加锁,示例代码如下 ; p" l3 E$ E# a# m
mMediaRecorder.stop(); $ c# s8 \7 g$ L
mMediaRecorder.reset();
' r; x" j2 D* X8 n% ~mMediaRecorder.release();
- L9 M7 y$ ^* R. }4 b  E0 AmMediaRecorder = null;
0 C! ?) p$ k4 Gif(camera != null)  
1 f, T+ x* M" S% [; A  }camera.lock();
7 n! c9 k/ \. n" g7 J; B

+ t- t* q" p" L1 [9 k1 v之后的操作根据交互要么重新录制要么就释放Camera对象回到拍照流程的8步骤了。在这里就不做赘述了。

4 ?! F, q! L! n7 A5 s' v8 A3 \3 a
* n, @7 O2 `7 \& C/ n+ u# x/ R0 e; A( w, L9 h$ u( m5 t
5 p! u$ ]6 n2 U
摄像头模组论坛网,行业交流,分享,学习...
) J0 g+ ^7 j/ }6 F, t. S6 v3 {www.ccm99.com0 W; b! a* _3 k- j- i( I* m$ ^3 `
高级模式
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.