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 { @Override protected 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(){ @Override public void onTop() { Toast.makeText(getApplicationContext(), "Top", Toast.LENGTH_SHORT).show(); } @Override public 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() { @Override public 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() { @Override public 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