Controllare la videocamera

In questa lezione parleremo di come controllare direttamente l'hardware della videocamera le API del framework.

Nota:questa pagina fa riferimento alla classe Fotocamera, che è stata ritirata. Ti consigliamo di utilizzare CameraX o, per casi d'uso specifici, Camera2. Sia CameraX che Camera2 supportano Android 5.0 (livello API 21) e versioni successive.

Il controllo diretto della fotocamera di un dispositivo richiede molto più codice rispetto alla richiesta di immagini o video da applicazioni di fotocamera esistenti. Se però vuoi creare un'applicazione per la videocamera specializzata o qualcosa di completamente integrato nell'interfaccia utente dell'app, questa lezione ti mostra come fare.

Consulta le seguenti risorse correlate:

Apri l'oggetto Fotocamera

Ottieni un'istanza dell'oggetto Camera è il primo passaggio nella per controllare direttamente la videocamera. Così come fa l'applicazione Fotocamera di Android, il modo consigliato per accedere alla fotocamera è aprire Camera in un thread separato lanciato da onCreate(). Questo approccio è una buona idea poiché può richiedere del tempo e potrebbe bloccare il thread dell'interfaccia utente. In un'implementazione più basilare, l'apertura della fotocamera può essere differita al metodo onResume() per facilitare il riutilizzo del codice e mantenere il flusso controllo semplice.

La chiamata a Camera.open() genera un un'altra eccezione se la videocamera è già utilizzata da un'altra applicazione, quindi in un blocco try.

private fun safeCameraOpen(id: Int): Boolean {
    return try {
        releaseCameraAndPreview()
        mCamera = Camera.open(id)
        true
    } catch (e: Exception) {
        Log.e(getString(R.string.app_name), "failed to open Camera")
        e.printStackTrace()
        false
    }
}

private fun releaseCameraAndPreview() {
    preview?.setCamera(null)
    mCamera?.also { camera ->
        camera.release()
        mCamera = null
    }
}
private boolean safeCameraOpen(int id) {
    boolean qOpened = false;

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

    return qOpened;
}

private void releaseCameraAndPreview() {
    preview.setCamera(null);
    if (camera != null) {
        camera.release();
        camera = null;
    }
}

A partire dal livello API 9, il framework della fotocamera supporta più videocamere. Se utilizzi API legacy e chiama open() senza un argomento, si ottiene la prima fotocamera posteriore.

Crea l'anteprima della fotocamera

Per scattare una foto, in genere gli utenti devono visualizzare un'anteprima del soggetto prima di fare clic il pulsante di scatto. Per farlo, puoi utilizzare un SurfaceView per disegnare anteprime di ciò che il sensore della videocamera sta rilevando la situazione.

Anteprima corso

Per iniziare a visualizzare un'anteprima, hai bisogno del corso Anteprima. La richiede l'implementazione dell'interfaccia android.view.SurfaceHolder.Callback, che viene utilizzata per trasmettere immagini dall'hardware della videocamera all'applicazione.

class Preview(
        context: Context,
        val surfaceView: SurfaceView = SurfaceView(context)
) : ViewGroup(context), SurfaceHolder.Callback {

    var mHolder: SurfaceHolder = surfaceView.holder.apply {
        addCallback(this@Preview)
        setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS)
    }
    ...
}
class Preview extends ViewGroup implements SurfaceHolder.Callback {

    SurfaceView surfaceView;
    SurfaceHolder holder;

    Preview(Context context) {
        super(context);

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

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

La classe di anteprima deve essere passata all'oggetto Camera prima della pubblicazione l'anteprima dell'immagine, come illustrato nella sezione successiva.

Imposta e avvia l'anteprima

Un'istanza della videocamera e la relativa anteprima devono essere create in un ambiente in cui l'oggetto fotocamera è il primo. Nello snippet seguente, il processo di inizializzazione della fotocamera viene incapsulato in modo che Camera.startPreview() venga chiamato setCamera(), ogni volta che l'utente fa qualcosa per modificare fotocamera. L'anteprima deve essere riavviata anche nel metodo di callback della classe di anteprima surfaceChanged().

fun setCamera(camera: Camera?) {
    if (mCamera == camera) {
        return
    }

    stopPreviewAndFreeCamera()

    mCamera = camera

    mCamera?.apply {
        mSupportedPreviewSizes = parameters.supportedPreviewSizes
        requestLayout()

        try {
            setPreviewDisplay(holder)
        } catch (e: IOException) {
            e.printStackTrace()
        }

        // Important: Call startPreview() to start updating the preview
        // surface. Preview must be started before you can take a picture.
        startPreview()
    }
}
public void setCamera(Camera camera) {
    if (mCamera == camera) { return; }

    stopPreviewAndFreeCamera();

    mCamera = camera;

    if (mCamera != null) {
        List<Size> localSizes = mCamera.getParameters().getSupportedPreviewSizes();
        supportedPreviewSizes = localSizes;
        requestLayout();

        try {
            mCamera.setPreviewDisplay(holder);
        } catch (IOException e) {
            e.printStackTrace();
        }

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

Modifica le impostazioni della videocamera

Le impostazioni della fotocamera modificano il modo in cui la fotocamera scatta le foto, dallo zoom livello alla compensazione dell'esposizione. Questo esempio modifica solo le dimensioni dell'anteprima, vedere il codice sorgente dell'applicazione Fotocamera per molte altre.

override fun surfaceChanged(holder: SurfaceHolder, format: Int, w: Int, h: Int) {
    mCamera?.apply {
        // Now that the size is known, set up the camera parameters and begin
        // the preview.
        parameters?.also { params ->
            params.setPreviewSize(previewSize.width, previewSize.height)
            requestLayout()
            parameters = params
        }

        // Important: Call startPreview() to start updating the preview surface.
        // Preview must be started before you can take a picture.
        startPreview()
    }
}
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int w, int h) {
    // Now that the size is known, set up the camera parameters and begin
    // the preview.
    Camera.Parameters parameters = mCamera.getParameters();
    parameters.setPreviewSize(previewSize.width, previewSize.height);
    requestLayout();
    mCamera.setParameters(parameters);

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

Imposta l'orientamento dell'anteprima

La maggior parte delle applicazioni della fotocamera blocca il display in modalità Orizzontale, l'orientamento del sensore della fotocamera. Questa impostazione non impedisce di utilizzare la modalità Ritratto foto, perché l'orientamento del dispositivo è registrato nell'intestazione EXIF. Il metodo setCameraDisplayOrientation() ti consente di modificare come viene visualizzata l'anteprima senza influire sulla registrazione dell'immagine. Tuttavia, nelle versioni precedenti di Android al livello API 14, devi interrompere l'anteprima prima di modificare l'orientamento e riavviarla.

Scatta una foto

Usa Camera.takePicture() per scattare una foto dopo l'avvio dell'anteprima. Puoi creare oggetti Camera.PictureCallback e Camera.ShutterCallback e trasmetterli a Camera.takePicture().

Se vuoi acquisire immagini in modo continuo, puoi creare un elemento Camera.PreviewCallback che implementa onPreviewFrame(). Per a una via di mezzo, puoi acquisire solo determinati fotogrammi di anteprima oppure configurare azione ritardata per chiamare takePicture().

Riavviare l'anteprima

Dopo aver scattato una foto, devi riavviare l'anteprima prima che l'utente può scattare un'altra foto. In questo esempio, il riavvio viene eseguito con un sovraccarico il pulsante di scatto.

fun onClick(v: View) {
    previewState = if (previewState == K_STATE_FROZEN) {
        camera?.startPreview()
        K_STATE_PREVIEW
    } else {
        camera?.takePicture(null, rawCallback, null)
        K_STATE_BUSY
    }
    shutterBtnConfig()
}
@Override
public void onClick(View v) {
    switch(previewState) {
    case K_STATE_FROZEN:
        camera.startPreview();
        previewState = K_STATE_PREVIEW;
        break;

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

Interrompi l'anteprima e rilascia la fotocamera

Una volta che l'applicazione ha finito di utilizzare la fotocamera, è il momento di eseguire la pulizia. Nella specifica, devi rilasciare l'oggetto Camera per non provocare l'arresto anomalo di altri delle tue applicazioni, incluse nuove istanze della tua applicazione.

Quando devi interrompere l'anteprima e rilasciare la fotocamera? Il fatto che la tua l'eliminazione della superficie di anteprima è un ottimo indizio che è il momento visualizza l'anteprima e rilascia la fotocamera, come mostrato in questi metodi dalla classe Preview.

override fun surfaceDestroyed(holder: SurfaceHolder) {
    // Surface will be destroyed when we return, so stop the preview.
    // Call stopPreview() to stop updating the preview surface.
    mCamera?.stopPreview()
}

/**
 * When this function returns, mCamera will be null.
 */
private fun stopPreviewAndFreeCamera() {
    mCamera?.apply {
        // Call stopPreview() to stop updating the preview surface.
        stopPreview()

        // Important: Call release() to release the camera for use by other
        // applications. Applications should release the camera immediately
        // during onPause() and re-open() it during onResume()).
        release()

        mCamera = null
    }
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
    // Surface will be destroyed when we return, so stop the preview.
    if (mCamera != null) {
        // Call stopPreview() to stop updating the preview surface.
        mCamera.stopPreview();
    }
}

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

    if (mCamera != null) {
        // Call stopPreview() to stop updating the preview surface.
        mCamera.stopPreview();

        // Important: Call release() to release the camera for use by other
        // applications. Applications should release the camera immediately
        // during onPause() and re-open() it during onResume()).
        mCamera.release();

        mCamera = null;
    }
}

All'inizio della lezione, questa procedura faceva parte del metodo setCamera(), quindi l'inizializzazione di una videocamera inizia sempre con l'interruzione l'anteprima.