package it.sephiroth.android.library.imagezoom; import android.content.Context; import android.graphics.Matrix; import android.graphics.Rect; import android.graphics.RectF; import android.graphics.drawable.Drawable; import android.util.AttributeSet; import android.util.Log; import android.view.GestureDetector; import android.view.GestureDetector.OnGestureListener; import android.view.MotionEvent; import android.view.ScaleGestureDetector; import android.view.ScaleGestureDetector.OnScaleGestureListener; import android.view.ViewConfiguration; public class ImageViewTouch extends ImageViewTouchBase { static final float SCROLL_DELTA_THRESHOLD = 1.0f; protected ScaleGestureDetector mScaleDetector; protected GestureDetector mGestureDetector; protected int mTouchSlop; protected float mScaleFactor; protected int mDoubleTapDirection; protected OnGestureListener mGestureListener; protected OnScaleGestureListener mScaleListener; protected boolean mDoubleTapEnabled = true; protected boolean mScaleEnabled = true; protected boolean mScrollEnabled = true; private OnImageViewTouchDoubleTapListener mDoubleTapListener; private OnImageViewTouchSingleTapListener mSingleTapListener; public ImageViewTouch(Context context) { super(context); } public ImageViewTouch(Context context, AttributeSet attrs) { this(context, attrs, 0); } public ImageViewTouch(Context context, AttributeSet attrs, int defStyle) { super(context, attrs, defStyle); } @Override protected void init(Context context, AttributeSet attrs, int defStyle) { super.init(context, attrs, defStyle); mTouchSlop = ViewConfiguration.get(getContext()).getScaledTouchSlop(); mGestureListener = getGestureListener(); mScaleListener = getScaleListener(); mScaleDetector = new ScaleGestureDetector(getContext(), mScaleListener); mGestureDetector = new GestureDetector(getContext(), mGestureListener, null, true); mDoubleTapDirection = 1; } public void setDoubleTapListener(OnImageViewTouchDoubleTapListener listener) { mDoubleTapListener = listener; } public void setSingleTapListener(OnImageViewTouchSingleTapListener listener) { mSingleTapListener = listener; } public void setDoubleTapEnabled(boolean value) { mDoubleTapEnabled = value; } public void setScaleEnabled(boolean value) { mScaleEnabled = value; } public void setScrollEnabled(boolean value) { mScrollEnabled = value; } public boolean getDoubleTapEnabled() { return mDoubleTapEnabled; } protected OnGestureListener getGestureListener() { return new GestureListener(); } protected OnScaleGestureListener getScaleListener() { return new ScaleListener(); } @Override protected void _setImageDrawable(final Drawable drawable, final Matrix initial_matrix, float min_zoom, float max_zoom) { super._setImageDrawable(drawable, initial_matrix, min_zoom, max_zoom); mScaleFactor = getMaxScale() / 3; } @Override public boolean onTouchEvent(MotionEvent event) { mScaleDetector.onTouchEvent(event); if (!mScaleDetector.isInProgress()) { mGestureDetector.onTouchEvent(event); } int action = event.getAction(); switch (action & MotionEvent.ACTION_MASK) { case MotionEvent.ACTION_UP: return onUp(event); } return true; } @Override protected void onZoomAnimationCompleted(float scale) { if (LOG_ENABLED) { Log.d(LOG_TAG, "onZoomAnimationCompleted. scale: " + scale + ", minZoom: " + getMinScale()); } if (scale < getMinScale()) { zoomTo(getMinScale(), 50); } } protected float onDoubleTapPost(float scale, float maxZoom) { if (mDoubleTapDirection == 1) { if ((scale + (mScaleFactor * 2)) <= maxZoom) { return scale + mScaleFactor; } else { mDoubleTapDirection = -1; return maxZoom; } } else { mDoubleTapDirection = 1; return 1f; } } public boolean onSingleTapConfirmed(MotionEvent e) { return true; } public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { if (getScale() == 1f) return false; mUserScaled = true; scrollBy(-distanceX, -distanceY); invalidate(); return true; } public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) { float diffX = e2.getX() - e1.getX(); float diffY = e2.getY() - e1.getY(); if (Math.abs(velocityX) > 800 || Math.abs(velocityY) > 800) { mUserScaled = true; scrollBy(diffX / 2, diffY / 2, 300); invalidate(); return true; } return false; } public boolean onDown(MotionEvent e) { return true; } public boolean onUp(MotionEvent e) { if (getScale() < getMinScale()) { zoomTo(getMinScale(), 50); } return true; } public boolean onSingleTapUp(MotionEvent e) { return true; } /** * Determines whether this ImageViewTouch can be scrolled. * * @param direction - positive direction value means scroll from right to * left, negative value means scroll from left to right * @return true if there is some more place to scroll, false - otherwise. */ public boolean canScroll(int direction) { RectF bitmapRect = getBitmapRect(); updateRect(bitmapRect, mScrollRect); Rect imageViewRect = new Rect(); getGlobalVisibleRect(imageViewRect); if (null == bitmapRect) { return false; } if (bitmapRect.right >= imageViewRect.right) { if (direction < 0) { return Math.abs(bitmapRect.right - imageViewRect.right) > SCROLL_DELTA_THRESHOLD; } } double bitmapScrollRectDelta = Math.abs(bitmapRect.left - mScrollRect.left); return bitmapScrollRectDelta > SCROLL_DELTA_THRESHOLD; } public class GestureListener extends GestureDetector.SimpleOnGestureListener { @Override public boolean onSingleTapConfirmed(MotionEvent e) { if (null != mSingleTapListener) { mSingleTapListener.onSingleTapConfirmed(); } return ImageViewTouch.this.onSingleTapConfirmed(e); } @Override public boolean onDoubleTap(MotionEvent e) { Log.i(LOG_TAG, "onDoubleTap. double tap enabled? " + mDoubleTapEnabled); if (mDoubleTapEnabled) { mUserScaled = true; float scale = getScale(); float targetScale = scale; targetScale = onDoubleTapPost(scale, getMaxScale()); targetScale = Math.min(getMaxScale(), Math.max(targetScale, getMinScale())); zoomTo(targetScale, e.getX(), e.getY(), DEFAULT_ANIMATION_DURATION); invalidate(); } if (null != mDoubleTapListener) { mDoubleTapListener.onDoubleTap(); } return super.onDoubleTap(e); } @Override public void onLongPress(MotionEvent e) { if (isLongClickable()) { if (!mScaleDetector.isInProgress()) { setPressed(true); performLongClick(); } } } @Override public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) { if (!mScrollEnabled) return false; return !(e1 == null || e2 == null) && !(e1.getPointerCount() > 1 || e2.getPointerCount() > 1) && !mScaleDetector.isInProgress() && ImageViewTouch.this.onScroll(e1, e2, distanceX, distanceY); } @Override public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) { if (!mScrollEnabled) return false; return !(e1.getPointerCount() > 1 || e2.getPointerCount() > 1) && !mScaleDetector.isInProgress() && getScale() != 1f && ImageViewTouch.this.onFling(e1, e2, velocityX, velocityY); } @Override public boolean onSingleTapUp(MotionEvent e) { return ImageViewTouch.this.onSingleTapUp(e); } @Override public boolean onDown(MotionEvent e) { return ImageViewTouch.this.onDown(e); } } public class ScaleListener extends ScaleGestureDetector.SimpleOnScaleGestureListener { protected boolean mScaled = false; @Override public boolean onScale(ScaleGestureDetector detector) { float span = detector.getCurrentSpan() - detector.getPreviousSpan(); float targetScale = getScale() * detector.getScaleFactor(); if (mScaleEnabled) { if (mScaled && span != 0) { mUserScaled = true; targetScale = Math.min(getMaxScale(), Math.max(targetScale, getMinScale() - 0.1f)); zoomTo(targetScale, detector.getFocusX(), detector.getFocusY()); mDoubleTapDirection = 1; invalidate(); return true; } // This is to prevent a glitch the first time // image is scaled. if (!mScaled) mScaled = true; } return true; } } public interface OnImageViewTouchDoubleTapListener { void onDoubleTap(); } public interface OnImageViewTouchSingleTapListener { void onSingleTapConfirmed(); } }