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