diff options
Diffstat (limited to 'src')
-rw-r--r-- | src/com/android/camera/Camera.java | 122 | ||||
-rw-r--r-- | src/com/android/camera/ImageGallery2.java | 15 | ||||
-rwxr-xr-x | src/com/android/camera/ImageManager.java | 8 | ||||
-rw-r--r-- | src/com/android/camera/MenuHelper.java | 1 | ||||
-rw-r--r-- | src/com/android/camera/VideoCamera.java | 75 | ||||
-rw-r--r-- | src/com/android/camera/ViewImage.java | 21 |
6 files changed, 123 insertions, 119 deletions
diff --git a/src/com/android/camera/Camera.java b/src/com/android/camera/Camera.java index da4c41c..9717d5e 100644 --- a/src/com/android/camera/Camera.java +++ b/src/com/android/camera/Camera.java @@ -144,10 +144,12 @@ public class Camera extends Activity implements View.OnClickListener, private boolean mPausing = false; - private boolean mIsFocusing = false; - private boolean mIsFocused = false; - private boolean mIsFocusButtonPressed = false; - private boolean mCaptureOnFocus = false; + private static final int FOCUS_NOT_STARTED = 0; + private static final int FOCUSING = 1; + private static final int FOCUSING_SNAP_ON_FINISH = 2; + private static final int FOCUS_SUCCESS = 3; + private static final int FOCUS_FAIL = 4; + private int mFocusState = FOCUS_NOT_STARTED; private static ContentResolver mContentResolver; private boolean mDidRegister = false; @@ -381,20 +383,27 @@ public class Camera extends Activity implements View.OnClickListener, Log.v(TAG, "Auto focus took " + (mFocusCallbackTime - mFocusStartTime) + " ms."); } - mIsFocusing = false; - mIsFocused = focused; - if (focused) { - if (mCaptureOnFocus && mCaptureObject != null) { - // No need to play the AF sound if we're about to play the shutter sound - mCaptureObject.onSnap(); - clearFocus(); - } else { - ToneGenerator tg = mFocusToneGenerator; - if (tg != null) - tg.startTone(ToneGenerator.TONE_PROP_BEEP2); + if (mFocusState == FOCUSING_SNAP_ON_FINISH && mCaptureObject != null) { + // Take the picture no matter focus succeeds or fails. + // No need to play the AF sound if we're about to play the shutter sound. + mCaptureObject.onSnap(); + clearFocusState(); + } else if (mFocusState == FOCUSING) { + // User is half-pressing the focus key. Play the focus tone. + // Do not take the picture now. + ToneGenerator tg = mFocusToneGenerator; + if (tg != null) + tg.startTone(ToneGenerator.TONE_PROP_BEEP2); + if (focused) { + mFocusState = FOCUS_SUCCESS; } - mCaptureOnFocus = false; - } + else { + mFocusState = FOCUS_FAIL; + } + } else if (mFocusState == FOCUS_NOT_STARTED) { + // User has released the focus key before focus completes. + // Do nothing. + } updateFocusIndicator(); } }; @@ -769,7 +778,11 @@ public class Camera extends Activity implements View.OnClickListener, mLocationManager = (LocationManager) getSystemService(Context.LOCATION_SERVICE); mOrientationListener = new OrientationEventListener(Camera.this) { public void onOrientationChanged(int orientation) { - mLastOrientation = orientation; + // We keep the last known orientation. So if the user + // first orient the camera then point the camera to + // floor/sky, we still have the correct orientation. + if (orientation != ORIENTATION_UNKNOWN) + mLastOrientation = orientation; } }; } @@ -993,7 +1006,7 @@ public class Camera extends Activity implements View.OnClickListener, public void onShutterButtonClick(ShutterButton button) { switch (button.getId()) { case R.id.shutter_button: - doSnap(false); + doSnap(); break; } } @@ -1204,38 +1217,33 @@ public class Camera extends Activity implements View.OnClickListener, private void autoFocus() { updateFocusIndicator(); - if (!mIsFocusing) { + if (mFocusState != FOCUSING && mFocusState != FOCUSING_SNAP_ON_FINISH) { if (mCameraDevice != null) { - mIsFocusing = true; - mIsFocused = false; - mCameraDevice.autoFocus(mAutoFocusCallback); if (DEBUG_TIME_OPERATIONS) { mFocusStartTime = System.currentTimeMillis(); } + mFocusState = FOCUSING; + mCameraDevice.autoFocus(mAutoFocusCallback); } } } - private void clearFocus() { - mIsFocusing = false; - mIsFocused = false; - mIsFocusButtonPressed = false; + private void clearFocusState() { + mFocusState = FOCUS_NOT_STARTED; } private void updateFocusIndicator() { mHandler.post(new Runnable() { public void run() { - if (mIsFocusing || !mIsFocusButtonPressed) { - mFocusIndicator.setVisibility(View.GONE); + if (mFocusState == FOCUS_SUCCESS) { + mFocusIndicator.setVisibility(View.VISIBLE); mFocusIndicator.clearAnimation(); + } else if (mFocusState == FOCUS_FAIL) { + mFocusIndicator.setVisibility(View.VISIBLE); + mFocusIndicator.startAnimation(mFocusBlinkAnimation); } else { - if (mIsFocused) { - mFocusIndicator.setVisibility(View.VISIBLE); - mFocusIndicator.clearAnimation(); - } else { - mFocusIndicator.setVisibility(View.VISIBLE); - mFocusIndicator.startAnimation(mFocusBlinkAnimation); - } + mFocusIndicator.setVisibility(View.GONE); + mFocusIndicator.clearAnimation(); } } }); @@ -1261,7 +1269,7 @@ public class Camera extends Activity implements View.OnClickListener, return true; case KeyEvent.KEYCODE_CAMERA: if (event.getRepeatCount() == 0) { - doSnap(false); + doSnap(); } return true; case KeyEvent.KEYCODE_DPAD_CENTER: @@ -1294,43 +1302,39 @@ public class Camera extends Activity implements View.OnClickListener, return super.onKeyUp(keyCode, event); } - private void doSnap(boolean needAutofocus) { - // The camera operates in focus-priority mode, meaning that we take a picture - // when focusing completes, and only if it completes successfully. If the user - // has half-pressed the shutter and already locked focus, we can take the photo - // right away, otherwise we need to start AF. - if (mIsFocused || !mPreviewing) { + private void doSnap() { + // If the user has half-pressed the shutter and focus is completed, we + // can take the photo right away. + if ((mFocusState == FOCUS_SUCCESS || mFocusState == FOCUS_FAIL) + || !mPreviewing) { // doesn't get set until the idler runs if (mCaptureObject != null) { mCaptureObject.onSnap(); } - clearFocus(); + clearFocusState(); updateFocusIndicator(); } else { - // Half pressing the shutter (i.e. the focus button event) will already have - // requested AF for us, so just request capture on focus here. If AF has - // already failed, we don't want to trigger it again. - mCaptureOnFocus = true; - if (needAutofocus && !mIsFocusButtonPressed) { - // But we do need to start AF for DPAD_CENTER - autoFocus(); - } + // Half pressing the shutter (i.e. the focus button event) will + // already have requested AF for us, so just request capture on + // focus here. + mFocusState = FOCUSING_SNAP_ON_FINISH; } } private void doFocus(boolean pressed) { - if (pressed) { - mIsFocusButtonPressed = true; - mCaptureOnFocus = false; + if (pressed) { // Focus key down. if (mPreviewing) { autoFocus(); } else if (mCaptureObject != null) { // Save and restart preview mCaptureObject.onSnap(); } - } else { - clearFocus(); - updateFocusIndicator(); + } else { // Focus key up. + if (mFocusState != FOCUSING_SNAP_ON_FINISH) { + // User releases half-pressed focus key. + clearFocusState(); + updateFocusIndicator(); + } } } @@ -1414,7 +1418,7 @@ public class Camera extends Activity implements View.OnClickListener, // let the user take a picture, and delete that file if needed to save the new photo. calculatePicturesRemaining(); - if (!isLastPictureValid()) { + if (!isImageCaptureIntent() && !isLastPictureValid()) { updateLastImage(); } diff --git a/src/com/android/camera/ImageGallery2.java b/src/com/android/camera/ImageGallery2.java index 008eb21..365be0f 100644 --- a/src/com/android/camera/ImageGallery2.java +++ b/src/com/android/camera/ImageGallery2.java @@ -254,6 +254,10 @@ public class ImageGallery2 extends Activity { @Override public boolean onKeyDown(int keyCode, KeyEvent event) { + if (mGvs.mCurrentSpec == null) { + // View.onLayout hasn't been called so we can't handle onKeyDown event yet. + return false; + } boolean handled = true; int sel = mGvs.mCurrentSelection; int columns = mGvs.mCurrentSpec.mColumns; @@ -332,7 +336,13 @@ public class ImageGallery2 extends Activity { private void launchCropperOrFinish(ImageManager.IImage img) { Bundle myExtras = getIntent().getExtras(); - if (MenuHelper.getImageFileSize(img) > mVideoSizeLimit) { + long size = MenuHelper.getImageFileSize(img); + if (size < 0) { + // return if there image file is not available. + return; + } + + if (size > mVideoSizeLimit) { DialogInterface.OnClickListener buttonListener = new DialogInterface.OnClickListener() { @@ -1553,7 +1563,8 @@ public class ImageGallery2 extends Activity { } if (ImageManager.isVideo(image)) { Drawable overlay = null; - if (MenuHelper.getImageFileSize(image) <= mVideoSizeLimit) { + long size = MenuHelper.getImageFileSize(image); + if (size >= 0 && size <= mVideoSizeLimit) { if (mVideoOverlay == null) { mVideoOverlay = getResources().getDrawable( R.drawable.ic_gallery_video_overlay); diff --git a/src/com/android/camera/ImageManager.java b/src/com/android/camera/ImageManager.java index 99e5366..6c3472b 100755 --- a/src/com/android/camera/ImageManager.java +++ b/src/com/android/camera/ImageManager.java @@ -73,7 +73,7 @@ public class ImageManager { public static String getBucketId(String path) { return String.valueOf(path.toLowerCase().hashCode()); } - + /** * OSX requires plugged-in USB storage to have path /DCIM/NNNAAAAA to be imported. * This is a temporary fix for bug#1655552. @@ -4123,12 +4123,14 @@ public class ImageManager { } static public boolean hasStorage(boolean requireWriteAccess) { + //TODO: After fix the bug, add "if (VERBOSE)" before logging errors. String state = Environment.getExternalStorageState(); - if (VERBOSE) Log.v(TAG, "state is " + state); + Log.v(TAG, "storage state is " + state); + if (Environment.MEDIA_MOUNTED.equals(state)) { if (requireWriteAccess) { boolean writable = checkFsWritable(); - if (VERBOSE) Log.v(TAG, "writable is " + writable); + Log.v(TAG, "storage writable is " + writable); return writable; } else { return true; diff --git a/src/com/android/camera/MenuHelper.java b/src/com/android/camera/MenuHelper.java index 7148cd6..2bb532e 100644 --- a/src/com/android/camera/MenuHelper.java +++ b/src/com/android/camera/MenuHelper.java @@ -112,6 +112,7 @@ public class MenuHelper { public static long getImageFileSize(ImageManager.IImage image) { java.io.InputStream data = image.fullSizeImageData(); + if (data == null) return -1; try { return data.available(); } catch (java.io.IOException ex) { diff --git a/src/com/android/camera/VideoCamera.java b/src/com/android/camera/VideoCamera.java index 37e1dcb..e9ffbe4 100644 --- a/src/com/android/camera/VideoCamera.java +++ b/src/com/android/camera/VideoCamera.java @@ -119,13 +119,11 @@ public class VideoCamera extends Activity implements View.OnClickListener, boolean mPausing = false; static ContentResolver mContentResolver; - boolean mDidRegister = false; int mCurrentZoomIndex = 0; private ShutterButton mShutterButton; private TextView mRecordingTimeView; - private boolean mHasSdCard; ArrayList<MenuItem> mGalleryItems = new ArrayList<MenuItem>(); @@ -195,25 +193,20 @@ public class VideoCamera extends Activity implements View.OnClickListener, public void onReceive(Context context, Intent intent) { String action = intent.getAction(); if (action.equals(Intent.ACTION_MEDIA_EJECT)) { - mHasSdCard = false; + updateStorageHint(false); stopVideoRecording(); initializeVideo(); } else if (action.equals(Intent.ACTION_MEDIA_MOUNTED)) { - // SD card available - // TODO put up a "please wait" message - // TODO also listen for the media scanner finished message - updateStorageHint(); - mHasSdCard = true; + updateStorageHint(true); initializeVideo(); } else if (action.equals(Intent.ACTION_MEDIA_UNMOUNTED)) { // SD card unavailable - updateStorageHint(); - mHasSdCard = false; + updateStorageHint(false); releaseMediaRecorder(); } else if (action.equals(Intent.ACTION_MEDIA_SCANNER_STARTED)) { Toast.makeText(VideoCamera.this, getResources().getString(R.string.wait), 5000); } else if (action.equals(Intent.ACTION_MEDIA_SCANNER_FINISHED)) { - updateStorageHint(); + updateStorageHint(true); } } }; @@ -266,29 +259,6 @@ public class VideoCamera extends Activity implements View.OnClickListener, mVideoFrame = (ImageView) findViewById(R.id.video_frame); } - @Override - public void onStart() { - if (DEBUG_LOG_APP_LIFECYCLE) { - Log.v(TAG, "onStart " + this.hashCode()); - } - super.onStart(); - - Thread t = new Thread(new Runnable() { - public void run() { - final boolean storageOK = getAvailableStorage() >= LOW_STORAGE_THRESHOLD; - - if (!storageOK) { - mHandler.post(new Runnable() { - public void run() { - updateStorageHint(); - } - }); - } - } - }); - t.start(); - } - public void onClick(View v) { switch (v.getId()) { @@ -375,17 +345,13 @@ public class VideoCamera extends Activity implements View.OnClickListener, private OnScreenHint mStorageHint; - private void updateStorageHint() { - long remaining = getAvailableStorage(); + private void updateStorageHint(boolean mayHaveSd) { + long remaining = mayHaveSd ? getAvailableStorage() : NO_STORAGE_ERROR; String errorMessage = null; if (remaining == NO_STORAGE_ERROR) { errorMessage = getString(R.string.no_storage); } else if (remaining < LOW_STORAGE_THRESHOLD) { errorMessage = getString(R.string.spaceIsLow_content); - if (mStorageHint != null) { - mStorageHint.cancel(); - mStorageHint = null; - } } if (errorMessage != null) { if (mStorageHint == null) { @@ -419,8 +385,7 @@ public class VideoCamera extends Activity implements View.OnClickListener, intentFilter.addAction(Intent.ACTION_MEDIA_SCANNER_FINISHED); intentFilter.addDataScheme("file"); registerReceiver(mReceiver, intentFilter); - mDidRegister = true; - mHasSdCard = ImageManager.hasStorage(); + updateStorageHint(true); mBlackout.setVisibility(View.INVISIBLE); if (mVideoFrameBitmap == null) { @@ -452,10 +417,7 @@ public class VideoCamera extends Activity implements View.OnClickListener, mPausing = true; - if (mDidRegister) { - unregisterReceiver(mReceiver); - mDidRegister = false; - } + unregisterReceiver(mReceiver); mBlackout.setVisibility(View.VISIBLE); setScreenTimeoutSystemDefault(); @@ -685,7 +647,7 @@ public class VideoCamera extends Activity implements View.OnClickListener, mMediaRecorder.setVideoSource(MediaRecorder.VideoSource.CAMERA); mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.THREE_GPP); - if (!mHasSdCard) { + if (mStorageHint != null) { mMediaRecorder.setOutputFile("/dev/null"); } else { // We try Uri in intent first. If it doesn't work, use our own instead. @@ -864,10 +826,23 @@ public class VideoCamera extends Activity implements View.OnClickListener, if (what == MediaRecorder.MEDIA_RECORDER_ERROR_UNKNOWN) { // We may have run out of space on the sdcard. stopVideoRecording(); - updateStorageHint(); + updateStorageHint(true); } } + /* + * Make sure we're not recording music playing in the background, ask + * the MediaPlaybackService to pause playback. + */ + private void pauseAudioPlayback() { + // Shamelessly copied from MediaPlaybackService.java, which + // should be public, but isn't. + Intent i = new Intent("com.android.music.musicservicecommand"); + i.putExtra("command", "pause"); + + sendBroadcast(i); + } + private void startVideoRecording() { Log.v(TAG, "startVideoRecording"); if (!mMediaRecorderRecording) { @@ -883,6 +858,8 @@ public class VideoCamera extends Activity implements View.OnClickListener, return; } + pauseAudioPlayback(); + try { mMediaRecorder.setOnErrorListener(this); mMediaRecorder.start(); // Recording is now started @@ -973,7 +950,7 @@ public class VideoCamera extends Activity implements View.OnClickListener, mRecordingTimeView.setVisibility(View.GONE); setScreenTimeoutLong(); } - if (needToRegisterRecording && mHasSdCard) registerVideo(); + if (needToRegisterRecording && mStorageHint == null) registerVideo(); mCameraVideoFilename = null; mCameraVideoFileDescriptor = null; diff --git a/src/com/android/camera/ViewImage.java b/src/com/android/camera/ViewImage.java index 07d396f..f6d0744 100644 --- a/src/com/android/camera/ViewImage.java +++ b/src/com/android/camera/ViewImage.java @@ -341,9 +341,19 @@ public class ViewImage extends Activity implements View.OnClickListener return (oldScale != getScale()); } - public void onSimpleZoom(boolean zoomIn) { - if (zoomIn) zoomIn(); - else zoomOut(); + public void onSimpleZoom(boolean zoomIn, int centerX, int centerY) { + // First move centerX/centerY to the center of the view. + int deltaX = getWidth() / 2 - centerX; + int deltaY = getHeight() / 2 - centerY; + panBy(deltaX, deltaY); + + if (zoomIn) { + zoomIn(); + } else { + zoomOut(); + } + + panBy(-deltaX, -deltaY); } }; @@ -361,8 +371,7 @@ public class ViewImage extends Activity implements View.OnClickListener mViewImage = (ViewImage) context; mZoomRingController = new ZoomRingController(context, this); mZoomRingController.setVibration(false); - mZoomRingController.setZoomCallbackThreshold(STEP_ANGLE); - mZoomRingController.setResetThumbAutomatically(false); + mZoomRingController.setTickDelta(STEP_ANGLE); mZoomRingController.setCallback(mZoomListener); mGestureDetector = new GestureDetector(getContext(), new MyGestureListener()); mGestureDetector.setOnDoubleTapListener(new MyDoubleTapListener()); @@ -382,7 +391,7 @@ public class ViewImage extends Activity implements View.OnClickListener } public void onZoom(boolean zoomIn) { - mZoomListener.onSimpleZoom(zoomIn); + mZoomListener.onSimpleZoom(zoomIn, getWidth()/2, getHeight()/2); } }); } |