Android如何控制照相机

时间:2021-08-24 20:27:26

本文译自:http://developer.android.com/training/camera/cameradirect.html

本文中,我们讨论如何使用框架API来直接的控制照相机硬件。

直接控制设备等的照相机,比从既存的照相机应用程序中获取图片或视频需要更多的编码。但是,如果你想要创建一个特殊的照相机应用程序,或者要完全与你的应用程序UI集成,那么本文会告诉你如何来做。

打开Camera对象

直接控制照相机的第一步时获取一个Camera对象实例。跟Android自己的Camera应用程序一样,推荐访问照相机的方法是从Activity的onCreate()回调中启动一个独立的线程来打开Camera对象。这是一种好的做法,因为打开Camera对象时需要一些时间,这可能会阻塞UI线程。打开照相机的一个更基本的实现,可延迟到onResume()回调方法中,这有助于代码重用,并保持了简单的控制流。

如果照相机已经被其他的应用程序使用,那么调用Camera.open()方法时会抛出一个异常,因此我们要代码包在在try…catch块中。

privatebooleansafeCameraOpen(int id){
    
boolean qOpened = false;
  
    try{
        releaseCameraAndPreview();
        mCamera = Camera.open(id);
        qOpened = (mCamera != null);
    } catch (Exception e) {
        Log.e(getString(R.string.app_name), "failed to open Camera");
        e.printStackTrace();
    }

    return qOpened;    
}

private void releaseCameraAndPreview() {
    mPreview.setCamera(null);
    if (mCamera != null) {
        mCamera.release();
        mCamera = null;
    }
}

API Level 9开始,照相机框架支持多个相机。如果你使用传统的API,并调用了没有参数的open()方法,就会获得后置的摄像头。

创建Camera预览

通常在点击快门拍照之前,需要预览图片。因此,你可以使用SurfaceView来描绘照相机所扑获的图像。

创建预览类

要显示预览图像,就需要创建一个预览类,这个类要实现android.view.SurfaceHolder.Callback接口,它用于把照相机硬件的图像数据传递给应用程序。

classPreviewextendsViewGroupimplementsSurfaceHolder.Callback{

    
SurfaceView mSurfaceView;
    SurfaceHolder mHolder;

    Preview(Context context) {
        super(context);

        mSurfaceView = new SurfaceView(context);
        addView(mSurfaceView);

        // Install aSurfaceHolder.Callback so we get notified when the
        //underlying surface is created and destroyed.
        mHolder = mSurfaceView.getHolder();
        mHolder.addCallback(this);
        mHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
    }
...
}

这个预览类在显示预览图像之前,必须把它传递给Camera对象。

设置和启动图像预览

一个照相机实例和它相关的预览必须按照特定的顺序来创建,首先是创建 Camera对象。在下面的代码片段中,封装了照相机的初始化过程,这样用户无论何时改变照相机,都可以通过setCamera()方法来调用Camera.startPreviw()方法。这个预览类也必须在预览类的surfaceChanged()回调方法中被重启。

publicvoidsetCamera(Camera camera){
    
if (mCamera == camera) { return; }
    
    stopPreviewAndFreeCamera();
    
    mCamera = camera;
    
    if (mCamera != null) {
        List<Size> localSizes = mCamera.getParameters().getSupportedPreviewSizes();
        mSupportedPreviewSizes = localSizes;
        requestLayout();
      
        try {
            mCamera.setPreviewDisplay(mHolder);
        } catch (IOException e) {
            e.printStackTrace();
        }
      
        /*
          Important: CallstartPreview() to start updating the preview surface. Preview must 
          be started before you cantake a picture.
          */
        mCamera.startPreview();
    }
}

修改照相机设置

照相机设置改变着照相机拍照的方式,从变焦级别到曝光度。这个例子仅改变了预览的尺寸,更多的设置可以看Camera应用程序的源代码。

publicvoidsurfaceChanged(SurfaceHolder holder,int format,int w,int h){
    
// Now that the size isknown, set up the camera parameters and begin
    // the preview.
    Camera.Parameters parameters = mCamera.getParameters();
    parameters.setPreviewSize(mPreviewSize.width, mPreviewSize.height);
    requestLayout();
    mCamera.setParameters(parameters);

    /*
      Important: Call startPreview() to startupdating the preview surface. Preview must be
      started before you can take a picture.
    */
    mCamera.startPreview();
}

设置预览方向

大多数照相机应用程序都会把显示锁定到横屏模式,因为这是照相机的自然方向。这种设置不会妨碍你拍摄纵向的照片,因为设备的方向被记录在EXIF头中。使用setCameraDisplayOrientation()方法会改变预览的显示方式,而不会影响到被记录的图像。但是,在API Level 14以前的Android版本,在改变方向之前,必须要先终止预览,然后再重启。

拍照

图像预览被启动之后,就可以使用Camera.takePicture()方法来拍摄照片。你可以创建Camera.PictureCallbackCamera.ShutterCallback对象,并把它们传递给Camera.takePicture()方法。

如果你要连续的抢拍图像,可以创建一个实现onPreviewFrame()方法的Camera.PreviewCallback接口。在这个方法中,你可以只选择预览的帧来拍照,或者建立一个延迟的动作来调用takePicture()方法。

重启预览

拍摄照片之后,在用户能够继续拍照之前,你必须重启图像预览。在下面的例子中,通过快门按钮来重启图像预览。

@Override
public
 void onClick(View v) {
    switch(mPreviewState) {
    case K_STATE_FROZEN:
        mCamera.startPreview();
        mPreviewState = K_STATE_PREVIEW;
        break;

    default:
        mCamera.takePicture( null, rawCallback, null);
        mPreviewState = K_STATE_BUSY;
    } // switch
    shutterBtnConfig();
}

终止预览并释放Camera对象

一旦你的应用程序使用完照相机,就是清理的时机。尤其是你必须释放Camera对象,否则可能会让其他应用程序崩溃,包括你自己的应用程序的新的实例。

在什么时候终止预览并释放Camera对象呢?图像预览类被销毁时是终止图像预览和释放Camera对象的一个相当好的时机,例如:

publicvoidsurfaceDestroyed(SurfaceHolder holder){
    
// Surface will bedestroyed when we return, so stop the preview.
    if (mCamera != null) {
        /*
          Call stopPreview() to stopupdating the preview surface.
        */
        mCamera.stopPreview();
    }
}

/**
  * When this function returns, mCamera will be null.
  */
private void stopPreviewAndFreeCamera() {

    if (mCamera != null) {
        /*
          Call stopPreview() to stopupdating the preview surface.
        */
        mCamera.stopPreview();
    
        /*
          Important: Call release() torelease the camera for use by other applications. 
          Applications should releasethe camera immediately in onPause() (and re-open() it in
          onResume()).
        */
        mCamera.release();
    
        mCamera = null;
    }
}