Android不用OnScrollListener采用GestureDetector结合OnTouchListener实现ListView下拉/上拉刷新

通常Android的ListView的下拉/上拉刷新实现,使用OnScrollListener比较简单,比如如果要实现下拉见顶刷新,思路是在OnScrollListener判断当前ListView的滚动状态,如果滚动停止,则将此时ListView可见区域内的第一个item的firstVisibleItem值取出来,最简单的情况(当然这种情况不完善,只是拿来说明原理和思路)就是判断firstVisibleItem是否等于0,如果等于0,则认为下拉见顶,触发写好的下拉加载代码块,上拉刷新原理相类似。
但是这种依靠OnScrollListener处理下拉/上拉事件有两个无法解决的问题:
(1)当前的ListView如果是一个空的ListView。
(2)当前的ListView非空,但其子item数量较小,以至于未能铺满整个ListView。
(3)手指的方向:是下拉还是上拉?
上述三种情况如果使用OnScrollListener则无能为力或者非常棘手解决该问题。所以在我写的附录文章3引入了OnTouchListener监听事件,然后用GestureDetector检测用户手指的方向,但仍然没有完全解决问题(1)(2),因为如果当前的ListView为空,为空就没有item,没有item就没有滚动事件,或者没有铺满超越整个屏幕,即ListView不须滚动则也就无法触发OnScrollListener进行后续的下拉/上拉处理逻辑代码块。
本文则完全不用Android ListView的OnScrollListener,仅仅依靠GestureDetector和OnTouchListener实现ListView下拉/上拉刷新。这么做的好处就是不管当前ListView的子item是否为空或者是否完全铺满或者超越整个ListView,都能正常运作。下面就是我写的支持下拉/上拉刷新事件的ListView。使用时候,和流行下拉刷新ListView一样,只需setOnPullToRefreshListener(),然后分别在onTop()和onBottom里面进行下拉见顶业务逻辑或者上拉见底逻辑即可,例如测试代码:

package zhangphil.listview;import android.app.Activity;import android.os.Bundle;import android.widget.ArrayAdapter;import android.widget.Toast;import zhangphil.listview.ZhangPhilPullToRefreshListView.OnPullToRefreshListener;public class MainActivity extends Activity {@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);String[] data = new String[20];for (int i = 0; i < data.length; i++) {data[i] = i + "";}ZhangPhilPullToRefreshListView listView = (ZhangPhilPullToRefreshListView) findViewById(R.id.listView);ArrayAdapter adapter = new ArrayAdapter(this,android.R.layout.simple_list_item_1, data);listView.setAdapter(adapter);listView.setOnPullToRefreshListener(new OnPullToRefreshListener(){@Overridepublic void onTop() {Toast.makeText(getApplicationContext(), "Top", Toast.LENGTH_SHORT).show();}@Overridepublic void onBottom() {Toast.makeText(getApplicationContext(), "Bottom", Toast.LENGTH_SHORT).show();}});}}


核心的ZhangPhilPullToRefreshListView:

package zhangphil.listview;import android.content.Context;import android.util.AttributeSet;import android.view.GestureDetector;import android.view.MotionEvent;import android.view.View;import android.widget.ListView;public class ZhangPhilPullToRefreshListView extends ListView {private Context context;private ListView listView;private OnPullToRefreshListener mOnPullToRefreshListener = null;public void setOnPullToRefreshListener(OnPullToRefreshListener l) {mOnPullToRefreshListener = l;final GestureDetector mGestureDetector = new GestureDetector(context,new GestureDetector.SimpleOnGestureListener() {@Overridepublic boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY) {float y1 = e1.getY();float y2 = e2.getY();int fp = listView.getFirstVisiblePosition();int lp = listView.getLastVisiblePosition();// 下拉boolean flag1 = (y2 - y1) > 0;// 如果当前ListView没有任何数据是一个空的ListView,但用户仍然下拉,那么直接触发下拉刷新事件if (flag1 && listView.getCount() == 0) {mOnPullToRefreshListener.onTop();return super.onFling(e1, e2, velocityX, velocityY);}if (flag1 && (fp == 0)) {if (isTop()) {mOnPullToRefreshListener.onTop();}}// 上拉boolean flag2 = (y2 - y1) < 0;// 如果当前ListView没有任何数据是一个空的ListView,但用户仍然上拉,那么直接触发上拉刷新事件if (flag2 && listView.getCount() == 0) {mOnPullToRefreshListener.onBottom();return super.onFling(e1, e2, velocityX, velocityY);}if (flag2 && (lp == (listView.getCount() - 1))) {if (isBottom()) {mOnPullToRefreshListener.onBottom();}}return super.onFling(e1, e2, velocityX, velocityY);}});this.setOnTouchListener(new View.OnTouchListener() {@Overridepublic boolean onTouch(View v, MotionEvent event) {return mGestureDetector.onTouchEvent(event);}});}public interface OnPullToRefreshListener {public void onTop();public void onBottom();};private boolean isTop() {int cnt = this.getCount();if (cnt > 0) {View view = this.getChildAt(0);if (view.getTop() == 0) {return true;}}return false;}// 判断是否底部的最后一个元素是否完全显示在屏幕上有一定的技巧// 最后一个元素getBottom()的值与ListView的getBottom()比较只有三种情况:大于,等于,小于。// 只有当最后一个元素滚到到ListView的底部可见视野以外时候,view.getBottom()才大于ListView的getBottom()// 剩余的情况均为等于或者小于。等于则说明刚好贴合在底部,小于则说明当前ListView的item数量少没有完全铺满屏幕private boolean isBottom() {int cnt = this.getCount();if (cnt > 0) {int fp = this.getFirstVisiblePosition();int lp = this.getLastVisiblePosition();View v = this.getChildAt(lp - fp);if (v.getBottom() <= this.getBottom()) {return true;}}return false;}public ZhangPhilPullToRefreshListView(Context context, AttributeSet attrs) {super(context, attrs);this.context = context;listView = this;}}



在判断ListView第一个item(即position=0)是否完全显现在屏幕可见视野范围内比较容易,但比较麻烦的是在于判断ListView最后一个item是否完全显现在屏幕的可见视野内。其要点是:ListView最后一个item之view的getBottom()值,与ListView的getBottom()值之间的数量关系,只有三种情况:
A,view的bottom值等于ListView的bottom值。那么此时刚好两者完全贴合在一起。

B,view的bottom值大于ListView的bottom值。这种情况说明当前ListView的子item数量少,未能完全充满整个屏幕的可见视野,及当前屏幕可见视野内有空白。
C,view的bottom值大于ListView的bottom值。这种情况说明当前的ListView有很多item,至于一屏已经容纳不下所有item,最后一个item已经滚到可见屏幕的下方,其view的bottom逻辑y坐标值已经超出ListView的边界。
明白了ABC三种情况代表的不同意义,就只需要处理A、B这两种情况进行上拉刷新代码逻辑即可。


附录我写的相关文章:
1,《Android AbsListView坐标体系解析》链接地址:http://blog.csdn.net/zhangphil/article/details/50360011
2、《Android判断ListView滚动到最顶部第0条item完全完整可见及最底部最后一条item完全完整可见》链接地址:http://blog.csdn.net/zhangphil/article/details/50329601
3、《Android ListView下拉/上拉刷新:设计原理与实现》链接地址:http://blog.csdn.net/zhangphil/article/details/47036177

更多相关文章

  1. Android仿人人客户端(v5.7.1)——新鲜事之完整篇
  2. 打造android万能上拉下拉刷新框架——XRefreshView (二)
  3. Android(安卓)之 Project Butter 详细介绍
  4. 让 Android(安卓)支持下拉刷新(Pull Refresh)
  5. Android(安卓)UI - ListView下拉刷新的实现
  6. 使用android自带的SwipeRefreshLayout实现下拉刷新
  7. Android实现拖拽GridView到目标View
  8. Android(安卓)当前时间的获取
  9. TimePicker组件&DatePicker组件

随机推荐

  1. android三种动画详解
  2. Android:简单的webView与js交互
  3. Android全屏
  4. android手势操作滑动效果触摸屏事件处理
  5. ui布局参数设置
  6. android:scaleType 属性
  7. 【 Android 】Android 动态矢量图
  8. Android开发之——底层驱动开发(-)
  9. Android IOS WebRTC 音视频开发总结(十五)-
  10. [转]Android媒体的一些使用总结