I am creating a custom camera app using surface view and Camera 1 API, there is some kind of problem in the lifecycle and i just cannot get it to work, but basically I am releasing the camera and then calling it again :
Camera is being used after Camera.release() was called
Ideally when back button is pressed should go back to previous activity, and when home button pressed should re open without any problem.
Would somebody guide me through a good implementation of this calls such as release and open the camera:
Custom Surface View Class:
public class ImageSurfaceView extends SurfaceView implements
SurfaceHolder.Callback {
private Camera camera;
private SurfaceHolder surfaceHolder;
public final String TAG = ImageSurfaceView.class.getSimpleName();
public ImageSurfaceView(Context context, Camera camera) {
super(context);
this.camera = camera;
this.surfaceHolder = getHolder();
this.surfaceHolder.addCallback(this);
}
@Override
public void surfaceCreated(SurfaceHolder holder) {
try {
this.camera.setPreviewDisplay(holder);
this.camera.startPreview();
} catch (IOException ex){
Log.e(TAG, "surfaceCreated: "+ex.getMessage() );
}
}
@Override
public void surfaceChanged(SurfaceHolder holder, int format, int width, int height) {
}
@Override
public void surfaceDestroyed(SurfaceHolder holder) {
this.camera.stopPreview();
this.camera.release();
this.camera = null;
}
}
and my CameraAcitivty LifeCycle CallBacks:
@Override
protected void onRestart() {
super.onRestart();
camera = null;
requestCamera();
}
@Override
protected void onResume() {
super.onResume();
try {
requestCamera();
} catch (RuntimeException ex){
Log.e(TAG, "onResume: "+ex.getMessage() );
}
}
@Override
protected void onPause() {
super.onPause();
if (camera != null) {
camera.setPreviewCallback(null);
imageSurfaceView.getHolder().removeCallback(imageSurfaceView);
camera.release();
camera = null;
}
}
@Override
protected void onStop() {
super.onStop();
isSurfaceCreated = false;
}
@Override
protected void onDestroy() {
super.onDestroy();
releaseCameraAndPreview();
}
private void releaseCameraAndPreview(){
if (camera != null) {
camera.stopPreview();
camera.release();
camera = null;
}
if(imageSurfaceView != null){
imageSurfaceView.destroyDrawingCache();
}
}
private void requestCamera(){
if (camera == null) {
if (checkPermission()) {
callCameraThread();
Toast.makeText(getApplicationContext(), "Permission already granted", Toast.LENGTH_LONG).show();
} else {
requestPermission();
}
}
}
public Camera checkDeviceCamera(){
Camera mCamera = null;
try {
mCamera = Camera.open(0);
} catch (Exception e) {
e.printStackTrace();
}
return mCamera;
}
private void callCameraThread(){
if(mThread == null){
mThread = new CameraHandlerThread();
}
synchronized (mThread){
mThread.openCamera();
}
}
public synchronized void loadSurface(){
while(camera == null){
try {
this.wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
if(!isSurfaceCreated) {
imageSurfaceView = new ImageSurfaceView(CameraActivity.this, camera);
cameraPreviewLayout.addView(imageSurfaceView);
isSurfaceCreated = true;
imgGhost = new ImageView(this);
imgGhost.setLayoutParams(new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.MATCH_PARENT));
imgGhost.setBackground(ContextCompat.getDrawable(this, R.drawable.fantasma_doc_front));
cameraPreviewLayout.addView(imgGhost);
}
}
private CameraHandlerThread mThread = null;
private class CameraHandlerThread extends HandlerThread {
Handler mHandler = null;
CameraHandlerThread(){
super("CameraHandlerThread");
start();
mHandler = new Handler(getLooper());
}
synchronized void notifyCameraOpened(){
notifyAll();
}
void openCamera(){
mHandler.post(new Runnable() {
@Override
public void run() {
camera = checkDeviceCamera();
notifyCameraOpened();
}
});
try{
wait();
} catch (InterruptedException ex){
Log.e(TAG, "openCamera: "+"wait was interrupted" );
}
}
}
I dont know how to handle properly the releasing of camera and creation of surface View, now after some changes I did, I am able to press home button and come back without problem, but onBackPressed crashes, before was the opposite, Thanks in advance for any help in this matter
so, pretty much I tried to minimize duplicated reference to the camera, here is how onBackPressed looks like:
This is how onResume() looks like now, I added support for most modern API's tested in lollipop, Marshmallow and Nougat
in onRestart just remove the requestCamera():
and lastly a null validation in onSurfaceDestroyed():
Now it doesn't have any problem when home button is pressed, or if back button is pressed, I make sure that camera is set to null everytime it passes trhough onRestart(), after closed, and in onStop i make sure the flag isSurfaceCreated is set to false.