Android中实现图片平移、缩放、旋转同步进行

jopen 8年前

转载请注明转自:noyet12的博客
博客原址:http://blog.csdn.net/u012975705/article/details/49797911
源码下载地址:
(github)https://github.com/noyo/RotateZoomImageView
(csdn)http://download.csdn.net/detail/u012975705/9263323
EventBus2.4.jar下载

前言

之前因为项目需求,其中使用到了图片的单击显示取消,图片平移缩放功能,昨天突然想再加上图片的旋转功能,在网上看了很多相关的例子,可是没看到能同时实现我想要的功能的。
需求:
(1)图片平移、缩放、旋转等一系列操作后,图片需要自动居中显示。
(2)图片旋转后选自动水平显示或者垂直显示
(3)图片在放大缩小的同时都能旋转

Demo实现部分效果截图

这里写图片描述

Demo主要代码

MainActivity.java    package com.practice.noyet.rotatezoomimageview;    import android.app.Activity;  import android.graphics.Bitmap;  import android.graphics.BitmapFactory;  import android.graphics.Matrix;  import android.graphics.PointF;  import android.graphics.RectF;  import android.os.AsyncTask;  import android.os.Bundle;  import android.util.DisplayMetrics;  import android.view.MotionEvent;  import android.view.View;  import android.widget.ImageView;    import com.ypy.eventbus.EventBus;    import java.io.File;  import java.math.BigDecimal;    /**   * package: com.practice.noyet.rotatezoomimageview   * Created by noyet on 2015/11/11.   */  public class MainActivity extends Activity implements View.OnTouchListener {        private ImageView mImageView;        private PointF point0 = new PointF();      private PointF pointM = new PointF();        private final int NONE = 0;      /**       * 平移       */      private final int DRAG = 1;      /**       * 旋转、缩放       */      private final int ZOOM = 2;      /**       * 设定事件模式       */      private int mode = NONE;      /**       * 图片缩放矩阵       */      private Matrix matrix = new Matrix();      /**       * 保存触摸前的图片缩放矩阵       */      private Matrix savedMatrix = new Matrix();      /**       * 保存触点移动过程中的图片缩放矩阵       */      private Matrix matrix1 = new Matrix();      /**       * 屏幕高度       */      private int displayHeight;      /**       * 屏幕宽度       */      private int displayWidth;      /**       * 最小缩放比例       */      protected float minScale = 1f;      /**       * 最大缩放比例       */      protected float maxScale = 3f;      /**       * 当前缩放比例       */      protected float currentScale = 1f;      /**       * 多点触摸2个触摸点间的起始距离       */      private float oldDist;      /**       * 多点触摸时图片的起始角度       */      private float oldRotation = 0;      /**       * 旋转角度       */      protected float rotation = 0;      /**       * 图片初始宽度       */      private int imgWidth;      /**       * 图片初始高度       */      private int imgHeight;      /**       * 设置单点触摸退出图片显示时,单点触摸的灵敏度(可针对不同手机单独设置)       */      protected final int MOVE_MAX = 2;      /**       * 单点触摸时手指触发的‘MotionEvent.ACTION_MOVE’次数       */      private int fingerNumMove = 0;        private Bitmap bm;      /**       * 保存matrix缩放比例       */      private float matrixScale= 1;      /*private String imagePath;*/        /**       * 显示被存入缓存中的网络图片       *       * @param event 观察者事件       */      public void onEventMainThread(CustomEventBus event) {          if (event == null) {              return;          }          if (event.type == CustomEventBus.EventType.SHOW_PICTURE) {              bm = (Bitmap) event.obj;              showImage();          }      }        @Override      protected void onCreate(Bundle savedInstanceState) {          super.onCreate(savedInstanceState);          setContentView(R.layout.activity_main);            initData();      }        public void initData() {          // TODO Auto-generated method stub          bm = BitmapFactory.decodeResource(getResources(), R.drawable.alipay);          DisplayMetrics dm = getResources().getDisplayMetrics();          displayWidth = dm.widthPixels;          displayHeight = dm.heightPixels;          mImageView = (ImageView) findViewById(R.id.image_view);          mImageView.setOnTouchListener(this);          showImage();            //显示网络图片时使用          /*File file = MainApplication.getInstance().getImageCache()                  .getDiskCache().get(图片路径);          if (!file.exists()) {              Toast.makeText(this, "图片路径错误", Toast.LENGTH_SHORT).show();          } else {              new MyTask().execute(file);          }*/      }        @Override      public boolean onTouch(View view, MotionEvent event) {          ImageView imageView = (ImageView) view;          switch (event.getAction() & MotionEvent.ACTION_MASK) {              case MotionEvent.ACTION_DOWN:                  savedMatrix.set(matrix);                  point0.set(event.getX(), event.getY());                  mode = DRAG;                  System.out.println("MotionEvent--ACTION_DOWN");                  break;              case MotionEvent.ACTION_POINTER_DOWN:                  oldDist = spacing(event);                  oldRotation = rotation(event);                  savedMatrix.set(matrix);                  setMidPoint(pointM, event);                  mode = ZOOM;                  System.out.println("MotionEvent--ACTION_POINTER_DOWN---" + oldRotation);                  break;              case MotionEvent.ACTION_UP:                  if (mode == DRAG && (fingerNumMove <= MOVE_MAX)) {                      MainActivity.this.finish();                  }                  checkView();                  centerAndRotate();                  imageView.setImageMatrix(matrix);                  System.out.println("MotionEvent--ACTION_UP");                  fingerNumMove = 0;                  break;              case MotionEvent.ACTION_POINTER_UP:                  mode = NONE;                  System.out.println("MotionEvent--ACTION_POINTER_UP");                  break;              case MotionEvent.ACTION_MOVE:                  operateMove(event);                  imageView.setImageMatrix(matrix1);                  fingerNumMove++;                  System.out.println("MotionEvent--ACTION_MOVE");                  break;            }          return true;      }        @Override      protected void onDestroy() {          // TODO Auto-generated method stub          super.onDestroy();          if (bm != null && !bm.isRecycled()) {              bm.recycle(); // 回收图片所占的内存              System.gc(); // 提醒系统及时回收          }      }        /**       * 显示图片       */      private void showImage() {          imgWidth = bm.getWidth();          imgHeight = bm.getHeight();          mImageView.setImageBitmap(bm);          matrix.setScale(1, 1);          centerAndRotate();          mImageView.setImageMatrix(matrix);      }        /**       * 触点移动时的操作       *       * @param event 触摸事件       */      private void operateMove(MotionEvent event) {          matrix1.set(savedMatrix);          switch (mode) {              case DRAG:                  matrix1.postTranslate(event.getX() - point0.x, event.getY() - point0.y);                  break;              case ZOOM:                  rotation = rotation(event) - oldRotation;                  float newDist = spacing(event);                  float scale = newDist / oldDist;                  currentScale = (scale > 3.5f) ? 3.5f : scale;                  System.out.println("缩放倍数---" + currentScale);                  System.out.println("旋转角度---" + rotation);                  /** 縮放 */                  matrix1.postScale(currentScale, currentScale, pointM.x, pointM.y);                  /** 旋轉 */                  matrix1.postRotate(rotation, displayWidth / 2, displayHeight / 2);                  break;          }      }        /**       * 两个触点的距离       *       * @param event 触摸事件       * @return float       */      private float spacing(MotionEvent event) {          float x = event.getX(0) - event.getX(1);          float y = event.getY(0) - event.getY(1);          return (float) Math.sqrt(x * x + y * y);      }        /**       * 获取旋转角度       */      private float rotation(MotionEvent event) {          double delta_x = (event.getX(0) - event.getX(1));          double delta_y = (event.getY(0) - event.getY(1));          double radians = Math.atan2(delta_y, delta_x);          return (float) Math.toDegrees(radians);      }        /**       * 两个触点的中间坐标       *       * @param pointM 中间坐标       * @param event  触摸事件       */      private void setMidPoint(PointF pointM, MotionEvent event) {          float x = event.getX(0) + event.getY(1);          float y = event.getY(0) + event.getY(1);          pointM.set(x / 2, y / 2);      }        /**       * 检查约束条件(缩放倍数)       */      private void checkView() {          if (currentScale > 1) {              if (currentScale * matrixScale > maxScale) {                  matrix.postScale(maxScale / matrixScale, maxScale / matrixScale, pointM.x, pointM.y);                  matrixScale = maxScale;              } else {                  matrix.postScale(currentScale, currentScale, pointM.x, pointM.y);                  matrixScale *= currentScale;              }          } else {              if (currentScale * matrixScale < minScale) {                  matrix.postScale(minScale / matrixScale, minScale / matrixScale, pointM.x, pointM.y);                  matrixScale = minScale;              } else {                  matrix.postScale(currentScale, currentScale, pointM.x, pointM.y);                  matrixScale *= currentScale;              }          }      }        /**       * 图片居中显示、判断旋转角度 小于(90 * x + 45)度图片旋转(90 * x)度 大于则旋转(90 * (x+1))       */      private void centerAndRotate() {          RectF rect = new RectF(0, 0, imgWidth, imgHeight);          matrix.mapRect(rect);          float width = rect.width();          float height = rect.height();          float dx = 0;          float dy = 0;            if (width < displayWidth) {              dx = displayWidth / 2 - width / 2 - rect.left;          } else if (rect.left > 0) {              dx = -rect.left;          } else if (rect.right < displayWidth) {              dx = displayWidth - rect.right;          }            if (height < displayHeight) {              dy = displayHeight / 2 - height / 2 - rect.top;          } else if (rect.top > 0) {              dy = -rect.top;          } else if (rect.bottom < displayHeight) {              dy = displayHeight - rect.bottom;          }            matrix.postTranslate(dx, dy);            if (rotation != 0) {              int rotationNum = (int) (rotation / 90);              float rotationAvai = new BigDecimal(rotation % 90).setScale(1, BigDecimal.ROUND_HALF_UP).floatValue();              float realRotation = 0;              if (rotation > 0) {                  realRotation = rotationAvai > 45 ? (rotationNum + 1) * 90 : rotationNum * 90;              } else if (rotation < 0) {                  realRotation = rotationAvai < -45 ? (rotationNum - 1) * 90 : rotationNum * 90;              }              System.out.println("realRotation: " + realRotation);              matrix.postRotate(realRotation, displayWidth / 2, displayHeight / 2);              rotation = 0;          }      }        /**       * 显示网络图片时使用       */      private class MyTask extends AsyncTask<File, File, Bitmap> {            Bitmap bitmap;          String path;          int scale = 1;          long size;            @Override          protected Bitmap doInBackground(File... params) {              // TODO Auto-generated method stub              try {                  size = params[0].length();                  path = params[0].getAbsolutePath();                  BitmapFactory.Options options = new BitmapFactory.Options();                  options.inJustDecodeBounds = true;                  BitmapFactory.decodeFile(path, options);                  scale = calculateInSampleSize(options, displayWidth,                          displayHeight);                  options.inJustDecodeBounds = false;                  options.inSampleSize = scale;                  bitmap = BitmapFactory.decodeFile(path, options);              } catch (Exception e) {                  // TODO Auto-generated catch block                  e.printStackTrace();              }              return bitmap;          }            @Override          protected void onPostExecute(Bitmap result) {              // TODO Auto-generated method stub              EventBus.getDefault().post(                      new CustomEventBus(CustomEventBus.EventType.SHOW_PICTURE, result));          }            /**           * 获取图片缩放比例           *           * @param paramOptions Options           * @param paramInt1    宽           * @param paramInt2    高           * @return int           */          private int calculateInSampleSize(BitmapFactory.Options paramOptions,                                            int paramInt1, int paramInt2) {              int i = paramOptions.outHeight;              int j = paramOptions.outWidth;              int k = 1;              if ((i > paramInt2) || (j > paramInt1)) {                  int m = Math.round(i / paramInt2);                  int n = Math.round(j / paramInt1);                  k = m < n ? n : m;              }              return k;          }      }  }

CustomEventBus.java  package com.practice.noyet.rotatezoomimageview;    /**   * package: com.practice.noyet.rotatezoomimageview   * Created by noyet on 2015/11/11.   */  public class CustomEventBus {        public EventType type;      public Object obj;        public CustomEventBus(EventType type, Object obj) {          this.type = type;          this.obj = obj;      }        enum EventType {          SHOW_PICTURE      }  }