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