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

AndroidGestureDetector手势滑动使用实例讲解

这篇文章主要为大家详细介绍了AndroidGestureDetector手势滑动使用实例,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

Gesture在 ViewGroup中使用

GestureDetector类可以让我们快速的处理手势事件,如点击,滑动等。

使用GestureDetector分三步:

1. 定义GestureDetector类
2. 初始化手势类,同时设置手势监听
3. 将touch事件交给gesture处理

先来了解一下如何使用,后面会有示例:

package com.example.y2222.myview;

import android.content.Context;
import android.util.AttributeSet;
import android.util.Log;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.widget.LinearLayout;

/**
 * Created by raise.yang on 2016/06/29.
 */
public class GestureDemoView extends LinearLayout {
 //1,定义GestureDetector类
 private GestureDetector m_gestureDetector;

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

 public GestureDemoView(Context context, AttributeSet attrs, int defStyleAttr) {
 super(context, attrs, defStyleAttr);
 //设置为可点击
 setClickable(true);
 //2,初始化手势类,同时设置手势监听
 m_gestureDetector = new GestureDetector(context, onGestureListener);
 //双击监听-一般很少用到
 m_gestureDetector.setOnDoubleTapListener(onDoubleTapListener);
 }

 @Override
 public boolean onTouchEvent(MotionEvent event) {
 //3,将touch事件交给gesture处理
 m_gestureDetector.onTouchEvent(event);
 return super.onTouchEvent(event);
 }

 //初始化手势监听对象,使用GestureDetector.OnGestureListener的实现抽象类,因为实际开发中好多方法用不上
 private final GestureDetector.OnGestureListener OnGestureListener= new GestureDetector.SimpleOnGestureListener() {
 @Override
 public boolean onSingleTapUp(MotionEvent e) {
  Log.d("GestureDemoView", "onSingleTapUp() ");
  return super.onSingleTapUp(e);
 }

 @Override
 public void onLongPress(MotionEvent e) {
  Log.d("GestureDemoView", "onLongPress() ");
  super.onLongPress(e);
 }

 @Override
 public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {
  Log.d("GestureDemoView", "onScroll() distanceX = " + distanceX);
  return super.onScroll(e1, e2, distanceX, distanceY);
 }

 @Override
 public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {
  Log.d("GestureDemoView", "onFling() velocityX = " + velocityX);
  return super.onFling(e1, e2, velocityX, velocityY);
 }

 @Override
 public void onShowPress(MotionEvent e) {
  Log.d("GestureDemoView", "onShowPress() ");
  super.onShowPress(e);
 }

 @Override
 public boolean onDown(MotionEvent e) {
  Log.d("GestureDemoView", "onDown() ");
  return super.onDown(e);
 }

 @Override
 public boolean onDoubleTap(MotionEvent e) {
  Log.d("GestureDemoView", "onDoubleTap() ");
  return super.onDoubleTap(e);
 }

 @Override
 public boolean onDoubleTapEvent(MotionEvent e) {
  Log.d("GestureDemoView", "onDoubleTapEvent() ");
  return super.onDoubleTapEvent(e);
 }

 @Override
 public boolean onSingleTapConfirmed(MotionEvent e) {
  Log.d("GestureDemoView", "onSingleTapConfirmed() ");
  return super.onSingleTapConfirmed(e);
 }

 @Override
 public boolean onContextClick(MotionEvent e) {
  Log.d("GestureDemoView", "onContextClick() ");
  return super.onContextClick(e);
 }
 };
 private final GestureDetector.OnDoubleTapListener OnDoubleTapListener= new GestureDetector.OnDoubleTapListener() {
 @Override
 public boolean onSingleTapConfirmed(MotionEvent e) {
  Log.d("GestureDemoView", "onSingleTapConfirmed() OnDoubleTapListener");
  return false;
 }

 @Override
 public boolean onDoubleTap(MotionEvent e) {
  Log.d("GestureDemoView", "onDoubleTap() OnDoubleTapListener");
  return false;
 }

 @Override
 public boolean onDoubleTapEvent(MotionEvent e) {
  Log.d("GestureDemoView", "onDoubleTapEvent() OnDoubleTapListener");
  return false;
 }
 };

}

注意:setClickable(true);一定要加,不然只会收到下例3个事件,被这个整了好长时间才找到原因.(⊙﹏⊙)b

对于单击,双击,拖动等事件调用见下图:

根据上图,每个方法大致都调用了,说明几个容易弄混的回调方法
1. onScroll()
public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY)
e1:滑动事件的起点(也就是说onDown()的时候)
e2:当前滑动位置点(手指的位置)
distanceX:上次滑动(调用onScroll)到这次滑动的X轴的距离px,不是e1点到e2点的X轴的距离
distanceY:上次滑动(调用onScroll)到这次滑动的Y轴的距离px,不是e1点到e2点的Y轴的距离
2. onFling()
public boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY)
e1:拖动动事件的起点(也就是说onDown()的时候)
e2:onFling()调用时,手指的位置
velocityX:X轴上每秒滑动像素值
velocityY:Y轴上每秒滑动像素值
注意:当拖动速率velocityX或velocityY超过ViewConfiguration.getMinimumFlingVelocity()最小拖动速率时,才会调用onFling(),也就是如果只拖动一点,或是慢慢的拖动,是不会触发该方法。
对应源码:

 if ((Math.abs(velocityY) > mMinimumFlingVelocity)
   || (Math.abs(velocityX) > mMinimumFlingVelocity)){
   handled = mListener.onFling(mCurrentDownEvent, ev, velocityX, velocityY);
  }

实践:使用GestureDetector实现左滑删除

在很多ListView中都有该效果,现在自己实现下,顺便熟悉GestureDetector的使用。
效果图:

GestureDemoView.java:

package com.example.y2222.myview;

import android.content.Context;
import android.util.AttributeSet;
import android.util.Log;
import android.view.GestureDetector;
import android.view.LayoutInflater;
import android.view.MotionEvent;
import android.widget.LinearLayout;

import com.example.y2222.myapplication.R;

/**
 * Created by raise.yang on 2016/06/29.
 */
public class GestureDemoView extends LinearLayout {
 //1,定义GestureDetector类
 private GestureDetector m_gestureDetector;

 private int m_max_scrollX;

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

 public GestureDemoView(Context context, AttributeSet attrs, int defStyleAttr) {
 super(context, attrs, defStyleAttr);
 //设置为可点击
 setClickable(true);
 //2,初始化手势类,同时设置手势监听
 m_gestureDetector = new GestureDetector(context, onGestureListener);

 LayoutInflater.from(context).inflate(R.layout.view_gesture, this);
 }

 @Override
 public boolean onTouchEvent(MotionEvent event) {
 //3,将touch事件交给gesture处理
 m_gestureDetector.onTouchEvent(event);
 if (event.getAction() == MotionEvent.ACTION_UP) {
  // GestureDetector没有处理up事件的方法,只能在这里处理了。
  int scrollX = getScrollX();
  if (scrollX > m_max_scrollX / 2) {
  show_right_view();
  } else {
  hide_right_view();
  }
 }
 return super.onTouchEvent(event);
 }

 @Override
 protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
 super.onMeasure(widthMeasureSpec, heightMeasureSpec);
 int childCount = getChildCount();
 for (int i = 0; i 
<&#63;xml version="1.0" encoding="utf-8"&#63;>


 

 

 

xml文件中根标签使用,可减少一层view树嵌套,并且使用getChildCount()能得到我们想要的子view个数。

关于标签的使用,详见郭神的blog:http://blog.csdn.net/guolin_blog/article/details/43376527

实现也很简单,在scroll和fling的时候,得到滑动距离或滑动速度,再调用view自己的scrollTo()或scrollBy()滑动内部元素即可。
从效果图中,当滑动到一半松手时,立即滑动到最左边,完全没有动画,这样的体验很差,所以还需优化。关于滑动时增加动画效果,可以使用Scroller类完成,准备下期补上。

Gesture在 View中使用

和在viewgroup中一样,在view中,同样是经过三步来实现:
1. 定义GestureDetector类
2. 初始化手势类,同时设置手势监听
3. 将touch事件交给gesture处理
举个荔枝:
做了一个小球跟随手指移动的效果,先绘制小球,当手指放在小球上滑动时,会调用onScroll(),在这个方法中,修改圆心的位置进行重绘,这样小球就能移动了。
这里有2个难点:
1. 如何判断手指落在了小球上;
2. 滑动到边界时,不能超过边界;

效果图:

GestureView.java代码:

package com.example.y2222.myview;

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.util.AttributeSet;
import android.view.GestureDetector;
import android.view.MotionEvent;
import android.view.View;

/**
 * Created by raise.yang on 2016/07/05.
 */
public class GestureView extends View {

 private GestureDetector m_gestureDetector;
 private Paint m_paint;
 //小球的中心点
 private float centerX;
 private float centerY;
 //小球的半径
 private int radius;
 //是否touch在小球上
 private boolean touch_bool;

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

 public GestureView(Context context, AttributeSet attrs, int defStyleAttr) {
 super(context, attrs, defStyleAttr);
 // 初始画笔
 m_paint = new Paint(Paint.ANTI_ALIAS_FLAG);
 m_paint.setColor(getResources().getColor(android.R.color.holo_blue_light));
 //设置为可点击
 setClickable(true);
 //2,初始化手势类,同时设置手势监听
 m_gestureDetector = new GestureDetector(context, onGestureListener);
 radius = 50;
 }

 @Override
 public boolean onTouchEvent(MotionEvent event) {
 //3,将touch事件交给gesture处理
 m_gestureDetector.onTouchEvent(event);
 if (event.getAction() == MotionEvent.ACTION_DOWN) {
  //判断手指落在了小球上
  if (getDistanceByPoint((int) centerX, (int) centerY, (int) event.getX(), (int) event.getY())  0) {
  centerX = w / 2;
 }
 if (h > 0) {
  centerY = h / 2;
 }
 }

 @Override
 protected void onDraw(Canvas canvas) {
 canvas.drawCircle(centerX, centerY, radius, m_paint);
 }

 GestureDetector.OnGestureListener OnGestureListener= new GestureDetector.SimpleOnGestureListener() {
 @Override
 public boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY) {
  if (touch_bool) {
  centerY -= distanceY;
  centerX -= distanceX;
  //处理边界问题
  if (centerX  getWidth() - radius) {
   centerX = getWidth() - radius;
  }
  if (centerY  getHeight() - radius) {
   centerY = getHeight() - radius;
  }
  //修改圆心后,通知重绘
  postInvalidate();
  }
  return true;
 }
 };

 /**
 * 计算两点间的距离
 */
 private int getDistanceByPoint(int x1, int y1, int x2, int y2) {
 double temp = Math.abs((x2 - x1) * (x2 - x1) - (y2 - y1) * (y2 - y1));
 return (int) Math.sqrt(temp);
 }

}

在处理问题1时,我设置了一个boolean值,在用户触摸的时候去判断,当前点和圆心点的距离是否小于半径,若小于,说明在圆内。这样在滑动的时候,就去判断一下,是否需要滑动小球。
控制边界,其实就是控制圆心点的坐标,只要保证落在(radius,radius),(getWidth()-radius,getHeight()-radius)两点矩形中即可。

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


推荐阅读
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • 本文详细介绍了如何使用PHP检测AJAX请求,通过分析预定义服务器变量来判断请求是否来自XMLHttpRequest。此方法简单实用,适用于各种Web开发场景。 ... [详细]
  • 本文探讨了 RESTful API 和传统接口之间的关键差异,解释了为什么 RESTful API 在设计和实现上具有独特的优势。 ... [详细]
  • 本文详细介绍了如何使用Spring Boot进行高效开发,涵盖了配置、实例化容器以及核心注解的使用方法。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • Python自动化处理:从Word文档提取内容并生成带水印的PDF
    本文介绍如何利用Python实现从特定网站下载Word文档,去除水印并添加自定义水印,最终将文档转换为PDF格式。该方法适用于批量处理和自动化需求。 ... [详细]
  • 在当前众多持久层框架中,MyBatis(前身为iBatis)凭借其轻量级、易用性和对SQL的直接支持,成为许多开发者的首选。本文将详细探讨MyBatis的核心概念、设计理念及其优势。 ... [详细]
  • 将Web服务部署到Tomcat
    本文介绍了如何在JDeveloper 12c中创建一个Java项目,并将其打包为Web服务,然后部署到Tomcat服务器。内容涵盖从项目创建、编写Web服务代码、配置相关XML文件到最终的本地部署和验证。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • 本文介绍如何在 Unity 的 XML 配置文件中,将参数传递给自定义生命周期管理器的构造函数。我们将详细探讨 CustomLifetimeManager 类的实现及其配置方法。 ... [详细]
  • 本文详细介绍了 Java 中 org.apache.xmlbeans.SchemaType 类的 getBaseEnumType() 方法,提供了多个代码示例,并解释了其在不同场景下的使用方法。 ... [详细]
  • 本文详细介绍了如何解决MyBatis中常见的BindingException错误,提供了多种排查和修复方法,确保Mapper接口与XML文件的正确配置。 ... [详细]
  • 基于KVM的SRIOV直通配置及性能测试
    SRIOV介绍、VF直通配置,以及包转发率性能测试小慢哥的原创文章,欢迎转载目录?1.SRIOV介绍?2.环境说明?3.开启SRIOV?4.生成VF?5.VF ... [详细]
author-avatar
JieGe木
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有