SurfaceRequest


public final class SurfaceRequest


A completable, single-use request of a Surface.

Contains requirements for surface characteristics along with methods for completing the request and listening for request cancellation.

Acts as a bridge between the surface provider and the surface requester. The diagram below describes how it works:

  1. The surface provider gives a reference to surface requester for providing Surface (e.g. setSurfaceProvider).
  2. The surface requester uses the reference to send a SurfaceRequest to get a Surface (e.g. onSurfaceRequested).
  3. The surface provider can use provideSurface to provide a Surface or inform the surface requester no Surface will be provided with willNotProvideSurface. If a Surface is provided, the connection between surface provider and surface requester is established.
  4. If the connection is established, the surface requester can get the Surface through getDeferrableSurface and start to send frame data.
  5. If for some reason the provided Surface is no longer valid (e.g. when the SurfaceView destroys its surface due to page being slid out in ViewPager2), the surface provider can use invalidate method to inform the surface requester and the established connection will be closed.
  6. The surface requester will re-send a new SurfaceRequest to establish a new connection.

Summary

Nested types

@AutoValue
public abstract class SurfaceRequest.Result

Result of providing a surface to a SurfaceRequest via provideSurface.

Transformation associated the preview output.

Listener that receives updates of the TransformationInfo associated with the SurfaceRequest.

Public methods

void
addRequestCancellationListener(
    @NonNull Executor executor,
    @NonNull Runnable listener
)

Adds a listener to be informed when the camera cancels the surface request.

void

Clears the TransformationInfoListener set via setTransformationInfoListener.

@NonNull DynamicRange

Returns the dynamic range expected to be used with the requested surface.

@NonNull Size

Returns the resolution of the requested Surface.

boolean

Invalidates the previously provided Surface to provide a new Surface.

void
provideSurface(
    @NonNull Surface surface,
    @NonNull Executor executor,
    @NonNull Consumer<SurfaceRequest.Result> resultListener
)

Completes the request for a Surface if it has not already been completed or cancelled.

void

Sets a listener to receive updates on transformation info.

boolean

Signals that the request will never be fulfilled.

Public methods

addRequestCancellationListener

Added in 1.0.0
public void addRequestCancellationListener(
    @NonNull Executor executor,
    @NonNull Runnable listener
)

Adds a listener to be informed when the camera cancels the surface request.

A surface request may be cancelled by the camera if the surface is no longer required. Examples of why cancellation may occur include (1) a UseCase being unbound from the camera, (2) surface requirements, such as resolution, changing due to newly bound use cases, or (3) the camera requiring a new Surface object on lower API levels to work around compatibility issues.

When a request is cancelled, the Runnables provided here will be invoked on the Executor they are added with, and can be used as a signal to stop any work that may be in progress to fulfill the surface request.

Once a surface request has been cancelled by the camera, willNotProvideSurface will have no effect and will return false. Attempting to complete the request via provideSurface will also have no effect, and any resultListener passed to provideSurface will be invoked with a Result containing RESULT_REQUEST_CANCELLED.

Note that due to the asynchronous nature of this listener, it is not guaranteed that the listener will be called before an attempt to complete the request with provideSurface or willNotProvideSurface, so the return values of these methods can always be checked if your workflow for producing a surface expects them to complete successfully.

Parameters
@NonNull Executor executor

The executor used to notify the listener.

@NonNull Runnable listener

The listener which will be run upon request cancellation.

clearTransformationInfoListener

Added in 1.0.0
public void clearTransformationInfoListener()

Clears the TransformationInfoListener set via setTransformationInfoListener.

getDynamicRange

Added in 1.3.0
public @NonNull DynamicRange getDynamicRange()

Returns the dynamic range expected to be used with the requested surface.

The dynamic range may have implications for which surface type is returned. Special care should be taken to ensure the provided surface can support the requested dynamic range. For example, if the returned dynamic range has getBitDepth equal to BIT_DEPTH_10_BIT, then the surface provided to provideSurface should use an android.graphics.ImageFormat that can support ten bits of dynamic range, such as PRIVATE or YCBCR_P010.

The dynamic range returned here will always be fully specified. That is, it will never have an encoding of ENCODING_UNSPECIFIED or ENCODING_HDR_UNSPECIFIED and will never have bit depth of BIT_DEPTH_UNSPECIFIED.

getResolution

Added in 1.0.0
public @NonNull Size getResolution()

Returns the resolution of the requested Surface.

The surface which fulfills this request must have the resolution specified here in order to fulfill the resource requirements of the camera. Fulfillment of the request with a surface of a different resolution may cause the resultListener passed to provideSurface to be invoked with a Result containing RESULT_INVALID_SURFACE.

Returns
@NonNull Size

The guaranteed supported resolution.

invalidate

Added in 1.3.0
public boolean invalidate()

Invalidates the previously provided Surface to provide a new Surface.

Call this method to inform the surface requester that the previously provided Surface is no longer valid (e.g. when the SurfaceView destroys its surface due to page being slid out in ViewPager2) and should re-send a SurfaceRequest to obtain a new Surface.

Calling this method will cause the camera to be reconfigured. The app should call this method when the surface provider is ready to provide a new Surface. (e.g. a SurfaceView's surface is created when its window is visible.)

If the provided Surface was already invalidated, invoking this method will return false, and will have no effect. The surface requester will not be notified again, so there will not be another SurfaceRequest.

Calling this method without provideSurface (regardless of whether @link #willNotProvideSurface()} has been called) will still trigger the surface requester to re-send a SurfaceRequest.

Since calling this method also means that the SurfaceRequest will not be fulfilled, if the SurfaceRequest has not responded, it will respond as if calling willNotProvideSurface.

Returns
boolean

true if the provided Surface is invalidated or false if it was already invalidated.

provideSurface

Added in 1.0.0
public void provideSurface(
    @NonNull Surface surface,
    @NonNull Executor executor,
    @NonNull Consumer<SurfaceRequest.Result> resultListener
)

Completes the request for a Surface if it has not already been completed or cancelled.

Once the camera no longer needs the provided surface, the resultListener will be invoked with a Result containing RESULT_SURFACE_USED_SUCCESSFULLY. At this point it is safe to release the surface and any underlying resources. Releasing the surface before receiving this signal may cause undesired behavior on lower API levels.

If the request is cancelled by the camera before successfully attaching the provided surface to the camera, then the resultListener will be invoked with a Result containing RESULT_REQUEST_CANCELLED. In addition, any cancellation listeners provided to addRequestCancellationListener will be invoked.

If the request was previously completed via willNotProvideSurface, then resultListener will be invoked with a Result containing RESULT_WILL_NOT_PROVIDE_SURFACE.

Upon returning from this method, the surface request is guaranteed to be complete. However, only the resultListener provided to the first invocation of this method should be used to track when the provided Surface is no longer in use by the camera, as subsequent invocations will always invoke the resultListener with a Result containing RESULT_SURFACE_ALREADY_PROVIDED.

Parameters
@NonNull Surface surface

The surface which will complete the request.

@NonNull Executor executor

Executor used to execute the resultListener.

@NonNull Consumer<SurfaceRequest.Result> resultListener

Listener used to track how the surface is used by the camera in response to being provided by this method.

setTransformationInfoListener

Added in 1.0.0
public void setTransformationInfoListener(
    @NonNull Executor executor,
    @NonNull SurfaceRequest.TransformationInfoListener listener
)

Sets a listener to receive updates on transformation info.

Sets a listener to receive the transformation info associated with this SurfaceRequest when it changes or becomes available. The listener is called immediately if transformation info is available at the time of setting.

Parameters
@NonNull Executor executor

The executor used to notify the listener.

@NonNull SurfaceRequest.TransformationInfoListener listener

the listener which will be called when transformation info changes.

willNotProvideSurface

Added in 1.0.0
public boolean willNotProvideSurface()

Signals that the request will never be fulfilled.

This may be called in the case where the application may be shutting down and a surface will never be produced to fulfill the request.

This should always be called as soon as it is known that the request will not be fulfilled. Failure to complete the SurfaceRequest via willNotProvideSurface() or provideSurface may cause long delays in shutting down the camera.

Upon returning from this method, the request is guaranteed to be complete, regardless of the return value. If the request was previously successfully completed by provideSurface, invoking this method will return false, and will have no effect on how the surface is used by the camera.

Returns
boolean

true if this call to willNotProvideSurface() successfully completes the request, i.e., the request has not already been completed via provideSurface or by a previous call to willNotProvideSurface() and has not already been cancelled by the camera.