【Android-自定义控件】SwipeRefreshDemo 下拉刷新,上拉加载
参考:https://github.com/PingerOne/SwipeRefreshDemo
谷歌官方的SwipeRefreshLayout控件,只有下拉刷新功能。
自定义的SwipeRefreshView ,继承自SwipeRefreshLayout,添加了上拉加载更多功能,添加对RecyclerView的支持。
添加“加载更多”底部布局 swiperefreshview_footer.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
android:layout_width="match_parent"
android:layout_height="match_parent">
<RelativeLayout
android:layout_width="match_parent"
android:layout_height="48dp">
<ProgressBar
android:id="@+id/load_progress"
android:layout_width="24dp"
android:layout_height="24dp"
android:layout_centerVertical="true"
android:layout_marginLeft="30dp"/>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_centerInParent="true"
android:text="正在努力加载中..."
android:textColor="@android:color/black"
android:textSize="16sp"/>
</RelativeLayout>
</RelativeLayout>
自定义view - SwipeRefreshView
/**
* 自定义View继承SwipeRefreshLayout
* 添加上拉加载更多的布局属性
* 添加对RecyclerView的支持
*/
public class SwipeRefreshView extends SwipeRefreshLayout {
private static final String TAG = SwipeRefreshView.class.getSimpleName();
private final int mScaledTouchSlop;
private final View mFooterView;
private ListView mListView;
private OnLoadMoreListener mListener;
/**
* 正在加载状态
*/
private boolean isLoading;
private RecyclerView mRecyclerView;
private int mItemCount;
public SwipeRefreshView(Context context, AttributeSet attrs) {
super(context, attrs);
// 填充底部加载布局
mFooterView = View.inflate(context, R.layout.swiperefreshview_footer, null);
// 表示控件移动的最小距离,手移动的距离大于这个距离才能拖动控件
mScaledTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
}
@Override
protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
super.onLayout(changed, left, top, right, bottom);
// 获取ListView,设置ListView的布局位置
if (mListView == null || mRecyclerView == null) {
// 判断容器有多少个孩子
if (getChildCount() > 0) {
// 判断第一个孩子是不是ListView
if (getChildAt(0) instanceof ListView) {
// 创建ListView对象
mListView = (ListView) getChildAt(0);
// 设置ListView的滑动监听
setListViewOnScroll();
} else if (getChildAt(0) instanceof RecyclerView) {
// 创建ListView对象
mRecyclerView = (RecyclerView) getChildAt(0);
// 设置RecyclerView的滑动监听
setRecyclerViewOnScroll();
}
}
}
}
/**
* 在分发事件的时候处理子控件的触摸事件
*/
private float mDownY, mUpY;
@Override
public boolean dispatchTouchEvent(MotionEvent ev) {
switch (ev.getAction()) {
case MotionEvent.ACTION_DOWN:
// 移动的起点
mDownY = ev.getY();
break;
case MotionEvent.ACTION_MOVE:
// 移动过程中判断时候能下拉加载更多
if (canLoadMore()) {
// 加载数据
loadData();
}
break;
case MotionEvent.ACTION_UP:
// 移动的终点
mUpY = getY();
break;
}
return super.dispatchTouchEvent(ev);
}
/**
* 判断是否满足加载更多条件
*/
private boolean canLoadMore() {
// 1. 是上拉状态
boolean condition1 = (mDownY - mUpY) >= mScaledTouchSlop;
if (condition1) {
Log.d(TAG, "-------> 是上拉状态");
}
// 2. 当前页面可见的item是最后一个条目,一般最后一个条目位置需要大于第一页的数据长度
boolean condition2 = false;
if (mListView != null && mListView.getAdapter() != null) {
if (mItemCount > 0) {
if (mListView.getAdapter().getCount() < mItemCount) {
// 第一页未满,禁止下拉
condition2 = false;
}else {
condition2 = mListView.getLastVisiblePosition() == (mListView.getAdapter().getCount() - 1);
}
} else {
// 未设置数据长度,则默认第一页数据不满时也可以上拉
condition2 = mListView.getLastVisiblePosition() == (mListView.getAdapter().getCount() - 1);
}
}
if (condition2) {
Log.d(TAG, "-------> 是最后一个条目");
}
// 3. 正在加载状态
boolean condition3 = !isLoading;
if (condition3) {
Log.d(TAG, "-------> 不是正在加载状态");
}
return condition1 && condition2 && condition3;
}
public void setItemCount(int itemCount) {
this.mItemCount = itemCount;
}
/**
* 处理加载数据的逻辑
*/
private void loadData() {
System.out.println("加载数据...");
if (mListener != null) {
// 设置加载状态,让布局显示出来
setLoading(true);
mListener.onLoadMore();
}
}
/**
* 设置加载状态,是否加载传入boolean值进行判断
*
* @param loading
*/
public void setLoading(boolean loading) {
// 修改当前的状态
isLoading = loading;
if (isLoading) {
// 显示布局
mListView.addFooterView(mFooterView);
} else {
// 隐藏布局
mListView.removeFooterView(mFooterView);
// 重置滑动的坐标
mDownY = 0;
mUpY = 0;
}
}
/**
* 设置ListView的滑动监听
*/
private void setListViewOnScroll() {
mListView.setOnScrollListener(new AbsListView.OnScrollListener() {
@Override
public void onScrollStateChanged(AbsListView view, int scrollState) {
// 移动过程中判断时候能下拉加载更多
if (canLoadMore()) {
// 加载数据
loadData();
}
}
@Override
public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
}
});
}
/**
* 设置RecyclerView的滑动监听
*/
private void setRecyclerViewOnScroll() {
mRecyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
// 移动过程中判断时候能下拉加载更多
if (canLoadMore()) {
// 加载数据
loadData();
}
}
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
}
});
}
/**
* 上拉加载的接口回调
*/
public interface OnLoadMoreListener {
void onLoadMore();
}
public void setOnLoadMoreListener(OnLoadMoreListener listener) {
this.mListener = listener;
}
}
应用布局
<com.example.customview.SwipeRefreshView
android:id="@+id/swipeRefreshView"
android:layout_width="match_parent"
android:layout_height="match_parent">
<ListView
android:id="@+id/listView"
android:layout_width="match_parent"
android:layout_height="match_parent">
</ListView>
</com.example.customview.SwipeRefreshView>
应用代码
/**
* 使用自定义的SwipeRefreshView
*/
public class RefreshActicity extends AppCompatActivity {
SwipeRefreshView swipeRefreshView;
ListView listView;
List<Product> list;
ProductAdapter adapter;
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_refresh);
swipeRefreshView = findViewById(R.id.swipeRefreshView);
listView = findViewById(R.id.listView);
//设置适配器
list=new ArrayList<>();
adapter = new ProductAdapter(this, list);
listView.setAdapter(adapter);
//设置下拉进度的背景颜色,默认是白色
swipeRefreshView.setProgressBackgroundColorSchemeResource(android.R.color.white);
//设置下拉进度的肢体颜色
swipeRefreshView.setColorSchemeResources(R.color.colorAccent, R.color.colorPrimary, R.color.colorPrimaryDark);
swipeRefreshView.setItemCount(20);
// 手动调用,通知系统去测量
swipeRefreshView.measure(0,0);
swipeRefreshView.setRefreshing(true);
initData();
//下拉时触发SwipeRefreshLayout的下拉动画,动画完毕后回调这个方法
swipeRefreshView.setOnRefreshListener(new SwipeRefreshLayout.OnRefreshListener() {
@Override
public void onRefresh() {
initData();
}
});
//设置下拉加载更多
swipeRefreshView.setOnLoadMoreListener(new SwipeRefreshView.OnLoadMoreListener() {
@Override
public void onLoadMore() {
loadMoreData();
}
});
}
/**
* 第一次刷新数据
*/
private void initData()
{
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
list.clear();
list.addAll(DataResource.getData());
adapter.notifyDataSetChanged();
//加载完数据,设置为不刷新状态,将下拉进度收起来
swipeRefreshView.setRefreshing(false);
Toast.makeText(RefreshActicity.this, "刷新了20条数据", Toast.LENGTH_SHORT).show();
// 加载完数据设置为不刷新状态,将下拉进度收起来
if (swipeRefreshView.isRefreshing()) {
swipeRefreshView.setRefreshing(false);
}
}
}, 2000);
}
/**
* 加载更多数据
*/
private void loadMoreData()
{
new Handler().postDelayed(new Runnable() {
@Override
public void run() {
list.clear();
list.addAll(DataResource.getMoreData());
Toast.makeText(RefreshActicity.this, "加载了" + 20 + "条数据", Toast.LENGTH_SHORT).show();
// 加载完数据设置为不加载状态,将加载进度收起来
swipeRefreshView.setLoading(false);
}
}, 2000);
}
public static class DataResource {
private static List<Product> datas = new ArrayList<>();
private static int page = 0;
//第一次加载数据
public static List<Product> getData() {
page = 0;
datas.clear();
for (int i = 0; i < 20; i++) {
Random random = new Random();
Product product = new Product("100" + (i+1), "testdata", random.nextInt(100));
datas.add(product);
}
return datas;
}
//加载更多数据
public static List<Product> getMoreData() {
page = page + 1;
for (int i = 20 * page; i < 20 * (page + 1); i++) {
Random random = new Random();
Product product = new Product("100" + (i+1), "testdata", random.nextInt(100));
datas.add(product);
}
return datas;
}
}
}
【Android-自定义控件】SwipeRefreshDemo 下拉刷新,上拉加载的更多相关文章
- Android 下拉刷新上啦加载SmartRefreshLayout + RecyclerView
在弄android刷新的时候,可算是耗费了一番功夫,最后发觉有现成的控件,并且非常好用,这里记录一下. 原文是 https://blog.csdn.net/huangxin112/article/de ...
- SwipeRefreshLayout实现下拉刷新上滑加载
1. 效果图 2.RefreshLayout.java package myapplication.com.myapplication; import android.content.Context; ...
- 移动端下拉刷新上拉加载-mescroll.js插件
最近无意间看到有这么一个上拉刷新下拉加载的插件 -- mescroll.js,个人感觉挺好用的,官网地址是:http://www.mescroll.com 然后我就看了一下文档,简单的写了一个小dem ...
- Android 下拉刷新上拉载入 多种应用场景 超级大放送(上)
转载请标明原文地址:http://blog.csdn.net/yalinfendou/article/details/47707017 关于Android下拉刷新上拉载入,网上的Demo太多太多了,这 ...
- 带你实现开发者头条APP(五)--RecyclerView下拉刷新上拉加载
title: 带你实现开发者头条APP(五)--RecyclerView下拉刷新上拉加载 tags: -RecyclerView,下拉刷新,上拉加载更多 grammar_cjkRuby: true - ...
- ListView实现Item上下拖动交换位置 并且实现下拉刷新 上拉加载更多
ListView实现Item上下拖动交换位置 并且实现下拉刷新 上拉加载更多 package com.example.ListViewDragItem; import android.app.Ac ...
- ListView下拉刷新上拉加载更多实现
这篇文章将带大家了解listview下拉刷新和上拉加载更多的实现过程,先看效果(注:图片中listview中的阴影可以加上属性android:fadingEdge="none"去掉 ...
- RecyclerView下拉刷新上拉加载(三)—对Adapter的封装
RecyclerView下拉刷新上拉加载(一) http://blog.csdn.net/baiyuliang2013/article/details/51506036 RecyclerView下拉刷 ...
- RecyclerView下拉刷新上拉加载(一)
listview下拉刷新上拉加载扩展(一) http://blog.csdn.net/baiyuliang2013/article/details/50252561 listview下拉刷新上拉加载扩 ...
- listview下拉刷新上拉加载扩展(二)-仿美团外卖
经过前几篇的listview下拉刷新上拉加载讲解,相信你对其实现机制有了一个深刻的认识了吧,那么这篇文章我们来实现一个高级的listview下拉刷新上拉加载-仿新版美团外卖的袋鼠动画: 项目结构: 是 ...
随机推荐
- java23种设计模式之七: 观察者模式
一.应用背景 观察者模式又称为发布/订阅(Publish/Subscribe)模式,我们可以理解为:只有关注信公众号关注后才能收到信息 二.优.缺点 优点: 1.可以动态的改变对象的行为 缺点 ...
- hanlp分词工具应用案例:商品图自动推荐功能的应用
本篇分享一个hanlp分词工具应用的案例,简单来说就是做一图库,让商家轻松方便的配置商品的图片,最好是可以一键完成配置的. 先看一下效果图吧: 商品单个推荐效果:匹配度高的放在最前面 这个想法很好,那 ...
- 解决redis运行期间key值过期但是内存memory依然占用过高
要解决这个问题,首先要了解redis info信息中几个数据的意义: used_memory:810575104 //数据占用了多少内存(字节) used_memory_human:773.02 ...
- CodeBlocks 配置
CodeBlocks 配置 Code::Blocks 17.12 时间:2019.6 下载网址 http://www.codeblocks.org/downloads/26 ,这里选择的是 mingw ...
- mysql8.0.13下载与安装图文教程
一.进入mysql网站:https://dev.mysql.com/downloads/mysql/ 二.进入Community选择MySQL Communtiy Server 三.将页面拉到最下面选 ...
- Photon Server初识(六) --- 客户端与服务端消息传递
前一章客户端与服务端连接成功,现在需要前后端进行数据传递. 一.前端发送消息.在项目Scripts目录中新建脚本 TestSer.cs.并挂载到相机上 二.客户端发送数据给服务端.编辑客户端代码 Te ...
- hdu 2610 2611 dfs的判重技巧
对于全排列枚举的数列的判重技巧 1:如果查找的是第一个元素 那么 从0开始到当前的位置看有没有出现过这个元素 出现过就pass 2: 如果查找的不是第一个元素 那么 从查找的子序列当前位置的前一个元素 ...
- MyEclipse eclipse console edit packageExplorer 颜色设置、个性化、常用设置
下列教程的图片是在 myeclipse2014 破解版上进行,会有些许不同,仅供参考! 1 编辑区颜色设置 主题设置 豆沙绿设置 RGB 203 233 207 2 console 3主题选择 4 去 ...
- BIOS将MBR读入0x7C00地址处(x86平台下)
BIOS将MBR读入0x7C00地址处(x86平台下) https://www.cnblogs.com/jikebiancheng/p/6193953.html http://www.ruanyife ...
- [转载]torch参数:torch.backends.cudnn.benchmark的意义
[转载]torch参数:torch.backends.cudnn.benchmark的意义 来源:https://zhuanlan.zhihu.com/p/73711222 完整版请看原文,这里只截取 ...