Android Camera(三)

来源:互联网 发布:淘一兔淘宝信誉查询 编辑:程序博客网 时间:2024/05/19 02:42

把预览类放到布局中

Camera的预览类,如上文所示的示例,必须要跟用户界面控件一起放到一个Activity的布局中,以便拍照或录像。本段向你显示如果构建一个基本的用于预览的Activity布局。

下面的布局代码提供了一个很基本的View,它能够用于显示Camera的预览图像。在这个示例中,FrameLayout元素被用于Camera预览类的容器。使用这个布局类型可以让额外的图像信息或控件能够覆盖在实时的Camera预览图像之上。

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"

   android:orientation="horizontal"

   android:layout_width="fill_parent"

   android:layout_height="fill_parent"

   >

 <FrameLayout

   android:id="@+id/camera_preview"

   android:layout_width="fill_parent"

   android:layout_height="fill_parent"

   android:layout_weight="1"

   />

 

 <Button

   android:id="@+id/button_capture"

   android:text="Capture"

   android:layout_width="wrap_content"

   android:layout_height="wrap_content"

   android:layout_gravity="center"

   />

</LinearLayout>

在大多数设备上,Camera预览的默认方向是横向。这个示例布局指定了一个水平(横向)布局,并且下面的代码把应用程序固定为横向。为了简化Camera预览中的展现,你应该通过在你的清单文件中添加以下设置,把应用程序的预览Activity的方向改变为横向的:

<activity android:name=".CameraActivity"

         android:label="@string/app_name"

 

         android:screenOrientation="landscape">

         <!-- configure this activity to use landscape orientation -->

 

         <intent-filter>

       <action android:name="android.intent.action.MAIN" />

       <category android:name="android.intent.category.LAUNCHER" />

   </intent-filter>

</activity>

注意:Camera的预览不一定就是横屏模式。从Android2.2(API Level8)开始,你能够使用setDisplayOrientation()方法来设置预览图像的旋转。为了重新定位用户手的预览方向,在你的预览类的surfaceChanged()方法中,首先要用Camera.stopPreview()方法停止预览,然后改变方向,再用Camera.startPreview()方法启动预览。

在Camera View的Activity中,把你的预览类添加到上例所示的FrameLayout元素中。你的Camera Activity还必须确保在它被挂起或关闭时,要释放Camera对象。以下示例显示如何把上文中创建的预览类添加到Camera的Activity中:

public class CameraActivity extends Activity {

 

   private Camera mCamera;

   private CameraPreview mPreview;

 

   @Override

   public void onCreate(Bundle savedInstanceState) {

       super.onCreate(savedInstanceState);

       setContentView(R.layout.main);

 

       // Create an instance of Camera

       mCamera = getCameraInstance();

 

       // Create our Preview view and set it as the content of our activity.

       mPreview = new CameraPreview(this, mCamera);

       FrameLayout preview = (FrameLayout) findViewById(R.id.camera_preview);

       preview.addView(mPreview);

   }

}

注意:上例中的getCameraInstance()方法引用了上文“访问Camera”中的示例方法。

采集图片

一旦你构建了一个预览类,并把它放到了一个View中,你就可以开始用你的应用程序来采集应用程序了。在你的应用程序代码中,必须建立用于响应用户拍照操作的用户界面控件的监听器。

要使用Camera.takePicture()方法来获取一张图片。这个方法需要3个参数来接收来自Camera的数据。为了接收JPEG格式的数据,你必须实现接收图片数据的Camera.PictureCallback接口,并把接收到数据写到一个文件中。下面的代码显示了一个Camera.PictureCallback接口的基本实现,它把从Camera接收到的数据保存到一个图片文件中:

private PictureCallback mPicture = new PictureCallback() {

 

   @Override

   public void onPictureTaken(byte[] data, Camera camera) {

 

       File pictureFile = getOutputMediaFile(MEDIA_TYPE_IMAGE);

       if (pictureFile == null){

           Log.d(TAG, "Error creating media file, check storage permissions: " +

               e.getMessage());

           return;

       }

 

       try {

           FileOutputStream fos = new FileOutputStream(pictureFile);

           fos.write(data);

           fos.close();

       } catch (FileNotFoundException e) {

           Log.d(TAG, "File not found: " + e.getMessage());

       } catch (IOException e) {

           Log.d(TAG, "Error accessing file: " + e.getMessage());

       }

   }

};

通过调用Camera.takePicture()方法来触发采集图片的操作。下面的示例代码显示了如何从一个按钮的View.OnClickListener中调用这个方法:

// Add a listener to the Capture button

Button captureButton = (Button) findViewById(id.button_capture);

captureButton.setOnClickListener(

   new View.OnClickListener() {

       @Override

       public void onClick(View v) {

           // get an image from the camera

           mCamera.takePicture(null, null, mPicture);

       }

   }

);

注意:代码中的mPicture成员变量会在下面的示例代码中引用。

警告:记住,在应用程序使用完Camera对象后,一定要调用Camera.release()方法来释放Camera对象。

采集视频

使用Android框架采集视频需要认真的管理Camera对象,以及跟MediaRecorder类的协调。当使用Camera对象记录视频时,除了Camera.open()和Camera.release()方法的调用以外,还必须管理Camera.lock()和Camera.unlock()方法的调用,从而允许MediaRecorder对象访问Camera硬件。

注意:从Android4.0(API Level 14)开始,Camera.lock()和Camera.unlock()方法调用系统会为你自动管理。

跟拍照不一样,采集视频需要非常特殊的调用顺序,必须按照一个特定执行顺序来准备和采集视频,详细如下:

1. 打开Camera:使用Camera.open()方法来获得一个Camera对象的实例。

2. 连接预览:使用Camera.setPreviewDisplay()方法把SurfaceView跟Camera连接起来,准备一个实时的Camera图像预览界面。

3. 启动预览:调用Camera.starPreview()方法开始显示实时的Camera图像。

4. 启动视频录像:为了成功的记录视频,必须完成以下步骤。

    a. 给Camera对象解锁:通过调用Camera.unlock()方法给由MediaRecorder对象使用的Camera对象解锁。

    b. 配置MediaRecorder对象:在这一步中要调用以下MediaRecorder方法。更多信息,请看MediaRecorder参考文档

       1. setCamera():使用应用程序当前的Camera实例,把它设置成用于视频采集;

       2. setAudiioSource():使用MediaRecorder.AudioSource.CAMCORDER,设置音频源;

       3. setVideoSource():使用MediaRecorder.VideoSource.CAMERA来设置视频源;

       4. 设置视频的输出格式和编码。对于Android2.2以上的版本,要使用MediaRecorder.setProfile方法,并且要私用CamcorderProfile.get()方法来获得一个Profile的示例。对于Android2.2以前的版本,必须要设置视频输出的格式和编码参数:

              I. setOutputFormat():设置输出格式,指定默认设置,或者是MediaRecorder.OutputFormat.MPEG_4;

             II. setAudioEncoder():设置声音的编码类型,指定默认设置,或者是MediaRecorder.AudioEncoder.AMR_NB;

             III.setVideoEncoder():设置视频的编码类型,指定默认设置,或者是MediaRecorder.VideoEncoder.MPEG_4_SP;

        5. setOutputFile():设置输出文件,可以使用下文“保存媒体文件”中的示例代码中的getOutputMediaFile(MEDIA_TYPE_VIDEO).toString()方法来获取输出文件路径。

        6. setPreviewDisplay():给应用程序指定SurfaceView的预览布局。要使用与连接预览相同的对象。

        警告:这步中必须要调用MediaRecorder的配置方法,否则你的应用程序将会发生错误,并且录像也会失败。

    c. 准备MediaRecorder对象:通过调用MediaRecorder.prepare()方法,准备给MediaRecorder对象提供的配置设置。

    d. 启动MediaRecorder对象:通过调用MediaRecorder.star()方法来记录视频。

5. 停止视频录像:在这步中要调用以下方法来成功完成视频记录:

    a. 停止MediaRecorder对象:通过调用MediaRecorder.stop()方法来停止视频采集

    b. 重置MediaRecorder对象:可选操作,通过调用MediaRecorder.reset()方法来删除记录器的配置设置。

    c. 释放MediaRecorder对象:调用MediaRecorder.release()方法来释放MediaRecorder对象。

    d. 锁定Camera对象:调用Camera.lock()方法来锁定Camera对象,以便MediaRecorder会话能够继续使用它。从Android4.0(API Level 14)开始,这个调用不再需要了,除非MediaRecorder.prepare()方法调用失败。

6. 停止预览:当Activity使用完Camera对象时,要使用Camera.stopPreview()方法来停止预览。

7. 释放Camera对象:调用Camera.release()方法来释放Camera对象,以便其他应用程序能够使用它。

注意:使用MediaRecorder对象,不首先创建Camera预览,并跳过这个过程开始的几步是可能的。但是,因为典型的在开始视频采集之前,要给用户看到图像预览,所以跳过的步骤It is possible to useMediaRecorder without creating a camera preview first and skip the first few steps of this process.在此就不讨论了。

提示:如果你的应用程序被用于典型的视频记录,那么在启动预览期间,把setRecordingHint(boolean)方法设置为true,这样有助于减少启动采集所需要的时间。

 

配置MediaRecorder对象

在使用MediaRecorder类来采集视频时,必须按照指定的顺利来执行配置处理,并且要调用MediaRecorder.prepare()方法来检查和实现配置。下面的示例代码演示了如何为采集视频配置和准备MediaRecorder类:

private boolean prepareVideoRecorder(){

 

   mCamera = getCameraInstance();

   mMediaRecorder = new MediaRecorder();

 

   // Step 1: Unlock and set camera to MediaRecorder

   mCamera.unlock();

   mMediaRecorder.setCamera(mCamera);

 

   // Step 2: Set sources

   mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.CAMCORDER);

   mMediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA);

 

   // Step 3: Set a CamcorderProfile (requires API Level 8 or higher)

   mMediaRecorder.setProfile(CamcorderProfile.get(CamcorderProfile.QUALITY_HIGH));

 

   // Step 4: Set output file

   mMediaRecorder.setOutputFile(getOutputMediaFile(MEDIA_TYPE_VIDEO).toString());

 

   // Step 5: Set the preview output

   mMediaRecorder.setPreviewDisplay(mPreview.getHolder().getSurface());

 

   // Step 6: Prepare configured MediaRecorder

   try {

       mMediaRecorder.prepare();

   } catch (IllegalStateException e) {

       Log.d(TAG, "IllegalStateException preparing MediaRecorder: " + e.getMessage());

       releaseMediaRecorder();

       return false;

   } catch (IOException e) {

       Log.d(TAG, "IOException preparing MediaRecorder: " + e.getMessage());

       releaseMediaRecorder();

       return false;

   }

   return true;

}

在Android2.2(API Level8)之前,必须要直接设置输出格式和编码格式参数,而不是使用CamcorderProfile类。下面的代码演示了这种方法:

// Step 3: Set output format and encoding (for versions prior to API Level 8)

   mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4);

   mMediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.DEFAULT);

   mMediaRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.DEFAULT);

以下是给MediaRecorder对象设置的默认的视频采集参数,但是你可以在你的应用程序中来调整这些设置:

1. setVideoEncodingBitRate()

2. setVideoSize()

3. setVideoFrameRate()

4. setAudioEncodingBitRate()

5. setAudioChannels()

6. setAudioSamplingRate()

启动和停止MediaRecorder对象

在使用MediaRecorder类来启动和停止视频采集时,必须按照以下顺序来执行:

1. 用Camera.unlock()方法来解锁Camera对象;

2. 向上面代码中显示的那样来配置MediaRecorder

3. 使用MediaRecorder.start()方法来启动采集;

4. 记录视频;

5. 使用MediaRecorder.stop()方法来停止采集;

6. 使用MediaRecorder.release()方法来释放媒体记录器;

7. 使用Camera.lock()方法来锁定Camera对象。

以下示例代码演示了如何触发一个按钮正确的使用Camera和MediaRecorder类来启动和停止视频采集。

注意:当完成视频采集时,不要释放Camera,否则你的预览将会被停止。

private boolean isRecording = false;

 

// Add a listener to the Capture button

Button captureButton = (Button) findViewById(id.button_capture);

captureButton.setOnClickListener(

   new View.OnClickListener() {

       @Override

       public void onClick(View v) {

           if (isRecording) {

               // stop recording and release camera

               mMediaRecorder.stop();  // stop the recording

               releaseMediaRecorder(); // release the MediaRecorder object

               mCamera.lock();         // take camera access back from MediaRecorder

 

               // inform the user that recording has stopped

               setCaptureButtonText("Capture");

               isRecording = false;

           } else {

               // initialize video camera

               if (prepareVideoRecorder()) {

                   // Camera is available and unlocked, MediaRecorder is prepared,

                   // now you can start recording

                   mMediaRecorder.start();

 

                   // inform the user that recording has started

                   setCaptureButtonText("Stop");

                   isRecording = true;

               } else {

                   // prepare didn't work, release the camera

                   releaseMediaRecorder();

                   // inform user

               }

           }

       }

   }

);

注意:在上面的例子中,prepareVideoRecorder()方法引用了“配置MediaRecorder”一节中的示例代码。这个方法要锁定Camera,然后配置和准备MediaRecorder示例。

释放Camera

Camera是一种由应用程序共享的设备资源。在获得一个Camera实例之后,你的应用程序才能够使用Camera,并且要特别小心,在应用程序停止使用它的时候,即使是是应用程序被挂起(Activity.onPause()),一定要释放Camera对象。如果应用程序没有正确的释放Camera,所有后续的访问Camera的尝试,包括你自己的应用程序,都会因访问失败而导致应用程序被关闭。

像下面的示例代码那样,使用Camera.release()方法来释放Camera对象:

public class CameraActivity extends Activity {

   private Camera mCamera;

   private SurfaceView mPreview;

   private MediaRecorder mMediaRecorder;

 

   ...

 

   @Override

   protected void onPause() {

       super.onPause();

       releaseMediaRecorder();       // if you are using MediaRecorder, release it first

       releaseCamera();              // release the camera immediately on pause event

   }

 

   private void releaseMediaRecorder(){

       if (mMediaRecorder != null) {

           mMediaRecorder.reset();   // clear recorder configuration

           mMediaRecorder.release(); // release the recorder object

           mMediaRecorder = null;

           mCamera.lock();           // lock camera for later use

       }

   }

 

   private void releaseCamera(){

       if (mCamera != null){

           mCamera.release();        // release the camera for other applications

           mCamera = null;

       }

   }

}