文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Android手势滑动实现ImageView缩放图片大小

2022-06-06 09:03

关注

本文推出了两种Android手势实现ImageView缩放图片大小的方法,分享给大家供大家参考,具体内容如下

方法一:
将以下代码写到MulitPointTouchListener.java中,然后对你相应的图片进行OnTouchListener。
例如:imageView.setOnTouchListener(new MulitPointTouchListener ());
在xml中要将ImageView的缩放格式改成Matrix
例如:android:scaleType="matrix"
这样就可以实现图片的缩放了
下面是MulitPointTouchListener.java代码:


public class MulitPointTouchListener implements OnTouchListener {  
    private static final String TAG = "Touch";  
    // These matrices will be used to move and zoom image  
    Matrix matrix = new Matrix();  
    Matrix savedMatrix = new Matrix();  
    // We can be in one of these 3 states  
    static final int NONE = 0;  
    static final int DRAG = 1;  
    static final int ZOOM = 2;  
    int mode = NONE;  
    // Remember some things for zooming  
    PointF start = new PointF();  
    PointF mid = new PointF();  
    float oldDist = 1f;  
    @Override 
    public boolean onTouch(View v, MotionEvent event) {  
        ImageView view = (ImageView) v;  
        // Log.e("view_width",  
        // view.getImageMatrix()..toString()+"*"+v.getWidth());  
        // Dump touch event to log  
        dumpEvent(event);  
        // Handle touch events here...  
        switch (event.getAction() & MotionEvent.ACTION_MASK) {  
        case MotionEvent.ACTION_DOWN:  
            matrix.set(view.getImageMatrix());  
            savedMatrix.set(matrix);  
            start.set(event.getX(), event.getY());  
            //Log.d(TAG, "mode=DRAG");  
            mode = DRAG;  
            //Log.d(TAG, "mode=NONE");  
            break;  
        case MotionEvent.ACTION_POINTER_DOWN:  
            oldDist = spacing(event);  
            //Log.d(TAG, "oldDist=" + oldDist);  
            if (oldDist > 10f) {  
                savedMatrix.set(matrix);  
                midPoint(mid, event);  
                mode = ZOOM;  
                //Log.d(TAG, "mode=ZOOM");  
            }  
            break;  
        case MotionEvent.ACTION_UP:  
        case MotionEvent.ACTION_POINTER_UP:  
            mode = NONE;  
            //Log.e("view.getWidth", view.getWidth() + "");  
            //Log.e("view.getHeight", view.getHeight() + "");  
            break;  
        case MotionEvent.ACTION_MOVE:  
            if (mode == DRAG) {  
                // ...  
                matrix.set(savedMatrix);  
                matrix.postTranslate(event.getX() - start.x, event.getY()  
                        - start.y);  
            } else if (mode == ZOOM) {  
                float newDist = spacing(event);  
                //Log.d(TAG, "newDist=" + newDist);  
                if (newDist > 10f) {  
                    matrix.set(savedMatrix);  
                    float scale = newDist / oldDist;  
                    matrix.postScale(scale, scale, mid.x, mid.y);  
                }  
            }  
            break;  
        }  
        view.setImageMatrix(matrix);  
        return true; // indicate event was handled  
    }  
    private void dumpEvent(MotionEvent event) {  
        String names[] = { "DOWN", "UP", "MOVE", "CANCEL", "OUTSIDE",  
                "POINTER_DOWN", "POINTER_UP", "7?", "8?", "9?" };  
        StringBuilder sb = new StringBuilder();  
        int action = event.getAction();  
        int actionCode = action & MotionEvent.ACTION_MASK;  
        sb.append("event ACTION_").append(names[actionCode]);  
        if (actionCode == MotionEvent.ACTION_POINTER_DOWN  
                || actionCode == MotionEvent.ACTION_POINTER_UP) {  
            sb.append("(pid ").append(  
                    action >> MotionEvent.ACTION_POINTER_ID_SHIFT);  
            sb.append(")");  
        }  
        sb.append("[");  
        for (int i = 0; i < event.getPointerCount(); i++) {  
            sb.append("#").append(i);  
            sb.append("(pid ").append(event.getPointerId(i));  
            sb.append(")=").append((int) event.getX(i));  
            sb.append(",").append((int) event.getY(i));  
            if (i + 1 < event.getPointerCount())  
                sb.append(";");  
        }  
        sb.append("]");  
        //Log.d(TAG, sb.toString());  
    }  
    private float spacing(MotionEvent event) {  
        float x = event.getX(0) - event.getX(1);  
        float y = event.getY(0) - event.getY(1);  
        return FloatMath.sqrt(x * x + y * y);  
    }  
    private void midPoint(PointF point, MotionEvent event) {  
        float x = event.getX(0) + event.getX(1);  
        float y = event.getY(0) + event.getY(1);  
        point.set(x / 2, y / 2);  
    }  
} 


方法二:自定义一个ImageView,例如TouchImageView:


import android.content.Context;
import android.graphics.Matrix;
import android.graphics.PointF;
import android.graphics.drawable.Drawable;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.ScaleGestureDetector;
import android.view.View;
import android.widget.ImageView;
public class TouchImageView extends ImageView {
  Matrix matrix;
  // We can be in one of these 3 states
  static final int NONE = 0;
  static final int DRAG = 1;
  static final int ZOOM = 2;
  int mode = NONE;
  // Remember some things for zooming
  PointF last = new PointF();
  PointF start = new PointF();
  float minScale = 1f;
  float maxScale = 3f;
  float[] m;
  int viewWidth, viewHeight;
  static final int CLICK = 3;
  float saveScale = 1f;
  protected float origWidth, origHeight;
  int oldMeasuredWidth, oldMeasuredHeight;
  ScaleGestureDetector mScaleDetector;
  Context context;
  public TouchImageView(Context context) {
    super(context);
    sharedConstructing(context);
  }
  public TouchImageView(Context context, AttributeSet attrs) {
    super(context, attrs);
    sharedConstructing(context);
  }
  private void sharedConstructing(Context context) {
    super.setClickable(true);
    this.context = context;
    mScaleDetector = new ScaleGestureDetector(context, new ScaleListener());
    matrix = new Matrix();
    m = new float[9];
    setImageMatrix(matrix);
    setScaleType(ScaleType.MATRIX);
    setOnTouchListener(new OnTouchListener() {
      @Override
      public boolean onTouch(View v, MotionEvent event) {
        mScaleDetector.onTouchEvent(event);
        PointF curr = new PointF(event.getX(), event.getY());
        switch (event.getAction()) {
          case MotionEvent.ACTION_DOWN:
           last.set(curr);
            start.set(last);
            mode = DRAG;
            break;
          case MotionEvent.ACTION_MOVE:
            if (mode == DRAG) {
              float deltaX = curr.x - last.x;
              float deltaY = curr.y - last.y;
              float fixTransX = getFixDragTrans(deltaX, viewWidth, origWidth * saveScale);
              float fixTransY = getFixDragTrans(deltaY, viewHeight, origHeight * saveScale);
              matrix.postTranslate(fixTransX, fixTransY);
              fixTrans();
              last.set(curr.x, curr.y);
            }
            break;
          case MotionEvent.ACTION_UP:
            mode = NONE;
            int xDiff = (int) Math.abs(curr.x - start.x);
            int yDiff = (int) Math.abs(curr.y - start.y);
            if (xDiff < CLICK && yDiff < CLICK)
              performClick();
            break;
          case MotionEvent.ACTION_POINTER_UP:
            mode = NONE;
            break;
        }
        setImageMatrix(matrix);
        invalidate();
        return true; // indicate event was handled
      }
    });
  }
  public void setMaxZoom(float x) {
    maxScale = x;
  }
  private class ScaleListener extends ScaleGestureDetector.SimpleOnScaleGestureListener {
    @Override
    public boolean onScaleBegin(ScaleGestureDetector detector) {
      mode = ZOOM;
      return true;
    }
    @Override
    public boolean onScale(ScaleGestureDetector detector) {
      float mScaleFactor = detector.getScaleFactor();
      float origScale = saveScale;
      saveScale *= mScaleFactor;
      if (saveScale > maxScale) {
        saveScale = maxScale;
        mScaleFactor = maxScale / origScale;
      } else if (saveScale < minScale) {
        saveScale = minScale;
        mScaleFactor = minScale / origScale;
      }
      if (origWidth * saveScale <= viewWidth || origHeight * saveScale <= viewHeight)
        matrix.postScale(mScaleFactor, mScaleFactor, viewWidth / 2, viewHeight / 2);
      else
        matrix.postScale(mScaleFactor, mScaleFactor, detector.getFocusX(), detector.getFocusY());
      fixTrans();
      return true;
    }
  }
  void fixTrans() {
    matrix.getValues(m);
    float transX = m[Matrix.MTRANS_X];
    float transY = m[Matrix.MTRANS_Y];
    float fixTransX = getFixTrans(transX, viewWidth, origWidth * saveScale);
    float fixTransY = getFixTrans(transY, viewHeight, origHeight * saveScale);
    if (fixTransX != 0 || fixTransY != 0)
      matrix.postTranslate(fixTransX, fixTransY);
  }
  float getFixTrans(float trans, float viewSize, float contentSize) {
    float minTrans, maxTrans;
    if (contentSize <= viewSize) {
      minTrans = 0;
      maxTrans = viewSize - contentSize;
    } else {
      minTrans = viewSize - contentSize;
      maxTrans = 0;
    }
    if (trans < minTrans)
      return -trans + minTrans;
    if (trans > maxTrans)
      return -trans + maxTrans;
    return 0;
  }
  float getFixDragTrans(float delta, float viewSize, float contentSize) {
    if (contentSize <= viewSize) {
      return 0;
    }
    return delta;
  }
  @Override
  protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    super.onMeasure(widthMeasureSpec, heightMeasureSpec);
    viewWidth = MeasureSpec.getSize(widthMeasureSpec);
    viewHeight = MeasureSpec.getSize(heightMeasureSpec);
    //
    // Rescales image on rotation
    //
    if (oldMeasuredHeight == viewWidth && oldMeasuredHeight == viewHeight
        || viewWidth == 0 || viewHeight == 0)
      return;
    oldMeasuredHeight = viewHeight;
    oldMeasuredWidth = viewWidth;
    if (saveScale == 1) {
      //Fit to screen.
      float scale;
      Drawable drawable = getDrawable();
      if (drawable == null || drawable.getIntrinsicWidth() == 0 || drawable.getIntrinsicHeight() == 0)
        return;
      int bmWidth = drawable.getIntrinsicWidth();
      int bmHeight = drawable.getIntrinsicHeight();
      Log.d("bmSize", "bmWidth: " + bmWidth + " bmHeight : " + bmHeight);
      float scaleX = (float) viewWidth / (float) bmWidth;
      float scaleY = (float) viewHeight / (float) bmHeight;
      scale = Math.min(scaleX, scaleY);
      matrix.setScale(scale, scale);
      // Center the image
      float redundantYSpace = (float) viewHeight - (scale * (float) bmHeight);
      float redundantXSpace = (float) viewWidth - (scale * (float) bmWidth);
      redundantYSpace /= (float) 2;
      redundantXSpace /= (float) 2;
      matrix.postTranslate(redundantXSpace, redundantYSpace);
      origWidth = viewWidth - 2 * redundantXSpace;
      origHeight = viewHeight - 2 * redundantYSpace;
      setImageMatrix(matrix);
    }
    fixTrans();
  }
}

然后在我们的Activity中就可以直接使用了:


public class TouchImageViewActivity extends Activity {
  
  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.main);
    TouchImageView img = (TouchImageView) findViewById(R.id.snoop);
    img.setImageResource(R.drawable.snoopy);
    img.setMaxZoom(4f);
  }
}
您可能感兴趣的文章:Android实现Banner界面广告图片循环轮播(包括实现手动滑动循环)Android开发之使用ViewPager实现图片左右滑动切换效果Android实现图片自动轮播并且支持手势左右无限滑动Android 高仿微信朋友圈动态支持双击手势放大并滑动查看图片效果Android实现手势滑动多点触摸放大缩小图片效果解决Android快速滑动时图片一闪一闪问题Android中RecyclerView 滑动时图片加载的优化Android 使用ViewPager实现图片左右循环滑动自动播放Android 利用ViewPager实现图片可以左右循环滑动效果附代码下载Android利用ViewPager实现带小圆球的图片滑动


阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     807人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     351人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     314人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     433人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     221人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-移动开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯