热门标签 | HotTags
当前位置:  开发笔记 > Android > 正文

Android实现ImageView图片缩放和拖动

这篇文章主要为大家详细介绍了Android实现ImageView图片缩放和拖动的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

今天我们来编写一个缩放效果的ImageView ,网上有很多人都讲了这些。但有许多人都直接使用了库文件,

那么我们今天做的是直接上代码编写一个拖动和缩放的ImageView,具体看效果图

那么简单了分析一下。在手机上缩放图片和拖动要用到什么?手指对不对

那么控件上什么事件和手机有关。View.OnTouchListener 对不对。

ok,那么先新建一个Class
···
public class BaseDragZoomImageView extends ImageView implements View.OnTouchListener
···

没错,继承ImageView 并且添加View.OnTouchListener事件

然后我们看看构造函数

  public BaseDragZoomImageView(Context context, AttributeSet attrs, int defStyle)
  {
    super(context, attrs, defStyle);
    setOnTouchListener(this);
  }

  public BaseDragZoomImageView(Context context, AttributeSet attrs)
  {
    this(context, attrs, 0);
    setOnTouchListener(this);
  }

  public BaseDragZoomImageView(Context context)
  {
    this(context, null);
    setOnTouchListener(this);
  }

对,在构造函数中,我们将setOnTouchListener 设置了。
那么这个setOnTouchListener 具体怎么做。我们就先分析一下onTouch 中MotionEvent。
我们都知道手指的操作有很多,那么Andorid自然也将这种情况分了很多种case。
- MotionEvent.ACTION_DOWN 手指按下屏幕
- MotionEvent.ACTION_MOVE 手指在屏幕上移动
- MotionEvent.ACTION_UP 手指离开屏幕
- MotionEvent.ACTION_POINTER_UP 当触点离开屏幕,但是屏幕上还有触点(手指)
- MotionEvent.ACTION_POINTER_DOWN 当屏幕上已经有触点(手指),再有一个触点压下屏幕

很显然,这些看起来好像都能够用得到。

  public boolean onTouch(View v, MotionEvent event) {
    /** 通过与运算保留最后八位 MotionEvent.ACTION_MASK = 255 */
    switch (event.getAction() & MotionEvent.ACTION_MASK) {
      // 手指压下屏幕
      case MotionEvent.ACTION_DOWN:
        mode = MODE_DRAG;
        // 记录ImageView当前的移动位置
        currentMatrix.set(getImageMatrix());
        startPoint.set(event.getX(), event.getY());
        break;
      // 手指在屏幕上移动,改事件会被不断触发
      case MotionEvent.ACTION_MOVE:
        // 拖拉图片
        if (mode == MODE_DRAG) {
          float dx = event.getX() - startPoint.x; // 得到x轴的移动距离
          float dy = event.getY() - startPoint.y; // 得到x轴的移动距离
          // 在没有移动之前的位置上进行移动
          matrix.set(currentMatrix);
          matrix.postTranslate(dx, dy);
        }
        // 放大缩小图片
        else if (mode == MODE_ZOOM) {
          float endDis = distance(event);// 结束距离
          if (endDis > 10f) { // 两个手指并拢在一起的时候像素大于10
            float scale = endDis / startDis;// 得到缩放倍数
            matrix.set(currentMatrix);
            matrix.postScale(scale, scale,midPoint.x,midPoint.y);
          }
        }
        break;
      // 手指离开屏幕
      case MotionEvent.ACTION_UP:
        // 当触点离开屏幕,但是屏幕上还有触点(手指)
      case MotionEvent.ACTION_POINTER_UP:
        mode = 0;
        break;
      // 当屏幕上已经有触点(手指),再有一个触点压下屏幕
      case MotionEvent.ACTION_POINTER_DOWN:
        mode = MODE_ZOOM;
        /** 计算两个手指间的距离 */
        startDis = distance(event);
        /** 计算两个手指间的中间点 */
        if (startDis > 10f) { // 两个手指并拢在一起的时候像素大于10
          midPoint = mid(event);
          //记录当前ImageView的缩放倍数
          currentMatrix.set(getImageMatrix());
        }
        break;
    }
    setImageMatrix(matrix);
    return true;
  }

ok,收工。具体就只有这么多。我们来看下整个class

package com.jonkming.easyui.image.dragzoom.ui;

import android.content.Context;
import android.graphics.Matrix;
import android.graphics.PointF;
import android.util.AttributeSet;
import android.view.MotionEvent;
import android.view.View;
import android.widget.ImageView;
import android.util.FloatMath;
/**
 * 图片缩放和拖动类
* @Title: BaseDragZoomImageView.java
* @Package com.jonkming.easyui.image.dragzoom.ui
* @author HuangMingming
* @date 2016/11/7 17:40
* @version V1.0
*/
public class BaseDragZoomImageView extends ImageView implements View.OnTouchListener{

  /** 记录是拖拉照片模式还是放大缩小照片模式 */
  private int mode = 0;// 初始状态
  /** 拖拉照片模式 */
  private static final int MODE_DRAG = 1;
  /** 放大缩小照片模式 */
  private static final int MODE_ZOOM = 2;

  /** 用于记录开始时候的坐标位置 */
  private PointF startPoint = new PointF();
  /** 用于记录拖拉图片移动的坐标位置 */
  private Matrix matrix = new Matrix();
  /** 用于记录图片要进行拖拉时候的坐标位置 */
  private Matrix currentMatrix = new Matrix();

  /** 两个手指的开始距离 */
  private float startDis;
  /** 两个手指的中间点 */
  private PointF midPoint;

  public BaseDragZoomImageView(Context context, AttributeSet attrs, int defStyle)
  {
    super(context, attrs, defStyle);
    setOnTouchListener(this);
  }

  public BaseDragZoomImageView(Context context, AttributeSet attrs)
  {
    this(context, attrs, 0);
    setOnTouchListener(this);
  }

  public BaseDragZoomImageView(Context context)
  {
    this(context, null);
    setOnTouchListener(this);
  }
  @Override
  public boolean onTouch(View v, MotionEvent event) {
    /** 通过与运算保留最后八位 MotionEvent.ACTION_MASK = 255 */
    switch (event.getAction() & MotionEvent.ACTION_MASK) {
      // 手指压下屏幕
      case MotionEvent.ACTION_DOWN:
        mode = MODE_DRAG;
        // 记录ImageView当前的移动位置
        currentMatrix.set(getImageMatrix());
        startPoint.set(event.getX(), event.getY());
        break;
      // 手指在屏幕上移动,改事件会被不断触发
      case MotionEvent.ACTION_MOVE:
        // 拖拉图片
        if (mode == MODE_DRAG) {
          float dx = event.getX() - startPoint.x; // 得到x轴的移动距离
          float dy = event.getY() - startPoint.y; // 得到x轴的移动距离
          // 在没有移动之前的位置上进行移动
          matrix.set(currentMatrix);
          matrix.postTranslate(dx, dy);
        }
        // 放大缩小图片
        else if (mode == MODE_ZOOM) {
          float endDis = distance(event);// 结束距离
          if (endDis > 10f) { // 两个手指并拢在一起的时候像素大于10
            float scale = endDis / startDis;// 得到缩放倍数
            matrix.set(currentMatrix);
            matrix.postScale(scale, scale,midPoint.x,midPoint.y);
          }
        }
        break;
      // 手指离开屏幕
      case MotionEvent.ACTION_UP:
        // 当触点离开屏幕,但是屏幕上还有触点(手指)
      case MotionEvent.ACTION_POINTER_UP:
        mode = 0;
        break;
      // 当屏幕上已经有触点(手指),再有一个触点压下屏幕
      case MotionEvent.ACTION_POINTER_DOWN:
        mode = MODE_ZOOM;
        /** 计算两个手指间的距离 */
        startDis = distance(event);
        /** 计算两个手指间的中间点 */
        if (startDis > 10f) { // 两个手指并拢在一起的时候像素大于10
          midPoint = mid(event);
          //记录当前ImageView的缩放倍数
          currentMatrix.set(getImageMatrix());
        }
        break;
    }
    setImageMatrix(matrix);
    return true;
  }

  /** 计算两个手指间的距离 */
  private float distance(MotionEvent event) {
    float dx = event.getX(1) - event.getX(0);
    float dy = event.getY(1) - event.getY(0);
    /** 使用勾股定理返回两点之间的距离 */
    return (float) Math.sqrt(dx * dx + dy * dy);
  }

  /** 计算两个手指间的中间点 */
  private PointF mid(MotionEvent event) {
    float midX = (event.getX(1) + event.getX(0)) / 2;
    float midY = (event.getY(1) + event.getY(0)) / 2;
    return new PointF(midX, midY);
  }
}

然后看布局文件

<&#63;xml version="1.0" encoding="utf-8"&#63;>

  

就是这么简单。

代码所在位置: GitHub

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。


推荐阅读
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • Android 九宫格布局详解及实现:人人网应用示例
    本文深入探讨了人人网Android应用中独特的九宫格布局设计,解析其背后的GridView实现原理,并提供详细的代码示例。这种布局方式不仅美观大方,而且在现代Android应用中较为少见,值得开发者借鉴。 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 解决微信电脑版无法刷朋友圈问题:使用安卓远程投屏方案
    在工作期间想要浏览微信和朋友圈却不太方便?虽然微信电脑版目前不支持直接刷朋友圈,但通过远程投屏技术,可以轻松实现在电脑上操作安卓设备的功能。 ... [详细]
  • 本文详细介绍了 Java 中 org.apache.xmlbeans.SchemaType 类的 getBaseEnumType() 方法,提供了多个代码示例,并解释了其在不同场景下的使用方法。 ... [详细]
  • 基于KVM的SRIOV直通配置及性能测试
    SRIOV介绍、VF直通配置,以及包转发率性能测试小慢哥的原创文章,欢迎转载目录?1.SRIOV介绍?2.环境说明?3.开启SRIOV?4.生成VF?5.VF ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
  • 本文介绍如何使用布局文件在Android应用中排列多行TextView和Button,使其占据屏幕的特定比例,并提供示例代码以帮助理解和实现。 ... [详细]
  • golang常用库:配置文件解析库/管理工具viper使用
    golang常用库:配置文件解析库管理工具-viper使用-一、viper简介viper配置管理解析库,是由大神SteveFrancia开发,他在google领导着golang的 ... [详细]
  • 本文详细介绍了 GWT 中 PopupPanel 类的 onKeyDownPreview 方法,提供了多个代码示例及应用场景,帮助开发者更好地理解和使用该方法。 ... [详细]
  • 本文探讨了在通过 API 端点调用时,使用猫鼬(Mongoose)的 findOne 方法总是返回 null 的问题,并提供了详细的解决方案和建议。 ... [详细]
  • 探讨如何真正掌握Java EE,包括所需技能、工具和实践经验。资深软件教学总监李刚分享了对毕业生简历中常见问题的看法,并提供了详尽的标准。 ... [详细]
  • 本文探讨了在Windows Server 2008环境下配置Tomcat使用80端口时遇到的问题,包括端口被占用、多项目访问失败等,并提供详细的解决方法和配置建议。 ... [详细]
author-avatar
手机用户2502887447
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有