Android 如何 ListView 判断滚动到最顶部或者底部
2016-02-03 17:27
459 查看
如何判断 ListView 滑动到顶部和底部?网上一搜,最多的答案都是这样的:
这一段代码并不能够精确地判断 ListView 是否到达最顶部或者最底部。因为只要ListView中第一个item出现在屏幕上端,即使只出现一部分,firstVisibleItem 的值也依然为0, onScroll()回调就会发生; 同样ListView 的最后一个 item 即使只显示一部分,((firstVisibleItem + visibleItemCount) == totalItemCount) 的值也是等于totalItemCount。
那么倒底如何精确判断 ListView 是否滚动到顶部或者底部?
一、首先想到的是 ListView.onScrollChanged()
这个函数是protected,直接引用不到,但是可以继承ListView类,然后重写这个函数:
可以参看 http://stackoverflow.com/questions/24856063/listview-onscrollchanged-always-returns-0
二、只能继续改进 ListView.setOnScrollListener()
1. 改进是否滚动到顶部:增加判断第一个 item 距离 ListView 顶部的距离是否为 0
mListView.setOnScrollListener(new AbsListView.OnScrollListener() { @Override public void onScrollStateChanged(AbsListView view, int scrollState) { } @Override public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) { if (firstVisibleItem == 0) { Log.d("ListView", "##### 滚动到顶部 #####"); } else if ((firstVisibleItem + visibleItemCount) == totalItemCount) { Log.d("ListView", "##### 滚动到底部 ######"); } } });
这一段代码并不能够精确地判断 ListView 是否到达最顶部或者最底部。因为只要ListView中第一个item出现在屏幕上端,即使只出现一部分,firstVisibleItem 的值也依然为0, onScroll()回调就会发生; 同样ListView 的最后一个 item 即使只显示一部分,((firstVisibleItem + visibleItemCount) == totalItemCount) 的值也是等于totalItemCount。
那么倒底如何精确判断 ListView 是否滚动到顶部或者底部?
一、首先想到的是 ListView.onScrollChanged()
这个函数是protected,直接引用不到,但是可以继承ListView类,然后重写这个函数:
@Override protected void onScrollChanged(int l, int t, int oldl, int oldt) { super.onScrollChanged(l, t, oldl, oldt); Log.d("ListView", String.format("onScrollChanged: l = %d, t = %d, oldl = %d, oldt = %d, ScrollY = %d", l, t, oldl, oldt, this.getScrollY())); if (t == 0) { Log.d("ListView", "onScrollChanged <----滚动到顶部----->"); } }经过测试发现:ListView的 onScrollChanged() 以及 getScrollX() 和 getScrollY() 总是返回0,根本不起作用。但是对于Android ScrollView 这三个函数都是有效的,为什么ListView就不起作用呢?尼玛,绝对没有想到Android 这么坑爹!
可以参看 http://stackoverflow.com/questions/24856063/listview-onscrollchanged-always-returns-0
二、只能继续改进 ListView.setOnScrollListener()
1. 改进是否滚动到顶部:增加判断第一个 item 距离 ListView 顶部的距离是否为 0
if (firstVisibleItem == 0) { View firstVisibleItemView = mListView.getChildAt(0); if (firstVisibleItemView != null && firstVisibleItemView.getTop() == 0) { Log.d("ListView", "##### 滚动到顶部 ######"); } }2. 改进是否滚动到底部:增加判断当前屏幕中可见的最后一个 item 的底端距离 ListView 的顶端是否为 ListView 的高度:
if ((firstVisibleItem + visibleItemCount) == totalItemCount) { View lastVisibleItemView = mListView.getChildAt(mListView.getChildCount() - 1); if (lastVisibleItemView != null && lastVisibleItemView.getBottom() == mListView.getHeight()) { Log.d("ListView", "##### 滚动到底部 ######"); } }全部完整代码如下:
mListView.setOnScrollListener(new OnScrollListener() {经过测试发现:上述代码能够正确判断 ListView 是否滚动到顶部或者底部
@Override
public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
if (firstVisibleItem == 0) {
View firstVisibleItemView = mListView.getChildAt(0);
if (firstVisibleItemView != null && firstVisibleItemView.getTop() == 0) {
Log.d("ListView", "##### 滚动到顶部 #####");
}
} else if ((firstVisibleItem + visibleItemCount) == totalItemCount) { View lastVisibleItemView = mListView.getChildAt(mListView.getChildCount() - 1); if (lastVisibleItemView != null && lastVisibleItemView.getBottom() == mListView.getHeight()) { Log.d("ListView", "##### 滚动到底部 ######"); } }
}
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
//do nothing
}
});
相关文章推荐
- 自定义可拖动GridView 仿android桌面launcher
- android:shape的使用(圆角按钮和按钮颜色)
- android wear 手表和androidphone 手机之间的数据共享
- 在android中如何显示维语
- android 如何设置自定义dialog的宽度
- android之选取本地图片
- Android 打jar包流程
- android 事件分发 点击事件 ondispathTouchEvent onTouchEvent onInterceptTouchEvent
- 打包android wear 应用程序 Package with Android Studio
- android 样式开发
- android 之截屏
- 单独编译使用WebRTC的音频处理模块 - android
- Android 内存泄漏总结
- Android相机开发那些坑
- Android 异步消息处理机制 让你深入理解 Looper、Handler、Message三者关系
- Android RecycleView(三)——增加点击事件
- android实现只切换本应用的语言(不切换android 系统语言)
- Android Studio 使用小技巧 持续更新
- 安卓开发——AndroidStudio生成getter,setter,tostring,constructor等函数的方式
- Android 监听返回键、HOME键