SwipeRefreshLayout自定义上拉加载更多问题解决和完善

时间:2022-12-22 22:19:21

          在看这个之前,需要先了解SwipeRefreshLayout的功能用法:https://github.com/PingerWan/SwipeRefreshDemo

        修改:

                 1.处理数据空的情况    isListNull
                 2.修改滑动过程中,mUpY的位置和值



附上修改后的代码:

  

import android.content.Context;
import android.support.v4.widget.SwipeRefreshLayout;
import android.util.AttributeSet;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.widget.AbsListView;
import android.widget.ListView;

/**
* 自定义View继承SwipeRefreshLayout,添加上拉加载更多的布局属性
*/

public class SwipeRefreshView extends SwipeRefreshLayout {

private final int mScaledTouchSlop;
private final View mFooterView;
private ListView mListView;
private OnLoadListener mOnLoadListener;
private boolean isListNull;

/**
* 正在加载状态
*/
private boolean isLoading;

public SwipeRefreshView(Context context, AttributeSet attrs) {
super(context, attrs);
// 填充底部加载布局
mFooterView = View.inflate(context, R.layout.view_footer, null);

// 表示控件移动的最小距离,手移动的距离大于这个距离才能拖动控件
mScaledTouchSlop = ViewConfiguration.get(context).getScaledTouchSlop();
//Log.e("akui", "====" + mScaledTouchSlop);
}

@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) {
// 判断容器有多少个孩子
if (getChildCount() > 0) {
// 判断第一个孩子是不是ListView
if (getChildAt(0) instanceof ListView) {
// 创建ListView对象
mListView = (ListView) getChildAt(0);
//ListView是否有数据
isListNull = mListView.getChildCount() <= 0;
// 设置ListView的滑动监听
setListViewOnScroll();
}
}
}
}

/**
* 在分发事件的时候处理子控件的触摸事件
*
* @param ev
* @return
*/
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:
// 移动的终点
mUpY = ev.getY();
// 移动过程中判断时候能下拉加载更多
if (canLoadMore()) {
// 加载数据
loadData();
}
break;
case MotionEvent.ACTION_UP:
mDownY = 0;
mUpY = 0;
break;
}
return super.dispatchTouchEvent(ev);
}

/**
* 判断是否满足加载更多条件
*
* @return
*/
private boolean canLoadMore() {
// 1. 是上拉状态
boolean condition1 = (mDownY - mUpY) >= mScaledTouchSlop;
if (condition1) {
//Log.e("akui", "是上拉状态");
}

// 2. 当前页面可见的item是最后一个条目
boolean condition2 = false;
if (mListView != null && mListView.getAdapter() != null) {
condition2 = mListView.getLastVisiblePosition() == (mListView.getAdapter().getCount() - 1);
}

if (condition2) {
//Log.e("akui", "是最后一个条目");
}
// 3. 正在加载状态
boolean condition3 = !isLoading;
if (condition3) {
//Log.e("akui", "不是正在加载状态");
}
// 4. listView 为空
boolean condition4 = !isListNull;
if (condition4) {
//Log.e("akui", "listView 为空");
}
return condition1 && condition2 && condition3 && condition4;
}

/**
* 处理加载数据的逻辑
*/
private void loadData() {
//Log.e("akui", "加载数据...");
if (mOnLoadListener != null) {
// 设置加载状态,让布局显示出来
setLoading(true);
mOnLoadListener.onLoad();
}

}

/**
* 设置加载状态,是否加载传入boolean值进行判断
*
* @param loading
*/
public void setLoading(boolean loading) {
// 修改当前的状态
isLoading = loading;
if (isLoading) {
// 显示布局
mListView.addFooterView(mFooterView);
} else {
// 隐藏布局
if (mListView != null)
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) {

}
});
}

/**
* 上拉加载的接口回调
*/
public interface OnLoadListener {
void onLoad();
}

public void setOnLoadListener(OnLoadListener listener) {
this.mOnLoadListener = listener;
}
}