1 /*
2  * Copyright (C) 2014 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.camera;
18 
19 import android.content.Context;
20 import android.graphics.Matrix;
21 import android.graphics.Point;
22 import android.graphics.RectF;
23 import android.graphics.SurfaceTexture;
24 import android.location.Location;
25 import android.media.MediaActionSound;
26 import android.net.Uri;
27 import android.os.AsyncTask;
28 import android.os.Handler;
29 import android.os.HandlerThread;
30 import android.os.SystemClock;
31 import android.view.GestureDetector;
32 import android.view.KeyEvent;
33 import android.view.MotionEvent;
34 import android.view.Surface;
35 import android.view.View;
36 
37 import com.android.camera.app.AppController;
38 import com.android.camera.app.CameraAppUI;
39 import com.android.camera.app.CameraAppUI.BottomBarUISpec;
40 import com.android.camera.app.LocationManager;
41 import com.android.camera.app.OrientationManager.DeviceOrientation;
42 import com.android.camera.async.MainThread;
43 import com.android.camera.burst.BurstFacade;
44 import com.android.camera.burst.BurstFacadeFactory;
45 import com.android.camera.burst.BurstReadyStateChangeListener;
46 import com.android.camera.burst.OrientationLockController;
47 import com.android.camera.captureintent.PreviewTransformCalculator;
48 import com.android.camera.debug.DebugPropertyHelper;
49 import com.android.camera.debug.Log;
50 import com.android.camera.debug.Log.Tag;
51 import com.android.camera.device.CameraId;
52 import com.android.camera.hardware.HardwareSpec;
53 import com.android.camera.hardware.HeadingSensor;
54 import com.android.camera.module.ModuleController;
55 import com.android.camera.one.OneCamera;
56 import com.android.camera.one.OneCamera.AutoFocusState;
57 import com.android.camera.one.OneCamera.CaptureReadyCallback;
58 import com.android.camera.one.OneCamera.Facing;
59 import com.android.camera.one.OneCamera.OpenCallback;
60 import com.android.camera.one.OneCamera.PhotoCaptureParameters;
61 import com.android.camera.one.OneCameraAccessException;
62 import com.android.camera.one.OneCameraCaptureSetting;
63 import com.android.camera.one.OneCameraCharacteristics;
64 import com.android.camera.one.OneCameraException;
65 import com.android.camera.one.OneCameraManager;
66 import com.android.camera.one.OneCameraModule;
67 import com.android.camera.one.OneCameraOpener;
68 import com.android.camera.one.config.OneCameraFeatureConfig;
69 import com.android.camera.one.v2.photo.ImageRotationCalculator;
70 import com.android.camera.one.v2.photo.ImageRotationCalculatorImpl;
71 import com.android.camera.remote.RemoteCameraModule;
72 import com.android.camera.session.CaptureSession;
73 import com.android.camera.settings.Keys;
74 import com.android.camera.settings.SettingsManager;
75 import com.android.camera.stats.CaptureStats;
76 import com.android.camera.stats.UsageStatistics;
77 import com.android.camera.stats.profiler.Profile;
78 import com.android.camera.stats.profiler.Profiler;
79 import com.android.camera.stats.profiler.Profilers;
80 import com.android.camera.ui.CountDownView;
81 import com.android.camera.ui.PreviewStatusListener;
82 import com.android.camera.ui.TouchCoordinate;
83 import com.android.camera.ui.focus.FocusController;
84 import com.android.camera.ui.focus.FocusSound;
85 import com.android.camera.util.AndroidServices;
86 import com.android.camera.util.ApiHelper;
87 import com.android.camera.util.CameraUtil;
88 import com.android.camera.util.GcamHelper;
89 import com.android.camera.util.Size;
90 import com.android.camera2.R;
91 import com.android.ex.camera2.portability.CameraAgent.CameraProxy;
92 import com.google.common.logging.eventprotos;
93 
94 import java.util.concurrent.Semaphore;
95 import java.util.concurrent.TimeUnit;
96 
97 import javax.annotation.Nonnull;
98 
99 /**
100  * New Capture module that is made to support photo and video capture on top of
101  * the OneCamera API, to transparently support GCam.
102  * <p>
103  * This has been a re-write with pieces taken and improved from GCamModule and
104  * PhotoModule, which are to be retired eventually.
105  * <p>
106  */
107 public class CaptureModule extends CameraModule implements
108         ModuleController,
109         CountDownView.OnCountDownStatusListener,
110         OneCamera.PictureCallback,
111         OneCamera.FocusStateListener,
112         OneCamera.ReadyStateChangedListener,
113         RemoteCameraModule {
114 
115     private static final Tag TAG = new Tag("CaptureModule");
116     /** Enable additional debug output. */
117     private static final boolean DEBUG = true;
118     /** Workaround Flag for b/19271661 to use autotransformation in Capture Layout in Nexus4 **/
119     private static final boolean USE_AUTOTRANSFORM_UI_LAYOUT = ApiHelper.IS_NEXUS_4;
120 
121     /** Timeout for camera open/close operations. */
122     private static final int CAMERA_OPEN_CLOSE_TIMEOUT_MILLIS = 2500;
123 
124     /** System Properties switch to enable debugging focus UI. */
125     private static final boolean CAPTURE_DEBUG_UI = DebugPropertyHelper.showCaptureDebugUI();
126 
127     private final Object mDimensionLock = new Object();
128 
129     /**
130      * Sticky Gcam mode is when this module's sole purpose it to be the Gcam
131      * mode. If true, the device uses {@link PhotoModule} for normal picture
132      * taking.
133      */
134     private final boolean mStickyGcamCamera;
135 
136     /** Controller giving us access to other services. */
137     private final AppController mAppController;
138     /** The applications settings manager. */
139     private final SettingsManager mSettingsManager;
140     /** Application context. */
141     private final Context mContext;
142     /** Module UI. */
143     private CaptureModuleUI mUI;
144     /** The camera manager used to open cameras. */
145     private OneCameraOpener mOneCameraOpener;
146     /** The manager to query for camera device information */
147     private OneCameraManager mOneCameraManager;
148     /** The currently opened camera device, or null if the camera is closed. */
149     private OneCamera mCamera;
150     /** The selected picture size. */
151     private Size mPictureSize;
152     /** Fair semaphore held when opening or closing the camera. */
153     private final Semaphore mCameraOpenCloseLock = new Semaphore(1, true);
154     /** The direction the currently opened camera is facing to. */
155     private Facing mCameraFacing;
156     /** Whether HDR Scene mode is currently enabled. */
157     private boolean mHdrSceneEnabled = false;
158     private boolean mHdrPlusEnabled = false;
159     private final Object mSurfaceTextureLock = new Object();
160     /**
161      * Flag that is used when Fatal Error Handler is running and the app should
162      * not continue execution
163      */
164     private boolean mShowErrorAndFinish;
165     private TouchCoordinate mLastShutterTouchCoordinate = null;
166 
167     private FocusController mFocusController;
168     private OneCameraCharacteristics mCameraCharacteristics;
169     final private PreviewTransformCalculator mPreviewTransformCalculator;
170 
171     /** The listener to listen events from the CaptureModuleUI. */
172     private final CaptureModuleUI.CaptureModuleUIListener mUIListener =
173             new CaptureModuleUI.CaptureModuleUIListener() {
174                 @Override
175                 public void onZoomRatioChanged(float zoomRatio) {
176                     mZoomValue = zoomRatio;
177                     if (mCamera != null) {
178                         mCamera.setZoom(zoomRatio);
179                     }
180                 }
181             };
182 
183     /** The listener to respond preview area changes. */
184     private final PreviewStatusListener.PreviewAreaChangedListener mPreviewAreaChangedListener =
185             new PreviewStatusListener.PreviewAreaChangedListener() {
186                 @Override
187                 public void onPreviewAreaChanged(RectF previewArea) {
188                     mPreviewArea = previewArea;
189                     mFocusController.configurePreviewDimensions(previewArea);
190                 }
191             };
192 
193     /** The listener to listen events from the preview. */
194     private final PreviewStatusListener mPreviewStatusListener = new PreviewStatusListener() {
195         @Override
196         public void onPreviewLayoutChanged(View v, int left, int top, int right,
197                 int bottom, int oldLeft, int oldTop, int oldRight, int oldBottom) {
198             int width = right - left;
199             int height = bottom - top;
200             updatePreviewTransform(width, height, false);
201         }
202 
203         @Override
204         public boolean shouldAutoAdjustTransformMatrixOnLayout() {
205             return USE_AUTOTRANSFORM_UI_LAYOUT;
206         }
207 
208         @Override
209         public void onPreviewFlipped() {
210             // Do nothing because when preview is flipped, TextureView will lay
211             // itself out again, which will then trigger a transform matrix
212             // update.
213         }
214 
215         @Override
216         public GestureDetector.OnGestureListener getGestureListener() {
217             return new GestureDetector.SimpleOnGestureListener() {
218                 @Override
219                 public boolean onSingleTapUp(MotionEvent ev) {
220                     Point tapPoint = new Point((int) ev.getX(), (int) ev.getY());
221                     Log.v(TAG, "onSingleTapUpPreview location=" + tapPoint);
222                     if (!mCameraCharacteristics.isAutoExposureSupported() &&
223                           !mCameraCharacteristics.isAutoFocusSupported()) {
224                         return false;
225                     }
226                     startActiveFocusAt(tapPoint.x, tapPoint.y);
227                     return true;
228                 }
229             };
230         }
231 
232         @Override
233         public View.OnTouchListener getTouchListener() {
234             return null;
235         }
236 
237         @Override
238         public void onSurfaceTextureAvailable(SurfaceTexture surface, int width, int height) {
239             Log.d(TAG, "onSurfaceTextureAvailable");
240             // Force to re-apply transform matrix here as a workaround for
241             // b/11168275
242             updatePreviewTransform(width, height, true);
243             synchronized (mSurfaceTextureLock) {
244                 mPreviewSurfaceTexture = surface;
245             }
246             reopenCamera();
247         }
248 
249         @Override
250         public boolean onSurfaceTextureDestroyed(SurfaceTexture surface) {
251             Log.d(TAG, "onSurfaceTextureDestroyed");
252             synchronized (mSurfaceTextureLock) {
253                 mPreviewSurfaceTexture = null;
254             }
255             closeCamera();
256             return true;
257         }
258 
259         @Override
260         public void onSurfaceTextureSizeChanged(SurfaceTexture surface, int width, int height) {
261             Log.d(TAG, "onSurfaceTextureSizeChanged");
262             updatePreviewBufferSize();
263         }
264 
265         @Override
266         public void onSurfaceTextureUpdated(SurfaceTexture surface) {
267             if (mState == ModuleState.UPDATE_TRANSFORM_ON_NEXT_SURFACE_TEXTURE_UPDATE) {
268                 Log.d(TAG, "onSurfaceTextureUpdated --> updatePreviewTransform");
269                 mState = ModuleState.IDLE;
270                 CameraAppUI appUI = mAppController.getCameraAppUI();
271                 updatePreviewTransform(appUI.getSurfaceWidth(), appUI.getSurfaceHeight(), true);
272             }
273         }
274     };
275 
276     private final OneCamera.PictureSaverCallback mPictureSaverCallback =
277             new OneCamera.PictureSaverCallback() {
278                 @Override
279                 public void onRemoteThumbnailAvailable(final byte[] jpegImage) {
280                     mMainThread.execute(new Runnable() {
281                         @Override
282                         public void run() {
283                             mAppController.getServices().getRemoteShutterListener()
284                                     .onPictureTaken(jpegImage);
285                         }
286                     });
287                 }
288             };
289 
290     /** State by the module state machine. */
291     private static enum ModuleState {
292         IDLE,
293         WATCH_FOR_NEXT_FRAME_AFTER_PREVIEW_STARTED,
294         UPDATE_TRANSFORM_ON_NEXT_SURFACE_TEXTURE_UPDATE,
295     }
296 
297     /** The current state of the module. */
298     private ModuleState mState = ModuleState.IDLE;
299     /** Current zoom value. */
300     private float mZoomValue = 1f;
301 
302     /** Records beginning frame of each AF scan. */
303     private long mAutoFocusScanStartFrame = -1;
304     /** Records beginning time of each AF scan in uptimeMillis. */
305     private long mAutoFocusScanStartTime;
306 
307     /** Heading sensor. */
308     private HeadingSensor mHeadingSensor;
309 
310     /** Used to fetch and embed the location into captured images. */
311     private final LocationManager mLocationManager;
312     /** Plays sounds for countdown timer. */
313     private SoundPlayer mSoundPlayer;
314     private final MediaActionSound mMediaActionSound;
315 
316     /** Whether the module is paused right now. */
317     private boolean mPaused;
318 
319     /** Main thread. */
320     private final MainThread mMainThread;
321     /** Handler thread for camera-related operations. */
322     private Handler mCameraHandler;
323 
324     /** Current display rotation in degrees. */
325     private int mDisplayRotation;
326     /** Current screen width in pixels. */
327     private int mScreenWidth;
328     /** Current screen height in pixels. */
329     private int mScreenHeight;
330     /** Current width of preview frames from camera. */
331     private int mPreviewBufferWidth;
332     /** Current height of preview frames from camera.. */
333     private int mPreviewBufferHeight;
334     /** Area used by preview. */
335     RectF mPreviewArea;
336 
337     /** The surface texture for the preview. */
338     private SurfaceTexture mPreviewSurfaceTexture;
339 
340     /** The burst manager for controlling the burst. */
341     private final BurstFacade mBurstController;
342     private static final String BURST_SESSIONS_DIR = "burst_sessions";
343 
344     private final Profiler mProfiler = Profilers.instance().guard();
345 
CaptureModule(AppController appController)346     public CaptureModule(AppController appController) {
347         this(appController, false);
348     }
349 
350     /** Constructs a new capture module. */
CaptureModule(AppController appController, boolean stickyHdr)351     public CaptureModule(AppController appController, boolean stickyHdr) {
352         super(appController);
353         Profile guard = mProfiler.create("new CaptureModule").start();
354         mPaused = true;
355         mMainThread = MainThread.create();
356         mAppController = appController;
357         mContext = mAppController.getAndroidContext();
358         mSettingsManager = mAppController.getSettingsManager();
359         mStickyGcamCamera = stickyHdr;
360         mLocationManager = mAppController.getLocationManager();
361         mPreviewTransformCalculator = new PreviewTransformCalculator(
362                 mAppController.getOrientationManager());
363 
364         mBurstController = BurstFacadeFactory.create(mContext,
365                 new OrientationLockController() {
366                     @Override
367                     public void unlockOrientation() {
368                         mAppController.getOrientationManager().unlockOrientation();
369                     }
370 
371                         @Override
372                     public void lockOrientation() {
373                         mAppController.getOrientationManager().lockOrientation();
374                     }
375                 },
376                 new BurstReadyStateChangeListener() {
377                    @Override
378                     public void onBurstReadyStateChanged(boolean ready) {
379                         // TODO: This needs to take into account the state of
380                         // the whole system, not just burst.
381                        onReadyStateChanged(false);
382                     }
383                 });
384         mMediaActionSound = new MediaActionSound();
385         guard.stop();
386     }
387 
updateCameraCharacteristics()388     private boolean updateCameraCharacteristics() {
389         try {
390             CameraId cameraId = mOneCameraManager.findFirstCameraFacing(mCameraFacing);
391             if (cameraId != null && cameraId.getValue() != null) {
392                 mCameraCharacteristics = mOneCameraManager.getOneCameraCharacteristics(cameraId);
393                 return mCameraCharacteristics != null;
394             }
395         } catch (OneCameraAccessException ignored) { }
396             mAppController.getFatalErrorHandler().onGenericCameraAccessFailure();
397             return false;
398     }
399 
400     @Override
init(CameraActivity activity, boolean isSecureCamera, boolean isCaptureIntent)401     public void init(CameraActivity activity, boolean isSecureCamera, boolean isCaptureIntent) {
402         Profile guard = mProfiler.create("CaptureModule.init").start();
403         Log.d(TAG, "init UseAutotransformUiLayout = " + USE_AUTOTRANSFORM_UI_LAYOUT);
404         HandlerThread thread = new HandlerThread("CaptureModule.mCameraHandler");
405         thread.start();
406         mCameraHandler = new Handler(thread.getLooper());
407         mOneCameraOpener = mAppController.getCameraOpener();
408 
409         try {
410             mOneCameraManager = OneCameraModule.provideOneCameraManager();
411         } catch (OneCameraException e) {
412             Log.e(TAG, "Unable to provide a OneCameraManager. ", e);
413         }
414         mDisplayRotation = CameraUtil.getDisplayRotation();
415         mCameraFacing = getFacingFromCameraId(
416               mSettingsManager.getInteger(mAppController.getModuleScope(), Keys.KEY_CAMERA_ID));
417         mShowErrorAndFinish = !updateCameraCharacteristics();
418         if (mShowErrorAndFinish) {
419             return;
420         }
421         mUI = new CaptureModuleUI(activity, mAppController.getModuleLayoutRoot(), mUIListener);
422         mAppController.setPreviewStatusListener(mPreviewStatusListener);
423         synchronized (mSurfaceTextureLock) {
424             mPreviewSurfaceTexture = mAppController.getCameraAppUI().getSurfaceTexture();
425         }
426         mSoundPlayer = new SoundPlayer(mContext);
427 
428         FocusSound focusSound = new FocusSound(mSoundPlayer, R.raw.material_camera_focus);
429         mFocusController = new FocusController(mUI.getFocusRing(), focusSound, mMainThread);
430 
431         mHeadingSensor = new HeadingSensor(AndroidServices.instance().provideSensorManager());
432 
433         View cancelButton = activity.findViewById(R.id.shutter_cancel_button);
434         cancelButton.setOnClickListener(new View.OnClickListener() {
435             @Override
436             public void onClick(View view) {
437                 cancelCountDown();
438             }
439         });
440 
441         mMediaActionSound.load(MediaActionSound.SHUTTER_CLICK);
442         guard.stop();
443     }
444 
445     @Override
onShutterButtonLongPressed()446     public void onShutterButtonLongPressed() {
447         try {
448             OneCameraCharacteristics cameraCharacteristics;
449             CameraId cameraId = mOneCameraManager.findFirstCameraFacing(mCameraFacing);
450             cameraCharacteristics = mOneCameraManager.getOneCameraCharacteristics(cameraId);
451             DeviceOrientation deviceOrientation = mAppController.getOrientationManager()
452                     .getDeviceOrientation();
453             ImageRotationCalculator imageRotationCalculator = ImageRotationCalculatorImpl
454                     .from(mAppController.getOrientationManager(), cameraCharacteristics);
455 
456             mBurstController.startBurst(
457                     new CaptureSession.CaptureSessionCreator() {
458                         @Override
459                         public CaptureSession createAndStartEmpty() {
460                             return createAndStartUntrackedCaptureSession();
461                         }
462                     },
463                     deviceOrientation,
464                     mCamera.getDirection(),
465                     imageRotationCalculator.toImageRotation().getDegrees());
466 
467         } catch (OneCameraAccessException e) {
468             Log.e(TAG, "Cannot start burst", e);
469             return;
470         }
471     }
472 
473     @Override
onShutterButtonFocus(boolean pressed)474     public void onShutterButtonFocus(boolean pressed) {
475         if (!pressed) {
476             // the shutter button was released, stop any bursts.
477             mBurstController.stopBurst();
478         }
479     }
480 
481     @Override
onShutterCoordinate(TouchCoordinate coord)482     public void onShutterCoordinate(TouchCoordinate coord) {
483         mLastShutterTouchCoordinate = coord;
484     }
485 
486     @Override
onShutterButtonClick()487     public void onShutterButtonClick() {
488         if (mCamera == null) {
489             return;
490         }
491 
492         int countDownDuration = mSettingsManager
493                 .getInteger(SettingsManager.SCOPE_GLOBAL, Keys.KEY_COUNTDOWN_DURATION);
494         if (countDownDuration > 0) {
495             // Start count down.
496             mAppController.getCameraAppUI().transitionToCancel();
497             mAppController.getCameraAppUI().hideModeOptions();
498             mUI.setCountdownFinishedListener(this);
499             mUI.startCountdown(countDownDuration);
500             // Will take picture later via listener callback.
501         } else {
502             takePictureNow();
503         }
504     }
505 
506 
decorateSessionAtCaptureTime(CaptureSession session)507     private void decorateSessionAtCaptureTime(CaptureSession session) {
508         String flashSetting =
509                 mSettingsManager.getString(mAppController.getCameraScope(),
510                         Keys.KEY_FLASH_MODE);
511         boolean gridLinesOn = Keys.areGridLinesOn(mSettingsManager);
512         float timerDuration = mSettingsManager
513                 .getInteger(SettingsManager.SCOPE_GLOBAL, Keys.KEY_COUNTDOWN_DURATION);
514 
515         session.getCollector().decorateAtTimeCaptureRequest(
516                 eventprotos.NavigationChange.Mode.PHOTO_CAPTURE,
517                 session.getTitle() + ".jpg",
518                 (mCameraFacing == Facing.FRONT),
519                 mHdrSceneEnabled,
520                 mZoomValue,
521                 flashSetting,
522                 gridLinesOn,
523                 timerDuration,
524                 mLastShutterTouchCoordinate,
525                 null /* TODO: Implement Volume Button Shutter Click Instrumentation */,
526                 mCameraCharacteristics.getSensorInfoActiveArraySize()
527         );
528     }
529 
takePictureNow()530     private void takePictureNow() {
531         if (mCamera == null) {
532             Log.i(TAG, "Not taking picture since Camera is closed.");
533             return;
534         }
535 
536         CaptureSession session = createAndStartCaptureSession();
537         int orientation = mAppController.getOrientationManager().getDeviceOrientation()
538                 .getDegrees();
539 
540         // TODO: This should really not use getExternalCacheDir and instead use
541         // the SessionStorage API. Need to sync with gcam if that's OK.
542         PhotoCaptureParameters params = new PhotoCaptureParameters(
543                 session.getTitle(), orientation, session.getLocation(),
544                 mContext.getExternalCacheDir(), this, mPictureSaverCallback,
545                 mHeadingSensor.getCurrentHeading(), mZoomValue, 0);
546         decorateSessionAtCaptureTime(session);
547         mCamera.takePicture(params, session);
548     }
549 
550     /**
551      * Creates, starts and returns a new capture session. The returned session
552      * will have been started with an empty placeholder image.
553      */
createAndStartCaptureSession()554     private CaptureSession createAndStartCaptureSession() {
555         long sessionTime = getSessionTime();
556         Location location = mLocationManager.getCurrentLocation();
557         String title = CameraUtil.instance().createJpegName(sessionTime);
558         CaptureSession session = getServices().getCaptureSessionManager()
559                 .createNewSession(title, sessionTime, location);
560 
561         session.startEmpty(new CaptureStats(mHdrPlusEnabled),
562               new Size((int) mPreviewArea.width(), (int) mPreviewArea.height()));
563         return session;
564     }
565 
createAndStartUntrackedCaptureSession()566     private CaptureSession createAndStartUntrackedCaptureSession() {
567         long sessionTime = getSessionTime();
568         Location location = mLocationManager.getCurrentLocation();
569         String title = CameraUtil.instance().createJpegName(sessionTime);
570         CaptureSession session = getServices().getCaptureSessionManager()
571               .createNewSession(title, sessionTime, location);
572 
573         session.startEmpty(null,
574               new Size((int) mPreviewArea.width(), (int) mPreviewArea.height()));
575         return session;
576     }
577 
getSessionTime()578     private long getSessionTime() {
579         // TODO: Replace with a mockable TimeProvider interface.
580         return System.currentTimeMillis();
581     }
582 
583     @Override
onCountDownFinished()584     public void onCountDownFinished() {
585         mAppController.getCameraAppUI().transitionToCapture();
586         mAppController.getCameraAppUI().showModeOptions();
587         if (mPaused) {
588             return;
589         }
590         takePictureNow();
591     }
592 
593     @Override
onRemainingSecondsChanged(int remainingSeconds)594     public void onRemainingSecondsChanged(int remainingSeconds) {
595         if (remainingSeconds == 1) {
596             mSoundPlayer.play(R.raw.timer_final_second, 0.6f);
597         } else if (remainingSeconds == 2 || remainingSeconds == 3) {
598             mSoundPlayer.play(R.raw.timer_increment, 0.6f);
599         }
600     }
601 
cancelCountDown()602     private void cancelCountDown() {
603         if (mUI.isCountingDown()) {
604             // Cancel on-going countdown.
605             mUI.cancelCountDown();
606         }
607 
608         if (!mPaused) {
609             mAppController.getCameraAppUI().showModeOptions();
610             mAppController.getCameraAppUI().transitionToCapture();
611         }
612     }
613 
614     @Override
onQuickExpose()615     public void onQuickExpose() {
616         mMainThread.execute(new Runnable() {
617             @Override
618             public void run() {
619                 // Starts the short version of the capture animation UI.
620                 mAppController.startFlashAnimation(true);
621                 mMediaActionSound.play(MediaActionSound.SHUTTER_CLICK);
622             }
623         });
624     }
625 
626     @Override
onRemoteShutterPress()627     public void onRemoteShutterPress() {
628         Log.d(TAG, "onRemoteShutterPress");
629         // TODO: Check whether shutter is enabled.
630         takePictureNow();
631     }
632 
initSurfaceTextureConsumer()633     private void initSurfaceTextureConsumer() {
634         synchronized (mSurfaceTextureLock) {
635             if (mPreviewSurfaceTexture != null) {
636                 mPreviewSurfaceTexture.setDefaultBufferSize(
637                         mAppController.getCameraAppUI().getSurfaceWidth(),
638                         mAppController.getCameraAppUI().getSurfaceHeight());
639             }
640         }
641         reopenCamera();
642     }
643 
reopenCamera()644     private void reopenCamera() {
645         if (mPaused) {
646             return;
647         }
648         AsyncTask.THREAD_POOL_EXECUTOR.execute(new Runnable() {
649             @Override
650             public void run() {
651                 closeCamera();
652                 if(!mAppController.isPaused()) {
653                     openCameraAndStartPreview();
654                 }
655             }
656         });
657     }
658 
getPreviewSurfaceTexture()659     private SurfaceTexture getPreviewSurfaceTexture() {
660         synchronized (mSurfaceTextureLock) {
661             return mPreviewSurfaceTexture;
662         }
663     }
664 
updatePreviewBufferSize()665     private void updatePreviewBufferSize() {
666         synchronized (mSurfaceTextureLock) {
667             if (mPreviewSurfaceTexture != null) {
668                 mPreviewSurfaceTexture.setDefaultBufferSize(mPreviewBufferWidth,
669                         mPreviewBufferHeight);
670             }
671         }
672     }
673 
674     @Override
resume()675     public void resume() {
676         if (mShowErrorAndFinish) {
677             return;
678         }
679         Profile guard = mProfiler.create("CaptureModule.resume").start();
680 
681         // We'll transition into 'ready' once the preview is started.
682         onReadyStateChanged(false);
683         mPaused = false;
684         mAppController.addPreviewAreaSizeChangedListener(mPreviewAreaChangedListener);
685         mAppController.addPreviewAreaSizeChangedListener(mUI);
686 
687         guard.mark();
688         getServices().getRemoteShutterListener().onModuleReady(this);
689         guard.mark("getRemoteShutterListener.onModuleReady");
690         mBurstController.initialize(new SurfaceTexture(0));
691 
692         // TODO: Check if we can really take a photo right now (memory, camera
693         // state, ... ).
694         mAppController.getCameraAppUI().enableModeOptions();
695         mAppController.setShutterEnabled(true);
696         mAppController.getCameraAppUI().showAccessibilityZoomUI(
697                 mCameraCharacteristics.getAvailableMaxDigitalZoom());
698 
699         mHdrPlusEnabled = mStickyGcamCamera || mAppController.getSettingsManager().getInteger(
700                 SettingsManager.SCOPE_GLOBAL, Keys.KEY_CAMERA_HDR_PLUS) == 1;
701 
702         mHdrSceneEnabled = !mStickyGcamCamera && mAppController.getSettingsManager().getBoolean(
703               SettingsManager.SCOPE_GLOBAL, Keys.KEY_CAMERA_HDR);
704 
705         // This means we are resuming with an existing preview texture. This
706         // means we will never get the onSurfaceTextureAvailable call. So we
707         // have to open the camera and start the preview here.
708         SurfaceTexture texture = getPreviewSurfaceTexture();
709 
710         guard.mark();
711         if (texture != null) {
712             initSurfaceTextureConsumer();
713             guard.mark("initSurfaceTextureConsumer");
714         }
715 
716         mSoundPlayer.loadSound(R.raw.timer_final_second);
717         mSoundPlayer.loadSound(R.raw.timer_increment);
718 
719         guard.mark();
720         mHeadingSensor.activate();
721         guard.stop("mHeadingSensor.activate()");
722     }
723 
724     @Override
pause()725     public void pause() {
726         if (mShowErrorAndFinish) {
727             return;
728         }
729         cancelCountDown();
730         mPaused = true;
731         mHeadingSensor.deactivate();
732 
733         mAppController.removePreviewAreaSizeChangedListener(mUI);
734         mAppController.removePreviewAreaSizeChangedListener(mPreviewAreaChangedListener);
735         getServices().getRemoteShutterListener().onModuleExit();
736         mBurstController.release();
737         closeCamera();
738         resetTextureBufferSize();
739         mSoundPlayer.unloadSound(R.raw.timer_final_second);
740         mSoundPlayer.unloadSound(R.raw.timer_increment);
741     }
742 
743     @Override
destroy()744     public void destroy() {
745         mSoundPlayer.release();
746         mMediaActionSound.release();
747         mCameraHandler.getLooper().quitSafely();
748     }
749 
750     @Override
onLayoutOrientationChanged(boolean isLandscape)751     public void onLayoutOrientationChanged(boolean isLandscape) {
752         Log.d(TAG, "onLayoutOrientationChanged");
753     }
754 
755     @Override
onCameraAvailable(CameraProxy cameraProxy)756     public void onCameraAvailable(CameraProxy cameraProxy) {
757         // Ignore since we manage the camera ourselves until we remove this.
758     }
759 
760     @Override
hardResetSettings(SettingsManager settingsManager)761     public void hardResetSettings(SettingsManager settingsManager) {
762         if (mStickyGcamCamera) {
763             // Sticky HDR+ mode should hard reset HDR+ to on, and camera back
764             // facing.
765             settingsManager.set(SettingsManager.SCOPE_GLOBAL, Keys.KEY_CAMERA_HDR_PLUS, true);
766             settingsManager.set(mAppController.getModuleScope(), Keys.KEY_CAMERA_ID,
767                   mOneCameraManager.findFirstCameraFacing(Facing.BACK).getValue());
768         }
769     }
770 
771     @Override
getHardwareSpec()772     public HardwareSpec getHardwareSpec() {
773         return new HardwareSpec() {
774             @Override
775             public boolean isFrontCameraSupported() {
776                 return mOneCameraManager.hasCameraFacing(Facing.FRONT);
777             }
778 
779             @Override
780             public boolean isHdrSupported() {
781                 if (ApiHelper.IS_NEXUS_4 && is16by9AspectRatio(mPictureSize)) {
782                     Log.v(TAG, "16:9 N4, no HDR support");
783                     return false;
784                 } else {
785                     return mCameraCharacteristics.isHdrSceneSupported();
786                 }
787             }
788 
789             @Override
790             public boolean isHdrPlusSupported() {
791                 OneCameraFeatureConfig featureConfig = mAppController.getCameraFeatureConfig();
792                 return featureConfig.getHdrPlusSupportLevel(mCameraFacing) !=
793                         OneCameraFeatureConfig.HdrPlusSupportLevel.NONE;
794             }
795 
796             @Override
797             public boolean isFlashSupported() {
798                 return mCameraCharacteristics.isFlashSupported();
799             }
800         };
801     }
802 
803     @Override
804     public BottomBarUISpec getBottomBarSpec() {
805         HardwareSpec hardwareSpec = getHardwareSpec();
806         BottomBarUISpec bottomBarSpec = new BottomBarUISpec();
807         bottomBarSpec.enableGridLines = true;
808         bottomBarSpec.enableCamera = true;
809         bottomBarSpec.cameraCallback = getCameraCallback();
810         bottomBarSpec.enableHdr =
811                 hardwareSpec.isHdrSupported() || hardwareSpec.isHdrPlusSupported();
812         bottomBarSpec.hdrCallback = getHdrButtonCallback();
813         bottomBarSpec.enableSelfTimer = true;
814         bottomBarSpec.showSelfTimer = true;
815         bottomBarSpec.isExposureCompensationSupported = mCameraCharacteristics
816                 .isExposureCompensationSupported();
817         bottomBarSpec.enableExposureCompensation = bottomBarSpec.isExposureCompensationSupported;
818 
819         // We must read the key from the settings because the button callback
820         // is not executed until after this method is called.
821         if ((hardwareSpec.isHdrPlusSupported() &&
822                 mAppController.getSettingsManager().getBoolean(
823                 SettingsManager.SCOPE_GLOBAL, Keys.KEY_CAMERA_HDR_PLUS)) ||
824               ( hardwareSpec.isHdrSupported() &&
825                 mAppController.getSettingsManager().getBoolean(
826                 SettingsManager.SCOPE_GLOBAL, Keys.KEY_CAMERA_HDR))) {
827             // Disable flash if this is a sticky gcam camera, or if
828             // HDR is enabled.
829             bottomBarSpec.enableFlash = false;
830             // Disable manual exposure if HDR is enabled.
831             bottomBarSpec.enableExposureCompensation = false;
832         } else {
833             // If we are not in HDR / GCAM mode, fallback on the
834             // flash supported property and manual exposure supported property
835             // for this camera.
836             bottomBarSpec.enableFlash = mCameraCharacteristics.isFlashSupported();
837         }
838 
839         bottomBarSpec.minExposureCompensation =
840                 mCameraCharacteristics.getMinExposureCompensation();
841         bottomBarSpec.maxExposureCompensation =
842                 mCameraCharacteristics.getMaxExposureCompensation();
843         bottomBarSpec.exposureCompensationStep =
844                 mCameraCharacteristics.getExposureCompensationStep();
845         bottomBarSpec.exposureCompensationSetCallback =
846                 new BottomBarUISpec.ExposureCompensationSetCallback() {
847                     @Override
848                     public void setExposure(int value) {
849                         mSettingsManager.set(
850                                 mAppController.getCameraScope(), Keys.KEY_EXPOSURE, value);
851                     }
852                 };
853 
854         return bottomBarSpec;
855     }
856 
857     @Override
858     public boolean isUsingBottomBar() {
859         return true;
860     }
861 
862     @Override
863     public boolean onKeyDown(int keyCode, KeyEvent event) {
864         switch (keyCode) {
865             case KeyEvent.KEYCODE_CAMERA:
866             case KeyEvent.KEYCODE_DPAD_CENTER:
867                 if (mUI.isCountingDown()) {
868                     cancelCountDown();
869                 } else if (event.getRepeatCount() == 0) {
870                     onShutterButtonClick();
871                 }
872                 return true;
873             case KeyEvent.KEYCODE_VOLUME_UP:
874             case KeyEvent.KEYCODE_VOLUME_DOWN:
875                 // Prevent default.
876                 return true;
877         }
878         return false;
879     }
880 
881     @Override
882     public boolean onKeyUp(int keyCode, KeyEvent event) {
883         switch (keyCode) {
884             case KeyEvent.KEYCODE_VOLUME_UP:
885             case KeyEvent.KEYCODE_VOLUME_DOWN:
886                 onShutterButtonClick();
887                 return true;
888         }
889         return false;
890     }
891 
892     // TODO: Consider refactoring FocusOverlayManager.
893     // Currently AF state transitions are controlled in OneCameraImpl.
894     // PhotoModule uses FocusOverlayManager which uses API1/portability
895     // logic and coordinates.
896     private void startActiveFocusAt(int viewX, int viewY) {
897         if (mCamera == null) {
898             // If we receive this after the camera is closed, do nothing.
899             return;
900         }
901 
902         // TODO: make mFocusController final and remove null check.
903         if (mFocusController == null) {
904             Log.v(TAG, "CaptureModule mFocusController is null!");
905             return;
906         }
907         mFocusController.showActiveFocusAt(viewX, viewY);
908 
909         // Normalize coordinates to [0,1] per CameraOne API.
910         float points[] = new float[2];
911         points[0] = (viewX - mPreviewArea.left) / mPreviewArea.width();
912         points[1] = (viewY - mPreviewArea.top) / mPreviewArea.height();
913 
914         // Rotate coordinates to portrait orientation per CameraOne API.
915         Matrix rotationMatrix = new Matrix();
916         rotationMatrix.setRotate(mDisplayRotation, 0.5f, 0.5f);
917         rotationMatrix.mapPoints(points);
918 
919         // Invert X coordinate on front camera since the display is mirrored.
920         if (mCameraCharacteristics.getCameraDirection() == Facing.FRONT) {
921             points[0] = 1 - points[0];
922         }
923 
924         mCamera.triggerFocusAndMeterAtPoint(points[0], points[1]);
925 
926         // Log touch (screen coordinates).
927         if (mZoomValue == 1f) {
928             TouchCoordinate touchCoordinate = new TouchCoordinate(
929                     viewX - mPreviewArea.left,
930                     viewY - mPreviewArea.top,
931                     mPreviewArea.width(),
932                     mPreviewArea.height());
933             // TODO: Add to logging: duration, rotation.
934             UsageStatistics.instance().tapToFocus(touchCoordinate, null);
935         }
936     }
937 
938     /**
939      * Show AF target in center of preview.
940      */
941     private void startPassiveFocus() {
942         // TODO: make mFocusController final and remove null check.
943         if (mFocusController == null) {
944             return;
945         }
946 
947         // TODO: Some passive focus scans may trigger on a location
948         // instead of the center of the screen.
949         mFocusController.showPassiveFocusAtCenter();
950     }
951 
952     /**
953      * Update UI based on AF state changes.
954      */
955     @Override
956     public void onFocusStatusUpdate(final AutoFocusState state, long frameNumber) {
957         Log.v(TAG, "AF status is state:" + state);
958 
959         switch (state) {
960             case PASSIVE_SCAN:
961                 startPassiveFocus();
962                 break;
963             case ACTIVE_SCAN:
964                 // Unused, manual scans are triggered via the UI
965                 break;
966             case PASSIVE_FOCUSED:
967             case PASSIVE_UNFOCUSED:
968                 // Unused
969                 break;
970             case ACTIVE_FOCUSED:
971             case ACTIVE_UNFOCUSED:
972                 // Unused
973                 break;
974         }
975 
976         if (CAPTURE_DEBUG_UI) {
977             measureAutoFocusScans(state, frameNumber);
978         }
979     }
980 
981     private void measureAutoFocusScans(final AutoFocusState state, long frameNumber) {
982         // Log AF scan lengths.
983         boolean passive = false;
984         switch (state) {
985             case PASSIVE_SCAN:
986             case ACTIVE_SCAN:
987                 if (mAutoFocusScanStartFrame == -1) {
988                     mAutoFocusScanStartFrame = frameNumber;
989                     mAutoFocusScanStartTime = SystemClock.uptimeMillis();
990                 }
991                 break;
992             case PASSIVE_FOCUSED:
993             case PASSIVE_UNFOCUSED:
994                 passive = true;
995             case ACTIVE_FOCUSED:
996             case ACTIVE_UNFOCUSED:
997                 if (mAutoFocusScanStartFrame != -1) {
998                     long frames = frameNumber - mAutoFocusScanStartFrame;
999                     long dt = SystemClock.uptimeMillis() - mAutoFocusScanStartTime;
1000                     int fps = Math.round(frames * 1000f / dt);
1001                     String report = String.format("%s scan: fps=%d frames=%d",
1002                             passive ? "CAF" : "AF", fps, frames);
1003                     Log.v(TAG, report);
1004                     mUI.showDebugMessage(String.format("%d / %d", frames, fps));
1005                     mAutoFocusScanStartFrame = -1;
1006                 }
1007                 break;
1008         }
1009     }
1010 
1011     @Override
1012     public void onReadyStateChanged(boolean readyForCapture) {
1013         if (readyForCapture) {
1014             mAppController.getCameraAppUI().enableModeOptions();
1015         }
1016         mAppController.setShutterEnabled(readyForCapture);
1017     }
1018 
1019     @Override
1020     public String getPeekAccessibilityString() {
1021         return mAppController.getAndroidContext()
1022                 .getResources().getString(R.string.photo_accessibility_peek);
1023     }
1024 
1025     @Override
1026     public void onThumbnailResult(byte[] jpegData) {
1027         getServices().getRemoteShutterListener().onPictureTaken(jpegData);
1028     }
1029 
1030     @Override
1031     public void onPictureTaken(CaptureSession session) {
1032         mAppController.getCameraAppUI().enableModeOptions();
1033     }
1034 
1035     @Override
1036     public void onPictureSaved(Uri uri) {
1037         mAppController.notifyNewMedia(uri);
1038     }
1039 
1040     @Override
1041     public void onTakePictureProgress(float progress) {
1042         mUI.setPictureTakingProgress((int) (progress * 100));
1043     }
1044 
1045     @Override
1046     public void onPictureTakingFailed() {
1047         mAppController.getFatalErrorHandler().onMediaStorageFailure();
1048     }
1049 
1050     /**
1051      * Updates the preview transform matrix to adapt to the current preview
1052      * width, height, and orientation.
1053      */
1054     public void updatePreviewTransform() {
1055         int width;
1056         int height;
1057         synchronized (mDimensionLock) {
1058             width = mScreenWidth;
1059             height = mScreenHeight;
1060         }
1061         updatePreviewTransform(width, height);
1062     }
1063 
1064     /**
1065      * @return Depending on whether we're in sticky-HDR mode or not, return the
1066      *         proper callback to be used for when the HDR/HDR+ button is
1067      *         pressed.
1068      */
1069     private ButtonManager.ButtonCallback getHdrButtonCallback() {
1070         if (mStickyGcamCamera) {
1071             return new ButtonManager.ButtonCallback() {
1072                 @Override
1073                 public void onStateChanged(int state) {
1074                     if (mPaused) {
1075                         return;
1076                     }
1077                     if (state == ButtonManager.ON) {
1078                         throw new IllegalStateException(
1079                                 "Can't leave hdr plus mode if switching to hdr plus mode.");
1080                     }
1081                     SettingsManager settingsManager = mAppController.getSettingsManager();
1082                     settingsManager.set(mAppController.getModuleScope(),
1083                             Keys.KEY_REQUEST_RETURN_HDR_PLUS, false);
1084                     switchToRegularCapture();
1085                 }
1086             };
1087         } else {
1088             return new ButtonManager.ButtonCallback() {
1089                 @Override
1090                 public void onStateChanged(int hdrEnabled) {
1091                     if (mPaused) {
1092                         return;
1093                     }
1094 
1095                     // Only reload the camera if we are toggling HDR+.
1096                     if (GcamHelper.hasGcamCapture(mAppController.getCameraFeatureConfig())) {
1097                         mHdrPlusEnabled = hdrEnabled == 1;
1098                         switchCamera();
1099                     } else {
1100                         mHdrSceneEnabled = hdrEnabled == 1;
1101                     }
1102                 }
1103             };
1104         }
1105     }
1106 
1107     /**
1108      * @return Depending on whether we're in sticky-HDR mode or not, this
1109      *         returns the proper callback to be used for when the camera
1110      *         (front/back switch) button is pressed.
1111      */
1112     private ButtonManager.ButtonCallback getCameraCallback() {
1113         if (mStickyGcamCamera) {
1114             return new ButtonManager.ButtonCallback() {
1115                 @Override
1116                 public void onStateChanged(int state) {
1117                     if (mPaused) {
1118                         return;
1119                     }
1120 
1121                     // At the time this callback is fired, the camera id setting
1122                     // has changed to the desired camera.
1123                     SettingsManager settingsManager = mAppController.getSettingsManager();
1124                     if (Keys.isCameraBackFacing(settingsManager,
1125                             mAppController.getModuleScope())) {
1126                         throw new IllegalStateException(
1127                                 "Hdr plus should never be switching from front facing camera.");
1128                     }
1129 
1130                     // Switch to photo mode, but request a return to hdr plus on
1131                     // switching to back camera again.
1132                     settingsManager.set(mAppController.getModuleScope(),
1133                             Keys.KEY_REQUEST_RETURN_HDR_PLUS, true);
1134                     switchToRegularCapture();
1135                 }
1136             };
1137         } else {
1138             return new ButtonManager.ButtonCallback() {
1139                 @Override
1140                 public void onStateChanged(int cameraId) {
1141                     if (mPaused) {
1142                         return;
1143                     }
1144 
1145                     ButtonManager buttonManager = mAppController.getButtonManager();
1146                     buttonManager.disableCameraButtonAndBlock();
1147 
1148                     // At the time this callback is fired, the camera id
1149                     // has be set to the desired camera.
1150                     mSettingsManager.set(mAppController.getModuleScope(), Keys.KEY_CAMERA_ID,
1151                             cameraId);
1152 
1153                     Log.d(TAG, "Start to switch camera. cameraId=" + cameraId);
1154                     mCameraFacing = getFacingFromCameraId(cameraId);
1155                     mShowErrorAndFinish = !updateCameraCharacteristics();
1156                     switchCamera();
1157                 }
1158             };
1159         }
1160     }
1161 
1162     /**
1163      * Switches to PhotoModule to do regular photo captures.
1164      * <p>
1165      * TODO: Remove this once we use CaptureModule for photo taking.
1166      */
1167     private void switchToRegularCapture() {
1168         // Turn off HDR+ before switching back to normal photo mode.
1169         SettingsManager settingsManager = mAppController.getSettingsManager();
1170         settingsManager.set(SettingsManager.SCOPE_GLOBAL, Keys.KEY_CAMERA_HDR_PLUS, false);
1171 
1172         // Disable this button to prevent callbacks from this module from firing
1173         // while we are transitioning modules.
1174         ButtonManager buttonManager = mAppController.getButtonManager();
1175         buttonManager.disableButtonClick(ButtonManager.BUTTON_HDR_PLUS);
1176         mAppController.getCameraAppUI().freezeScreenUntilPreviewReady();
1177         mAppController.onModeSelected(mContext.getResources().getInteger(
1178                 R.integer.camera_mode_photo));
1179         buttonManager.enableButtonClick(ButtonManager.BUTTON_HDR_PLUS);
1180     }
1181 
1182     /**
1183      * Called when the preview started. Informs the app controller and queues a
1184      * transform update when the next preview frame arrives.
1185      */
1186     private void onPreviewStarted() {
1187         if (mState == ModuleState.WATCH_FOR_NEXT_FRAME_AFTER_PREVIEW_STARTED) {
1188             mState = ModuleState.UPDATE_TRANSFORM_ON_NEXT_SURFACE_TEXTURE_UPDATE;
1189         }
1190         mAppController.onPreviewStarted();
1191     }
1192 
1193     /**
1194      * Update the preview transform based on the new dimensions. Will not force
1195      * an update, if it's not necessary.
1196      */
1197     private void updatePreviewTransform(int incomingWidth, int incomingHeight) {
1198         updatePreviewTransform(incomingWidth, incomingHeight, false);
1199     }
1200 
1201     /**
1202      * Returns whether it is necessary to apply device-specific fix for b/19271661
1203      * on the AutoTransform Path, i.e. USE_AUTOTRANSFORM_UI_LAYOUT == true
1204      *
1205      * @return whether to apply workaround fix for b/19271661
1206      */
1207     private boolean requiresNexus4SpecificFixFor16By9Previews() {
1208         return USE_AUTOTRANSFORM_UI_LAYOUT && ApiHelper.IS_NEXUS_4
1209                 && is16by9AspectRatio(mPictureSize);
1210     }
1211 
1212     /***
1213      * Update the preview transform based on the new dimensions. TODO: Make work
1214      * with all: aspect ratios/resolutions x screens/cameras.
1215      */
1216     private void updatePreviewTransform(int incomingWidth, int incomingHeight,
1217             boolean forceUpdate) {
1218         Log.d(TAG, "updatePreviewTransform: " + incomingWidth + " x " + incomingHeight);
1219 
1220         synchronized (mDimensionLock) {
1221             int incomingRotation = CameraUtil.getDisplayRotation();
1222             // Check for an actual change:
1223             if (mScreenHeight == incomingHeight && mScreenWidth == incomingWidth &&
1224                     incomingRotation == mDisplayRotation && !forceUpdate) {
1225                 return;
1226             }
1227             // Update display rotation and dimensions
1228             mDisplayRotation = incomingRotation;
1229             mScreenWidth = incomingWidth;
1230             mScreenHeight = incomingHeight;
1231             updatePreviewBufferDimension();
1232 
1233             // Assumptions:
1234             // - Aspect ratio for the sensor buffers is in landscape
1235             // orientation,
1236             // - Dimensions of buffers received are rotated to the natural
1237             // device orientation.
1238             // - The contents of each buffer are rotated by the inverse of
1239             // the display rotation.
1240             // - Surface scales the buffer to fit the current view bounds.
1241 
1242             // Get natural orientation and buffer dimensions
1243 
1244             if(USE_AUTOTRANSFORM_UI_LAYOUT) {
1245                 // Use PhotoUI-based AutoTransformation Interface
1246                 if (mPreviewBufferWidth != 0 && mPreviewBufferHeight != 0) {
1247                     if (requiresNexus4SpecificFixFor16By9Previews()) {
1248                         // Force preview size to be 16:9, even though surface is 4:3
1249                         // Surface content is assumed to be 16:9.
1250                         mAppController.updatePreviewAspectRatio(16.f / 9.f);
1251                     } else {
1252                         mAppController.updatePreviewAspectRatio(
1253                                 mPreviewBufferWidth / (float) mPreviewBufferHeight);
1254                     }
1255                 }
1256             } else {
1257                 Matrix transformMatrix = mPreviewTransformCalculator.toTransformMatrix(
1258                         new Size(mScreenWidth, mScreenHeight),
1259                         new Size(mPreviewBufferWidth, mPreviewBufferHeight));
1260                 mAppController.updatePreviewTransform(transformMatrix);
1261             }
1262         }
1263     }
1264 
1265 
1266     /**
1267      * Calculates whether a picture size is 16:9 ratio, regardless of its
1268      * orientation.
1269      *
1270      * @param size the size of the picture to be considered
1271      * @return true, if the picture is 16:9; false if it's invalid or size is null
1272      */
1273     private boolean is16by9AspectRatio(Size size) {
1274         if (size == null || size.getWidth() == 0 || size.getHeight() == 0) {
1275             return false;
1276         }
1277 
1278         // Normalize aspect ratio to be greater than 1.
1279         final float aspectRatio = (size.getHeight() > size.getWidth())
1280                 ? (size.getHeight() / (float) size.getWidth())
1281                 : (size.getWidth() / (float) size.getHeight());
1282 
1283         return Math.abs(aspectRatio - (16.f / 9.f)) < 0.001f;
1284     }
1285 
1286     /**
1287      * Based on the current picture size, selects the best preview dimension and
1288      * stores it in {@link #mPreviewBufferWidth} and
1289      * {@link #mPreviewBufferHeight}.
1290      */
1291     private void updatePreviewBufferDimension() {
1292         if (mCamera == null) {
1293             return;
1294         }
1295 
1296         Size previewBufferSize = mCamera.pickPreviewSize(mPictureSize, mContext);
1297         mPreviewBufferWidth = previewBufferSize.getWidth();
1298         mPreviewBufferHeight = previewBufferSize.getHeight();
1299 
1300         // Workaround for N4 TextureView/HAL issues b/19271661 for 16:9 preview
1301         // streams.
1302         if (requiresNexus4SpecificFixFor16By9Previews()) {
1303             // Override the preview selection logic to the largest N4 4:3
1304             // preview size but pass in 16:9 aspect ratio in
1305             // UpdatePreviewAspectRatio later.
1306             mPreviewBufferWidth = 1280;
1307             mPreviewBufferHeight = 960;
1308         }
1309         updatePreviewBufferSize();
1310     }
1311 
1312     /**
1313      * Open camera and start the preview.
1314      */
1315     private void openCameraAndStartPreview() {
1316         Profile guard = mProfiler.create("CaptureModule.openCameraAndStartPreview()").start();
1317         try {
1318             // TODO Given the current design, we cannot guarantee that one of
1319             // CaptureReadyCallback.onSetupFailed or onReadyForCapture will
1320             // be called (see below), so it's possible that
1321             // mCameraOpenCloseLock.release() is never called under extremely
1322             // rare cases. If we leak the lock, this timeout ensures that we at
1323             // least crash so we don't deadlock the app.
1324             if (!mCameraOpenCloseLock.tryAcquire(CAMERA_OPEN_CLOSE_TIMEOUT_MILLIS,
1325                     TimeUnit.MILLISECONDS)) {
1326                 throw new RuntimeException("Time out waiting to acquire camera-open lock.");
1327             }
1328         } catch (InterruptedException e) {
1329             throw new RuntimeException("Interrupted while waiting to acquire camera-open lock.", e);
1330         }
1331 
1332         guard.mark("Acquired mCameraOpenCloseLock");
1333 
1334         if (mOneCameraOpener == null) {
1335             Log.e(TAG, "no available OneCameraManager, showing error dialog");
1336             mCameraOpenCloseLock.release();
1337             mAppController.getFatalErrorHandler().onGenericCameraAccessFailure();
1338             guard.stop("No OneCameraManager");
1339             return;
1340         }
1341         if (mCamera != null) {
1342             // If the camera is already open, do nothing.
1343             Log.d(TAG, "Camera already open, not re-opening.");
1344             mCameraOpenCloseLock.release();
1345             guard.stop("Camera is already open");
1346             return;
1347         }
1348 
1349         // Derive objects necessary for camera creation.
1350         MainThread mainThread = MainThread.create();
1351         ImageRotationCalculator imageRotationCalculator = ImageRotationCalculatorImpl
1352                 .from(mAppController.getOrientationManager(), mCameraCharacteristics);
1353 
1354         // Only enable GCam on the back camera
1355         boolean useHdr = mHdrPlusEnabled && mCameraFacing == Facing.BACK;
1356 
1357         CameraId cameraId = mOneCameraManager.findFirstCameraFacing(mCameraFacing);
1358         final String settingScope = SettingsManager.getCameraSettingScope(cameraId.getValue());
1359 
1360         OneCameraCaptureSetting captureSetting;
1361         // Read the preferred picture size from the setting.
1362         try {
1363             mPictureSize = mAppController.getResolutionSetting().getPictureSize(
1364                     cameraId, mCameraFacing);
1365             captureSetting = OneCameraCaptureSetting.create(mPictureSize, mSettingsManager,
1366                     getHardwareSpec(), settingScope, useHdr);
1367         } catch (OneCameraAccessException ex) {
1368             mAppController.getFatalErrorHandler().onGenericCameraAccessFailure();
1369             return;
1370         }
1371 
1372         mOneCameraOpener.open(cameraId, captureSetting, mCameraHandler, mainThread,
1373               imageRotationCalculator, mBurstController, mSoundPlayer,
1374               new OpenCallback() {
1375                   @Override
1376                   public void onFailure() {
1377                       Log.e(TAG, "Could not open camera.");
1378                       // Sometimes the failure happens due to the controller
1379                       // being in paused state but mCamera is already
1380                       // initialized.  In these cases we just need to close the
1381                       // camera device without showing the error dialog.
1382                       // Application will properly reopen the camera on the next
1383                       // resume operation (b/21025113).
1384                       boolean isControllerPaused = mAppController.isPaused();
1385                       if (mCamera != null) {
1386                           mCamera.close();
1387                       }
1388                       mCamera = null;
1389                       mCameraOpenCloseLock.release();
1390                       if (!isControllerPaused) {
1391                           mAppController.getFatalErrorHandler().onCameraOpenFailure();
1392                       }
1393                   }
1394 
1395                   @Override
1396                   public void onCameraClosed() {
1397                       mCamera = null;
1398                       mCameraOpenCloseLock.release();
1399                   }
1400 
1401                   @Override
1402                   public void onCameraOpened(@Nonnull final OneCamera camera) {
1403                       Log.d(TAG, "onCameraOpened: " + camera);
1404                       mCamera = camera;
1405 
1406                       // A race condition exists where the camera may be in the process
1407                       // of opening (blocked), but the activity gets destroyed. If the
1408                       // preview is initialized or callbacks are invoked on a destroyed
1409                       // activity, bad things can happen.
1410                       if (mAppController.isPaused()) {
1411                           onFailure();
1412                           return;
1413                       }
1414 
1415                       // When camera is opened, the zoom is implicitly reset to 1.0f
1416                       mZoomValue = 1.0f;
1417 
1418                       updatePreviewBufferDimension();
1419 
1420                       // If the surface texture is not destroyed, it may have
1421                       // the last frame lingering. We need to hold off setting
1422                       // transform until preview is started.
1423                       updatePreviewBufferSize();
1424                       mState = ModuleState.WATCH_FOR_NEXT_FRAME_AFTER_PREVIEW_STARTED;
1425                       Log.d(TAG, "starting preview ...");
1426 
1427                       // TODO: make mFocusController final and remove null
1428                       // check.
1429                       if (mFocusController != null) {
1430                           camera.setFocusDistanceListener(mFocusController);
1431                       }
1432 
1433                       mMainThread.execute(new Runnable() {
1434                           @Override
1435                           public void run() {
1436                               mAppController.getCameraAppUI().onChangeCamera();
1437                               mAppController.getButtonManager().enableCameraButton();
1438                           }
1439                       });
1440 
1441                       // TODO: Consider rolling these two calls into one.
1442                       camera.startPreview(new Surface(getPreviewSurfaceTexture()),
1443                             new CaptureReadyCallback() {
1444                                 @Override
1445                                 public void onSetupFailed() {
1446                                     // We must release this lock here,
1447                                     // before posting to the main handler
1448                                     // since we may be blocked in pause(),
1449                                     // getting ready to close the camera.
1450                                     mCameraOpenCloseLock.release();
1451                                     Log.e(TAG, "Could not set up preview.");
1452                                     mMainThread.execute(new Runnable() {
1453                                         @Override
1454                                         public void run() {
1455                                             if (mCamera == null) {
1456                                                 Log.d(TAG, "Camera closed, aborting.");
1457                                                 return;
1458                                             }
1459                                             mCamera.close();
1460                                             mCamera = null;
1461                                             // TODO: Show an error message
1462                                             // and exit.
1463                                         }
1464                                     });
1465                                 }
1466 
1467                                 @Override
1468                                 public void onReadyForCapture() {
1469                                     // We must release this lock here,
1470                                     // before posting to the main handler
1471                                     // since we may be blocked in pause(),
1472                                     // getting ready to close the camera.
1473                                     mCameraOpenCloseLock.release();
1474                                     mMainThread.execute(new Runnable() {
1475                                         @Override
1476                                         public void run() {
1477                                             Log.d(TAG, "Ready for capture.");
1478                                             if (mCamera == null) {
1479                                                 Log.d(TAG, "Camera closed, aborting.");
1480                                                 return;
1481                                             }
1482                                             onPreviewStarted();
1483                                             // May be overridden by
1484                                             // subsequent call to
1485                                             // onReadyStateChanged().
1486                                             onReadyStateChanged(true);
1487                                             mCamera.setReadyStateChangedListener(
1488                                                   CaptureModule.this);
1489                                             // Enable zooming after preview
1490                                             // has started.
1491                                             mUI.initializeZoom(mCamera.getMaxZoom());
1492                                             mCamera.setFocusStateListener(CaptureModule.this);
1493                                         }
1494                                     });
1495                                 }
1496                             });
1497                   }
1498               }, mAppController.getFatalErrorHandler());
1499         guard.stop("mOneCameraOpener.open()");
1500     }
1501 
1502     private void closeCamera() {
1503         Profile profile = mProfiler.create("CaptureModule.closeCamera()").start();
1504         try {
1505             mCameraOpenCloseLock.acquire();
1506         } catch (InterruptedException e) {
1507             throw new RuntimeException("Interrupted while waiting to acquire camera-open lock.", e);
1508         }
1509         profile.mark("mCameraOpenCloseLock.acquire()");
1510         try {
1511             if (mCamera != null) {
1512                 mCamera.close();
1513                 profile.mark("mCamera.close()");
1514                 mCamera.setFocusStateListener(null);
1515                 mCamera = null;
1516             }
1517         } finally {
1518             mCameraOpenCloseLock.release();
1519         }
1520         profile.stop();
1521     }
1522 
1523     /**
1524      * Re-initialize the camera if e.g. the HDR mode or facing property changed.
1525      */
1526     private void switchCamera() {
1527         if (mShowErrorAndFinish) {
1528             return;
1529         }
1530         if (mPaused) {
1531             return;
1532         }
1533         cancelCountDown();
1534         mAppController.freezeScreenUntilPreviewReady();
1535         initSurfaceTextureConsumer();
1536     }
1537 
1538     /**
1539      * Returns which way around the camera is facing, based on it's ID.
1540      * <p>
1541      * TODO: This needs to change so that we store the direction directly in the
1542      * settings, rather than a Camera ID.
1543      */
1544     private static Facing getFacingFromCameraId(int cameraId) {
1545         return cameraId == 1 ? Facing.FRONT : Facing.BACK;
1546     }
1547 
1548     private void resetTextureBufferSize() {
1549         // According to the documentation for
1550         // SurfaceTexture.setDefaultBufferSize,
1551         // photo and video based image producers (presumably only Camera 1 api),
1552         // override this buffer size. Any module that uses egl to render to a
1553         // SurfaceTexture must have these buffer sizes reset manually. Otherwise
1554         // the SurfaceTexture cannot be transformed by matrix set on the
1555         // TextureView.
1556         updatePreviewBufferSize();
1557     }
1558 }
1559