CameraController.java revision 8be316c7a8caf962cf3fcf5e49d332fb2718319f
120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong/* 220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * Copyright (C) 2013 The Android Open Source Project 320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * 420fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * Licensed under the Apache License, Version 2.0 (the "License"); 520fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * you may not use this file except in compliance with the License. 620fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * You may obtain a copy of the License at 720fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * 820fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * http://www.apache.org/licenses/LICENSE-2.0 920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * 1020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * Unless required by applicable law or agreed to in writing, software 1120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * distributed under the License is distributed on an "AS IS" BASIS, 1220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 1320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * See the License for the specific language governing permissions and 1420fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * limitations under the License. 1520fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong */ 1620fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 1720fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kongpackage com.android.camera.app; 1820fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 1920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kongimport android.content.Context; 2020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kongimport android.os.Handler; 2120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 2220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kongimport com.android.camera.CameraDisabledException; 235596b4c902dcb685928b43678f428746ca5ffd08Angus Kongimport com.android.camera.debug.Log; 2420fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kongimport com.android.camera.util.CameraUtil; 2545f316bcecb8713cbb217507981fd657fcdaa274Alan Newbergerimport com.android.camera.util.GservicesHelper; 265a344964cb003525727c31d295ca3a592c245606Sol Boucherimport com.android.ex.camera2.portability.CameraAgent; 2729a009c974f153f72cd5e28851a8cb85a691e215Alan Newbergerimport com.android.ex.camera2.portability.CameraDeviceInfo; 282a0e18e280dfe541d4740345511ad299099cda79Senpo Huimport com.android.ex.camera2.portability.CameraExceptionHandler; 2920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 3020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong/** 3120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * A class which implements {@link com.android.camera.app.CameraProvider} used 3220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * by {@link com.android.camera.CameraActivity}. 3320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * TODO: Make this class package private. 3420fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong */ 355a344964cb003525727c31d295ca3a592c245606Sol Boucherpublic class CameraController implements CameraAgent.CameraOpenCallback, CameraProvider { 365596b4c902dcb685928b43678f428746ca5ffd08Angus Kong private static final Log.Tag TAG = new Log.Tag("CameraController"); 37432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong private static final int EMPTY_REQUEST = -1; 38846d3abfe3da2fa2a5593c7d40a196005408bed1Sascha Haeberling private final Context mContext; 395a344964cb003525727c31d295ca3a592c245606Sol Boucher private CameraAgent.CameraOpenCallback mCallbackReceiver; 40846d3abfe3da2fa2a5593c7d40a196005408bed1Sascha Haeberling private final Handler mCallbackHandler; 415a344964cb003525727c31d295ca3a592c245606Sol Boucher private final CameraAgent mCameraAgent; 4244ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher private final CameraAgent mCameraAgentNg; 4344ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher 4444ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher /** The one for the API that is currently in use (deprecated one by default). */ 4544ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher private CameraDeviceInfo mInfo; 4620fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 475a344964cb003525727c31d295ca3a592c245606Sol Boucher private CameraAgent.CameraProxy mCameraProxy; 48432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong private int mRequestingCameraId = EMPTY_REQUEST; 4920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 5020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong /** 5144ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher * Determines which of mCameraAgent and mCameraAgentNg is currently in use. 5244ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher * <p>It's only possible to enable this if the new API is actually 5344ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher * supported.</p> 5444ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher */ 5544ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher private boolean mUsingNewApi = false; 5644ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher 5744ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher /** 5820fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * Constructor. 5920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * 6020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * @param context The {@link android.content.Context} used to check if the 6120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * camera is disabled. 6220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * @param handler The {@link android.os.Handler} to post the camera 6320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * callbacks to. 6420fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * @param cameraManager Used for camera open/close. 6544ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher * @param cameraManagerNg Used for camera open/close with the new API. If 6644ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher * {@code null} or the same object as 6744ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher * {@code cameraManager}, the new API will not be 6844ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher * exposed and requests for it will get the old one. 6920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong */ 705a344964cb003525727c31d295ca3a592c245606Sol Boucher public CameraController(Context context, CameraAgent.CameraOpenCallback callbackReceiver, 7144ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher Handler handler, CameraAgent cameraManager, CameraAgent cameraManagerNg) { 7220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong mContext = context; 7320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong mCallbackReceiver = callbackReceiver; 7420fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong mCallbackHandler = handler; 755a344964cb003525727c31d295ca3a592c245606Sol Boucher mCameraAgent = cameraManager; 7644ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher // If the new implementation is the same as the old, the 7744ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher // CameraAgentFactory decided this device doesn't support the new API. 7844ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher mCameraAgentNg = cameraManagerNg != cameraManager ? cameraManagerNg : null; 795a344964cb003525727c31d295ca3a592c245606Sol Boucher mInfo = mCameraAgent.getCameraDeviceInfo(); 80c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong if (mInfo == null && mCallbackReceiver != null) { 81c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong mCallbackReceiver.onDeviceOpenFailure(-1, "GETTING_CAMERA_INFO"); 82c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong } 8320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 8420fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 8520fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 862a0e18e280dfe541d4740345511ad299099cda79Senpo Hu public void setCameraExceptionHandler(CameraExceptionHandler exceptionHandler) { 872a0e18e280dfe541d4740345511ad299099cda79Senpo Hu mCameraAgent.setCameraExceptionHandler(exceptionHandler); 8844ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher if (mCameraAgentNg != null) { 892a0e18e280dfe541d4740345511ad299099cda79Senpo Hu mCameraAgentNg.setCameraExceptionHandler(exceptionHandler); 9044ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher } 91630d55ba8111357730119899565026be503264c0Erin Dahlgren } 92630d55ba8111357730119899565026be503264c0Erin Dahlgren 93630d55ba8111357730119899565026be503264c0Erin Dahlgren @Override 9443e1813a5d4a59ef107768eb523d76ae2781cb02Sol Boucher public CameraDeviceInfo.Characteristics getCharacteristics(int cameraId) { 95c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong if (mInfo == null) { 96c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong return null; 97c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong } 9843e1813a5d4a59ef107768eb523d76ae2781cb02Sol Boucher return mInfo.getCharacteristics(cameraId); 9920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 10020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 10120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 1026190c36db653e848f29d1caefc4fa1bb61e8a662Erin Dahlgren public int getCurrentCameraId() { 1036190c36db653e848f29d1caefc4fa1bb61e8a662Erin Dahlgren if (mCameraProxy != null) { 1046190c36db653e848f29d1caefc4fa1bb61e8a662Erin Dahlgren return mCameraProxy.getCameraId(); 1056190c36db653e848f29d1caefc4fa1bb61e8a662Erin Dahlgren } else { 106395bc17b71d13d97b3fd9f258fcf4d5748e66e5eSol Boucher Log.v(TAG, "getCurrentCameraId without an open camera... returning requested id"); 107395bc17b71d13d97b3fd9f258fcf4d5748e66e5eSol Boucher return mRequestingCameraId; 1086190c36db653e848f29d1caefc4fa1bb61e8a662Erin Dahlgren } 1096190c36db653e848f29d1caefc4fa1bb61e8a662Erin Dahlgren } 1106190c36db653e848f29d1caefc4fa1bb61e8a662Erin Dahlgren 1116190c36db653e848f29d1caefc4fa1bb61e8a662Erin Dahlgren @Override 11220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong public int getNumberOfCameras() { 113c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong if (mInfo == null) { 114c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong return 0; 115c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong } 116c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong return mInfo.getNumberOfCameras(); 11720fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 11820fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 11920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 12020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong public int getFirstBackCameraId() { 121c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong if (mInfo == null) { 122c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong return -1; 123c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong } 124c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong return mInfo.getFirstBackCameraId(); 12520fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 12620fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 12720fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 12820fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong public int getFirstFrontCameraId() { 129c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong if (mInfo == null) { 130c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong return -1; 131c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong } 132c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong return mInfo.getFirstFrontCameraId(); 13320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 13420fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 13520fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 13639b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong public boolean isFrontFacingCamera(int id) { 137c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong if (mInfo == null) { 138c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong return false; 139c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong } 14043e1813a5d4a59ef107768eb523d76ae2781cb02Sol Boucher if (id >= mInfo.getNumberOfCameras() || mInfo.getCharacteristics(id) == null) { 14139b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong Log.e(TAG, "Camera info not available:" + id); 14239b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong return false; 14339b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong } 14443e1813a5d4a59ef107768eb523d76ae2781cb02Sol Boucher return mInfo.getCharacteristics(id).isFacingFront(); 14539b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong } 14639b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong 14739b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong @Override 14839b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong public boolean isBackFacingCamera(int id) { 149c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong if (mInfo == null) { 150c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong return false; 151c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong } 15243e1813a5d4a59ef107768eb523d76ae2781cb02Sol Boucher if (id >= mInfo.getNumberOfCameras() || mInfo.getCharacteristics(id) == null) { 15339b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong Log.e(TAG, "Camera info not available:" + id); 15439b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong return false; 15539b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong } 15643e1813a5d4a59ef107768eb523d76ae2781cb02Sol Boucher return mInfo.getCharacteristics(id).isFacingBack(); 15739b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong } 15839b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong 15939b460b805adaa54f310d8637868b3a5c792b2f5Angus Kong @Override 1605a344964cb003525727c31d295ca3a592c245606Sol Boucher public void onCameraOpened(CameraAgent.CameraProxy camera) { 1610b9eb5bb8d2ef409e8c88196c0c82c8ece65728bAngus Kong Log.v(TAG, "onCameraOpened"); 1620b9eb5bb8d2ef409e8c88196c0c82c8ece65728bAngus Kong if (mRequestingCameraId != camera.getCameraId()) { 163f578829a0f9a298fd5e4fdd9081b768f25811599Angus Kong return; 164f578829a0f9a298fd5e4fdd9081b768f25811599Angus Kong } 1650b9eb5bb8d2ef409e8c88196c0c82c8ece65728bAngus Kong mCameraProxy = camera; 166432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong mRequestingCameraId = EMPTY_REQUEST; 1670fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling if (mCallbackReceiver != null) { 1680fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling mCallbackReceiver.onCameraOpened(camera); 1690fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling } 17020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 17120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 17220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 17320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong public void onCameraDisabled(int cameraId) { 1740fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling if (mCallbackReceiver != null) { 1750fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling mCallbackReceiver.onCameraDisabled(cameraId); 1760fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling } 17720fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 17820fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 17920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 1801b9d4fb6f649262e491c0fa743229b4930a87802Angus Kong public void onDeviceOpenFailure(int cameraId, String info) { 1810fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling if (mCallbackReceiver != null) { 1821b9d4fb6f649262e491c0fa743229b4930a87802Angus Kong mCallbackReceiver.onDeviceOpenFailure(cameraId, info); 1830fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling } 18420fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 18520fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 18620fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 1870b9eb5bb8d2ef409e8c88196c0c82c8ece65728bAngus Kong public void onDeviceOpenedAlready(int cameraId, String info) { 1880fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling if (mCallbackReceiver != null) { 1890b9eb5bb8d2ef409e8c88196c0c82c8ece65728bAngus Kong mCallbackReceiver.onDeviceOpenedAlready(cameraId, info); 1900fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling } 19162753ae94be2ae7c08bf24ebcf1c462975af230fAngus Kong } 19262753ae94be2ae7c08bf24ebcf1c462975af230fAngus Kong 19362753ae94be2ae7c08bf24ebcf1c462975af230fAngus Kong @Override 1945a344964cb003525727c31d295ca3a592c245606Sol Boucher public void onReconnectionFailure(CameraAgent mgr, String info) { 1950fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling if (mCallbackReceiver != null) { 1960b9eb5bb8d2ef409e8c88196c0c82c8ece65728bAngus Kong mCallbackReceiver.onReconnectionFailure(mgr, info); 1970fed7e62bfa0d8351b91d3531228a113637963a3Sascha Haeberling } 19820fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 19920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 20020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 20120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong public void requestCamera(int id) { 20244ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher requestCamera(id, false); 20344ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher } 20444ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher 20544ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher @Override 20644ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher public void requestCamera(int id, boolean useNewApi) { 20729a009c974f153f72cd5e28851a8cb85a691e215Alan Newberger Log.v(TAG, "requestCamera"); 208432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong // Based on 209432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong // (mRequestingCameraId == id, mRequestingCameraId == EMPTY_REQUEST), 210432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong // we have (T, T), (T, F), (F, T), (F, F). 211432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong // (T, T): implies id == EMPTY_REQUEST. We don't allow this to happen 212432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong // here. Return. 213432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong // (F, F): A previous request hasn't been fulfilled yet. Return. 214432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong // (T, F): Already requested the same camera. No-op. Return. 215432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong // (F, T): Nothing is going on. Continue. 216432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong if (mRequestingCameraId != EMPTY_REQUEST || mRequestingCameraId == id) { 21720fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong return; 21820fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 219c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong if (mInfo == null) { 220c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong return; 221c44c5d93a681d98719dfffdf7223c694e1cab715Angus Kong } 22220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong mRequestingCameraId = id; 22344ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher 22444ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher // Only actually use the new API if it's supported on this device. 22544ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher useNewApi = mCameraAgentNg != null && useNewApi; 22644ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher CameraAgent cameraManager = useNewApi ? mCameraAgentNg : mCameraAgent; 22744ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher 22820fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong if (mCameraProxy == null) { 22920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong // No camera yet. 23044ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher checkAndOpenCamera(mContext, cameraManager, id, mCallbackHandler, this); 23144ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher } else if (mCameraProxy.getCameraId() != id || mUsingNewApi != useNewApi) { 23245f316bcecb8713cbb217507981fd657fcdaa274Alan Newberger boolean syncClose = GservicesHelper.useCamera2ApiThroughPortabilityLayer(mContext); 23329a009c974f153f72cd5e28851a8cb85a691e215Alan Newberger Log.v(TAG, "different camera already opened, closing then reopening"); 23444ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher // Already has camera opened, and is switching cameras and/or APIs. 23544ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher if (mUsingNewApi) { 23645f316bcecb8713cbb217507981fd657fcdaa274Alan Newberger mCameraAgentNg.closeCamera(mCameraProxy, true); 23744ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher } else { 23845f316bcecb8713cbb217507981fd657fcdaa274Alan Newberger // if using API2 ensure API1 usage is also synced 23945f316bcecb8713cbb217507981fd657fcdaa274Alan Newberger mCameraAgent.closeCamera(mCameraProxy, syncClose); 24044ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher } 24144ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher checkAndOpenCamera(mContext, cameraManager, id, mCallbackHandler, this); 24220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } else { 24320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong // The same camera, just do a reconnect. 2440b9eb5bb8d2ef409e8c88196c0c82c8ece65728bAngus Kong Log.v(TAG, "reconnecting to use the existing camera"); 24520fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong mCameraProxy.reconnect(mCallbackHandler, this); 24620fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong mCameraProxy = null; 24720fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 24844ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher 24944ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher mUsingNewApi = useNewApi; 25044ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher mInfo = cameraManager.getCameraDeviceInfo(); 25120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 25220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 25320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 25497e282a3764215ac193726e0d259e55bf94369feAngus Kong public boolean waitingForCamera() { 25597e282a3764215ac193726e0d259e55bf94369feAngus Kong return mRequestingCameraId != EMPTY_REQUEST; 25697e282a3764215ac193726e0d259e55bf94369feAngus Kong } 25797e282a3764215ac193726e0d259e55bf94369feAngus Kong 25897e282a3764215ac193726e0d259e55bf94369feAngus Kong @Override 25920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong public void releaseCamera(int id) { 260f578829a0f9a298fd5e4fdd9081b768f25811599Angus Kong if (mCameraProxy == null) { 261432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong if (mRequestingCameraId == EMPTY_REQUEST) { 262f578829a0f9a298fd5e4fdd9081b768f25811599Angus Kong // Camera not requested yet. 263f578829a0f9a298fd5e4fdd9081b768f25811599Angus Kong Log.w(TAG, "Trying to release the camera before requesting"); 264f578829a0f9a298fd5e4fdd9081b768f25811599Angus Kong } 265f578829a0f9a298fd5e4fdd9081b768f25811599Angus Kong // Camera requested but not available yet. 266432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong mRequestingCameraId = EMPTY_REQUEST; 267f578829a0f9a298fd5e4fdd9081b768f25811599Angus Kong return; 268f578829a0f9a298fd5e4fdd9081b768f25811599Angus Kong } 269662082d70ce8dc5f679a25322a00c2733f0339f2Alan Newberger int currentId = mCameraProxy.getCameraId(); 270662082d70ce8dc5f679a25322a00c2733f0339f2Alan Newberger if (currentId != id) { 271662082d70ce8dc5f679a25322a00c2733f0339f2Alan Newberger if (mRequestingCameraId == id) { 272662082d70ce8dc5f679a25322a00c2733f0339f2Alan Newberger Log.w(TAG, "Releasing camera which was requested but not yet " 273662082d70ce8dc5f679a25322a00c2733f0339f2Alan Newberger + "opened (current:requested): " + currentId + ":" + id); 274662082d70ce8dc5f679a25322a00c2733f0339f2Alan Newberger } else { 275662082d70ce8dc5f679a25322a00c2733f0339f2Alan Newberger throw new IllegalStateException("Trying to release a camera neither opened" 276662082d70ce8dc5f679a25322a00c2733f0339f2Alan Newberger + "nor requested (current:requested:for-release): " 277662082d70ce8dc5f679a25322a00c2733f0339f2Alan Newberger + currentId + ":" + mRequestingCameraId + ":" + id); 278662082d70ce8dc5f679a25322a00c2733f0339f2Alan Newberger } 27920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 280432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong mRequestingCameraId = EMPTY_REQUEST; 28120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 28220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 283846d3abfe3da2fa2a5593c7d40a196005408bed1Sascha Haeberling public void removeCallbackReceiver() { 284846d3abfe3da2fa2a5593c7d40a196005408bed1Sascha Haeberling mCallbackReceiver = null; 285846d3abfe3da2fa2a5593c7d40a196005408bed1Sascha Haeberling } 286846d3abfe3da2fa2a5593c7d40a196005408bed1Sascha Haeberling 28720fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong /** 28820fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * Closes the opened camera device. 28920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong * TODO: Make this method package private. 29020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong */ 2910b9eb5bb8d2ef409e8c88196c0c82c8ece65728bAngus Kong public void closeCamera(boolean synced) { 292dcea7639255d5bc4b17a5976bb84de1f6dba0ed1Sascha Haeberling Log.v(TAG, "Closing camera"); 29320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong mCameraProxy = null; 29444ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher if (mUsingNewApi) { 29544ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher mCameraAgentNg.closeCamera(mCameraProxy, synced); 29644ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher } else { 29744ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher mCameraAgent.closeCamera(mCameraProxy, synced); 29844ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher } 299432c964bb528a28600f19e18e81ff2b6407bb22dAngus Kong mRequestingCameraId = EMPTY_REQUEST; 30044ce4b26614f85518d548a6750c7a08908ae3596Sol Boucher mUsingNewApi = false; 30120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 30220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong 3035a344964cb003525727c31d295ca3a592c245606Sol Boucher private static void checkAndOpenCamera(Context context, CameraAgent cameraManager, 3045a344964cb003525727c31d295ca3a592c245606Sol Boucher final int cameraId, Handler handler, final CameraAgent.CameraOpenCallback cb) { 30529a009c974f153f72cd5e28851a8cb85a691e215Alan Newberger Log.v(TAG, "checkAndOpenCamera"); 30620fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong try { 3078be316c7a8caf962cf3fcf5e49d332fb2718319fPaul Rohde CameraUtil.throwIfCameraDisabled(); 3080b9eb5bb8d2ef409e8c88196c0c82c8ece65728bAngus Kong cameraManager.openCamera(handler, cameraId, cb); 30920fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } catch (CameraDisabledException ex) { 31020fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong handler.post(new Runnable() { 31120fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong @Override 31220fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong public void run() { 31320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong cb.onCameraDisabled(cameraId); 31420fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 31520fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong }); 31620fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 31720fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong } 3184d4a4bc8891bbfc5a7ca842f2981829ac3a716e8Doris Liu 3194d4a4bc8891bbfc5a7ca842f2981829ac3a716e8Doris Liu public void setOneShotPreviewCallback(Handler handler, 3205a344964cb003525727c31d295ca3a592c245606Sol Boucher CameraAgent.CameraPreviewDataCallback cb) { 3214d4a4bc8891bbfc5a7ca842f2981829ac3a716e8Doris Liu mCameraProxy.setOneShotPreviewCallback(handler, cb); 3224d4a4bc8891bbfc5a7ca842f2981829ac3a716e8Doris Liu } 32320fad249cbb587b8cf03e8e53bc64ff8e0bec7fdAngus Kong} 324