From c4ef8241b05b9266de77defcfdf1154f400c77f4 Mon Sep 17 00:00:00 2001 From: The Android Open Source Project Date: Fri, 20 Feb 2009 07:38:33 -0800 Subject: auto import from //branches/cupcake/...@132569 --- src/com/android/camera/Camera.java | 31 ++---- src/com/android/camera/GalleryPicker.java | 2 + src/com/android/camera/ImageGallery2.java | 65 ++++++++++-- src/com/android/camera/ImageManager.java | 12 +++ src/com/android/camera/MenuHelper.java | 34 ++++-- src/com/android/camera/ViewImage.java | 168 +++++++++--------------------- 6 files changed, 148 insertions(+), 164 deletions(-) (limited to 'src') diff --git a/src/com/android/camera/Camera.java b/src/com/android/camera/Camera.java index 9bf101c..04a30cf 100644 --- a/src/com/android/camera/Camera.java +++ b/src/com/android/camera/Camera.java @@ -39,12 +39,10 @@ import android.content.SharedPreferences; import android.content.pm.ActivityInfo; import android.content.res.AssetFileDescriptor; import android.content.res.Resources; -import android.database.Cursor; import android.graphics.Bitmap; import android.graphics.BitmapFactory; import android.graphics.Matrix; import android.graphics.drawable.BitmapDrawable; -import android.graphics.drawable.ColorDrawable; import android.graphics.drawable.Drawable; import android.graphics.drawable.LayerDrawable; import android.graphics.drawable.TransitionDrawable; @@ -62,25 +60,23 @@ import android.os.Debug; import android.os.Environment; import android.os.Handler; import android.os.Message; -import android.os.StatFs; import android.os.SystemClock; import android.preference.PreferenceManager; import android.provider.MediaStore; -import android.provider.MediaStore.Images; import android.text.format.DateFormat; import android.util.Config; import android.util.Log; import android.view.KeyEvent; import android.view.Menu; import android.view.MenuItem; +import android.view.MenuItem.OnMenuItemClickListener; import android.view.OrientationEventListener; import android.view.SurfaceHolder; import android.view.View; -import android.view.Window; -import android.view.WindowManager; -import android.view.MenuItem.OnMenuItemClickListener; import android.view.ViewGroup; import android.view.ViewGroup.LayoutParams; +import android.view.Window; +import android.view.WindowManager; import android.view.animation.AlphaAnimation; import android.view.animation.Animation; import android.view.animation.AnimationUtils; @@ -94,7 +90,6 @@ public class Camera extends Activity implements View.OnClickListener, private static final boolean DEBUG = false; private static final boolean DEBUG_TIME_OPERATIONS = DEBUG && false; - private static final boolean DEBUG_FAKE_GPS_LOCATION = DEBUG && false; private static final int CROP_MSG = 1; private static final int KEEP = 2; @@ -581,23 +576,7 @@ public class Camera extends Activity implements View.OnClickListener, mParameters.remove("gps-altitude"); mParameters.remove("gps-timestamp"); - if (DEBUG_FAKE_GPS_LOCATION) { - // Google London office, having trouble encoding longitude - - if (false) { - // This fails: - mParameters.set("gps-latitude", "51.49473309516907"); - mParameters.set("gps-longitude", "-0.14598190784454346"); - mParameters.set("gps-altitude", "71.0"); // meters - mParameters.set("gps-timestamp", "1233744883"); - } else { - // This works OK: - mParameters.set("gps-latitude", "51.49473309516907"); - mParameters.set("gps-longitude", "-1.0"); - mParameters.set("gps-altitude", "71.0"); // meters - mParameters.set("gps-timestamp", "1233744883"); - } - } else if (loc != null) { + if (loc != null) { double lat = loc.getLatitude(); double lon = loc.getLongitude(); boolean hasLatLon = (lat != 0.0d) || (lon != 0.0d); @@ -853,6 +832,8 @@ public class Camera extends Activity implements View.OnClickListener, loadServiceThread.join(); } catch (InterruptedException ex) { } + + ImageManager.ensureOSXCompatibleFolder(); } @Override diff --git a/src/com/android/camera/GalleryPicker.java b/src/com/android/camera/GalleryPicker.java index cf883dd..9c687c8 100644 --- a/src/com/android/camera/GalleryPicker.java +++ b/src/com/android/camera/GalleryPicker.java @@ -228,6 +228,8 @@ public class GalleryPicker extends Activity { }); } }); + + ImageManager.ensureOSXCompatibleFolder(); } private void launchFolderGallery(int position) { diff --git a/src/com/android/camera/ImageGallery2.java b/src/com/android/camera/ImageGallery2.java index 566bfcb..008eb21 100644 --- a/src/com/android/camera/ImageGallery2.java +++ b/src/com/android/camera/ImageGallery2.java @@ -19,8 +19,10 @@ package com.android.camera; import android.content.BroadcastReceiver; import android.app.Activity; import android.app.Dialog; +import android.app.AlertDialog; import android.app.ProgressDialog; import android.content.Context; +import android.content.DialogInterface; import android.content.Intent; import android.content.IntentFilter; import android.content.SharedPreferences; @@ -68,13 +70,13 @@ public class ImageGallery2 extends Activity { private View mNoImagesView; public final static int CROP_MSG = 2; public final static int VIEW_MSG = 3; - private static final String INSTANCE_STATE_TAG = "scrollY"; private Dialog mMediaScanningDialog; private MenuItem mSlideShowItem; private SharedPreferences mPrefs; + private long mVideoSizeLimit = Long.MAX_VALUE; public ImageGallery2() { } @@ -106,7 +108,13 @@ public class ImageGallery2 extends Activity { mGvs = (GridViewSpecial) findViewById(R.id.grid); mGvs.requestFocus(); - if (!isPickIntent()) { + if (isPickIntent()) { + mVideoSizeLimit = getIntent().getLongExtra( + MediaStore.EXTRA_SIZE_LIMIT, Long.MAX_VALUE); + mGvs.mVideoSizeLimit = mVideoSizeLimit; + } else { + mVideoSizeLimit = Long.MAX_VALUE; + mGvs.mVideoSizeLimit = mVideoSizeLimit; mGvs.setOnCreateContextMenuListener(new View.OnCreateContextMenuListener() { public void onCreateContextMenu(ContextMenu menu, View v, ContextMenu.ContextMenuInfo menuInfo) { if (mSelectedImageGetter.getCurrentImage() == null) @@ -306,7 +314,7 @@ public class ImageGallery2 extends Activity { default: handled = false; break; - } + } } if (handled) { mGvs.select(sel, pressed); @@ -324,6 +332,23 @@ public class ImageGallery2 extends Activity { private void launchCropperOrFinish(ImageManager.IImage img) { Bundle myExtras = getIntent().getExtras(); + if (MenuHelper.getImageFileSize(img) > mVideoSizeLimit) { + + DialogInterface.OnClickListener buttonListener = + new DialogInterface.OnClickListener() { + public void onClick(DialogInterface dialog, int which) { + dialog.dismiss(); + } + }; + new AlertDialog.Builder(this) + .setIcon(android.R.drawable.ic_dialog_info) + .setTitle(R.string.file_info_title) + .setMessage(R.string.video_exceed_mms_limit) + .setNeutralButton(R.string.details_ok, buttonListener) + .show(); + return; + } + String cropValue = myExtras != null ? myExtras.getString("crop") : null; if (cropValue != null) { Bundle newExtras = new Bundle(); @@ -714,6 +739,8 @@ public class ImageGallery2 extends Activity { private boolean mDirectionBiasDown = true; private final static boolean sDump = false; + private long mVideoSizeLimit; + class LayoutSpec { LayoutSpec(int cols, int w, int h, int leftEdgePadding, int rightEdgePadding, int intercellSpacing) { mColumns = cols; @@ -997,6 +1024,9 @@ public class ImageGallery2 extends Activity { private Bitmap mMissingImageThumbnailBitmap; private Bitmap mMissingVideoThumbnailBitmap; + private Drawable mVideoOverlay; + private Drawable mVideoMmsErrorOverlay; + public void dump() { synchronized (ImageBlockManager.this) { StringBuilder line1 = new StringBuilder(); @@ -1350,7 +1380,6 @@ public class ImageGallery2 extends Activity { int mRequestedMask; // columns which have been requested to the loader int mCompletedMask; // columns which have been completed from the loader boolean mIsVisible; - Drawable mVideoOverlay; public void dump(StringBuilder line1, StringBuilder line2) { synchronized (ImageBlock.this) { @@ -1523,17 +1552,31 @@ public class ImageGallery2 extends Activity { mCanvas.drawBitmap(error, source, dest, mPaint); } if (ImageManager.isVideo(image)) { - if (mVideoOverlay == null) { - mVideoOverlay = getResources().getDrawable( - R.drawable.ic_gallery_video_overlay); + Drawable overlay = null; + if (MenuHelper.getImageFileSize(image) <= mVideoSizeLimit) { + if (mVideoOverlay == null) { + mVideoOverlay = getResources().getDrawable( + R.drawable.ic_gallery_video_overlay); + } + overlay = mVideoOverlay; + } else { + if (mVideoMmsErrorOverlay == null) { + mVideoMmsErrorOverlay = getResources().getDrawable( + R.drawable.ic_error_mms_video_overlay); + } + overlay = mVideoMmsErrorOverlay; + Paint paint = new Paint(); + paint.setARGB(0x80, 0x00, 0x00, 0x00); + mCanvas.drawRect(xPos, yPos, xPos + mCurrentSpec.mCellWidth, + yPos + mCurrentSpec.mCellHeight, paint); } - int width = mVideoOverlay.getIntrinsicWidth(); - int height = mVideoOverlay.getIntrinsicHeight(); + int width = overlay.getIntrinsicWidth(); + int height = overlay.getIntrinsicHeight(); int left = (mCurrentSpec.mCellWidth - width) / 2 + xPos; int top = (mCurrentSpec.mCellHeight - height) / 2 + yPos; Rect newBounds = new Rect(left, top, left + width, top + height); - mVideoOverlay.setBounds(newBounds); - mVideoOverlay.draw(mCanvas); + overlay.setBounds(newBounds); + overlay.draw(mCanvas); } paintSel(base + baseOffset, xPos, yPos); } diff --git a/src/com/android/camera/ImageManager.java b/src/com/android/camera/ImageManager.java index 4a83958..cc76e83 100755 --- a/src/com/android/camera/ImageManager.java +++ b/src/com/android/camera/ImageManager.java @@ -73,6 +73,18 @@ 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. + */ + public static void ensureOSXCompatibleFolder() { + File nnnAAAAA = new File( + Environment.getExternalStorageDirectory().toString() + "/DCIM/100ANDRO"); + if ((!nnnAAAAA.exists()) && (!nnnAAAAA.mkdir())) { + Log.e(TAG, "create NNNAAAAA file: "+ nnnAAAAA.getPath()+" failed"); + } + } // To enable verbose logging for this class, change false to true. The other logic ensures that // this logging can be disabled by turned off DEBUG and lower, and that it can be enabled by diff --git a/src/com/android/camera/MenuHelper.java b/src/com/android/camera/MenuHelper.java index 2aedb02..ffb99ef 100644 --- a/src/com/android/camera/MenuHelper.java +++ b/src/com/android/camera/MenuHelper.java @@ -16,6 +16,7 @@ package com.android.camera; +import java.io.Closeable; import java.util.ArrayList; import android.app.Activity; @@ -101,6 +102,25 @@ public class MenuHelper { public void run(Uri uri, ImageManager.IImage image); } + private static void closeSilently(Closeable target) { + try { + if (target != null) target.close(); + } catch (Throwable t) { + // ignore all exceptions, that's what silently means + } + } + + public static long getImageFileSize(ImageManager.IImage image) { + java.io.InputStream data = image.fullSizeImageData(); + try { + return data.available(); + } catch (java.io.IOException ex) { + return -1; + } finally { + closeSilently(data); + } + } + static MenuItemsResult addImageMenuItems( Menu menu, int inclusions, @@ -264,17 +284,9 @@ public class MenuHelper { TextView textView = (TextView) d.findViewById(R.id.details_image_title); textView.setText(image.getDisplayName()); - java.io.InputStream data = image.fullSizeImageData(); - String lengthString = ""; - try { - long length = data.available(); - lengthString = - android.text.format.Formatter.formatFileSize(activity, length); - data.close(); - } catch (java.io.IOException ex) { - - } finally { - } + long length = getImageFileSize(image); + String lengthString = lengthString = length < 0 ? "" + : android.text.format.Formatter.formatFileSize(activity, length); ((TextView)d.findViewById(R.id.details_file_size_value)) .setText(lengthString); diff --git a/src/com/android/camera/ViewImage.java b/src/com/android/camera/ViewImage.java index 57d7051..ad27ae3 100644 --- a/src/com/android/camera/ViewImage.java +++ b/src/com/android/camera/ViewImage.java @@ -34,12 +34,12 @@ import android.provider.MediaStore; import android.util.AttributeSet; import android.util.Config; import android.util.Log; +import android.view.GestureDetector; import android.view.KeyEvent; import android.view.Menu; import android.view.MenuItem; import android.view.MotionEvent; import android.view.View; -import android.view.ViewConfiguration; import android.view.Window; import android.view.WindowManager; import android.view.animation.AlphaAnimation; @@ -225,21 +225,10 @@ public class ViewImage extends Activity implements View.OnClickListener static public class ImageViewTouch extends ImageViewTouchBase { private ViewImage mViewImage; private boolean mEnableTrackballScroll; - - private static int TOUCH_STATE_REST = 0; - private static int TOUCH_STATE_LEFT_PRESS = 1; - private static int TOUCH_STATE_RIGHT_PRESS = 2; - private static int TOUCH_STATE_PANNING = 3; - + private GestureDetector mGestureDetector; + private static int TOUCH_AREA_WIDTH = 60; - private int mTouchState = TOUCH_STATE_REST; - - // The event time of the previous touch up. - private long mPreviousUpTime; - // The duration in milliseconds we will wait to see if it is a double tap. - private static final int DOUBLE_TAP_TIMEOUT = ViewConfiguration.getDoubleTapTimeout(); - // Returns current scale step (numbered from 0 to 10). private int getCurrentStep() { float s = getScale(); @@ -356,18 +345,20 @@ public class ViewImage extends Activity implements View.OnClickListener public ImageViewTouch(Context context) { super(context); - mViewImage = (ViewImage) context; - - mZoomRingController = new ZoomRingController(context, this); - mZoomRingController.setCallback(mZoomListener); + setup(context); } public ImageViewTouch(Context context, AttributeSet attrs) { super(context, attrs); - mViewImage = (ViewImage) context; + setup(context); + } + private void setup(Context context) { + mViewImage = (ViewImage) context; mZoomRingController = new ZoomRingController(context, this); mZoomRingController.setCallback(mZoomListener); + mGestureDetector = new GestureDetector(new MyGestureListener()); + mGestureDetector.setOnDoubleTapListener(new MyDoubleTapListener()); } public void setEnableTrackballScroll(boolean enable) { @@ -389,105 +380,47 @@ public class ViewImage extends Activity implements View.OnClickListener } public boolean handleTouchEvent(MotionEvent m) { - int viewWidth = getWidth(); - ViewImage viewImage = mViewImage; - int x = (int) m.getX(); - int y = (int) m.getY(); - - switch (m.getAction()) { - case MotionEvent.ACTION_DOWN: - long downTime = m.getEventTime(); - if ((downTime - mPreviousUpTime) < DOUBLE_TAP_TIMEOUT) { - mZoomRingController.setVisible(true); - } else { - viewImage.setMode(MODE_NORMAL); - viewImage.showOnScreenControls(); - mLastXTouchPos = x; - mLastYTouchPos = y; - mTouchState = TOUCH_STATE_REST; - } - break; - case MotionEvent.ACTION_MOVE: - if (x < TOUCH_AREA_WIDTH) { - if (mTouchState == TOUCH_STATE_REST) { - mTouchState = TOUCH_STATE_LEFT_PRESS; - } - if (mTouchState == TOUCH_STATE_LEFT_PRESS) { - viewImage.mPrevImageView.setPressed(true); - viewImage.mNextImageView.setPressed(false); - } - mLastXTouchPos = x; - mLastYTouchPos = y; - } else if (x > viewWidth - TOUCH_AREA_WIDTH) { - if (mTouchState == TOUCH_STATE_REST) { - mTouchState = TOUCH_STATE_RIGHT_PRESS; - } - if (mTouchState == TOUCH_STATE_RIGHT_PRESS) { - viewImage.mPrevImageView.setPressed(false); - viewImage.mNextImageView.setPressed(true); - } - mLastXTouchPos = x; - mLastYTouchPos = y; - } else { - mTouchState = TOUCH_STATE_PANNING; - viewImage.mPrevImageView.setPressed(false); - viewImage.mNextImageView.setPressed(false); - - int deltaX; - int deltaY; - - if (mLastXTouchPos == -1) { - deltaX = 0; - deltaY = 0; - } else { - deltaX = x - mLastXTouchPos; - deltaY = y - mLastYTouchPos; - } - - mLastXTouchPos = x; - mLastYTouchPos = y; + return mGestureDetector.onTouchEvent(m); + } - if (mBitmapDisplayed == null) - return true; + private class MyGestureListener extends GestureDetector.SimpleOnGestureListener { + public boolean onScroll(MotionEvent e1, MotionEvent e2, + float distanceX, float distanceY) { + if (getScale() > 1F) { + postTranslate(-distanceX, -distanceY, sUseBounce); + ImageViewTouch.this.center(true, true, false); + } + return true; + } + } + + private class MyDoubleTapListener implements GestureDetector.OnDoubleTapListener { + // On single tap, we show the arrows. We also change to the + // prev/next image if the user taps on the left/right region. + public boolean onSingleTapConfirmed(MotionEvent e) { + ViewImage viewImage = mViewImage; + + int viewWidth = getWidth(); + int x = (int) e.getX(); + int y = (int) e.getY(); + if (x < TOUCH_AREA_WIDTH) { + viewImage.moveNextOrPrevious(-1); + } else if (x > viewWidth - TOUCH_AREA_WIDTH) { + viewImage.moveNextOrPrevious(1); + } + + viewImage.setMode(MODE_NORMAL); + viewImage.showOnScreenControls(); - if (deltaX != 0) { - // Second. Pan to whatever degree is possible. - if (getScale() > 1F) { - postTranslate(deltaX, deltaY, sUseBounce); - ImageViewTouch.this.center(true, true, false); - } - } - setImageMatrix(getImageViewMatrix()); - } - break; - case MotionEvent.ACTION_UP: - int nextImagePos = -1; - if (mTouchState == TOUCH_STATE_LEFT_PRESS && x < TOUCH_AREA_WIDTH) { - nextImagePos = viewImage.mCurrentPosition - 1; - } else if (mTouchState == TOUCH_STATE_RIGHT_PRESS && - x > viewWidth - TOUCH_AREA_WIDTH) { - nextImagePos = viewImage.mCurrentPosition + 1; - } - if (nextImagePos >= 0 - && nextImagePos < viewImage.mAllImages.getCount()) { - synchronized (viewImage) { - viewImage.setMode(MODE_NORMAL); - viewImage.setImage(nextImagePos); - } - } - viewImage.scheduleDismissOnScreenControls(); - viewImage.mPrevImageView.setPressed(false); - viewImage.mNextImageView.setPressed(false); - mTouchState = TOUCH_STATE_REST; - mPreviousUpTime = m.getEventTime(); - break; - case MotionEvent.ACTION_CANCEL: - viewImage.mPrevImageView.setPressed(false); - viewImage.mNextImageView.setPressed(false); - mTouchState = TOUCH_STATE_REST; - break; + return true; + } + + // On double tap, we show the zoom ring control. + public boolean onDoubleTapEvent(MotionEvent e) { + mViewImage.setMode(MODE_NORMAL); + mZoomRingController.setVisible(true); + return true; } - return true; } @Override @@ -1254,13 +1187,14 @@ public class ViewImage extends Activity implements View.OnClickListener mNextImageView = findViewById(R.id.next_image); mPrevImageView = findViewById(R.id.prev_image); + mNextImageView.setOnClickListener(this); + mPrevImageView.setOnClickListener(this); if (mShowActionIcons) { - mNextImageView.setOnClickListener(this); mNextImageView.setFocusable(true); - mPrevImageView.setOnClickListener(this); mPrevImageView.setFocusable(true); } + setOrientation(); // Show a tutorial for the new zoom interaction (the method ensure we only show it once) -- cgit v1.1