下拉刷新 一(SwipeRefreshLayout深入理解)
▼ SwipeRefreshLayout简述
SwipeRefreshLayout为5.0系统后,谷歌官方推出SwipeRefreshLayout控件,在android-support-v4.jar包中,有一股简洁的风格。
QQ截图20170406110842.png上面的意思就是说,当用户通过swipe gesture(刷卡手势)进行刷新的时候,都可以使用SwipeRefreshLayout进行数据刷新。同时呢?我们应该在展现该控件的activity中添加OnRefreshListener接口来控制刷新什么时候完成。同时我们发现SwipeRefreshLayout是继承在ViewGroup,不是我们常见的继承ListView,所以它的实现逻辑和ListView是没关系的,这样就解放了我们使用ListView,不需要复杂的逻辑判断处理。SwipeRefreshLayout应该是需要刷新的View的父控件,它只能有一个子View。同时它的直接子View要具有滑动功能。
<pre>
<android.support.v4.widget.SwipeRefreshLayout
android:id="@+id/main_refresh"
android:layout_width="match_parent"
android:layout_height="match_parent">
<android.support.v7.widget.RecyclerView
android:id="@+id/list_context_main"
android:layout_width="match_parent"
android:layout_height="match_parent"/>
</android.support.v4.widget.SwipeRefreshLayout>
</pre>
▼ SwipeRefreshLayout方法
● setDistanceToTriggerSync();// 设置手指在屏幕下拉多少距离会触发下拉刷新
● setProgressBackgroundColor(); // 设定下拉圆圈的背景
● setColorSchemeResources();//设置刷新控件动画中的颜色。参数为资源id
● setColorSchemeColors();//设置刷新控件动画中的颜色。参数为颜色值。
● setDistanceToTriggerSync();// 设置手指在屏幕下拉多少距离会触发下拉刷新
● setRefreshing();//设置控件是否进行刷新
● setSize(); // 设置圆圈的大小 SwipeRefreshLayout.DEFAULT 和 LARGE 两个参数
▼ SwipeRefreshLayout.OnRefreshListener 监听
● setOnRefreshListener(this);设置监听回调 重写onRefresh()方法
Screenshot_20170406-115406.png▼ SwipeRefreshLayout注意
● 像模仿知乎上来就加载下拉刷新的进度条时,当你setRefreshing(true),发现不会调用onRefresh()方法;需要手动自己调用。
<pre>
swipeRefreshLayout.post(new Runnable(){
@Override
public void run() {
swipeRefreshLayout.setRefreshing(true);
}
});
this.onRefresh();
</pre>
这是为什么呢?
查看onRefresh()注释:
<pre>
/**
* Classes that wish to be notified when the swipe gesture correctly
* triggers a refresh should implement this interface.
*/
</pre>
意思是 希望得到通知时,滑动手势正确触发刷新应该实现这个接口。
查看源码 只有 动画结束时会调用 onRefresh()
<pre>
@Override
public void onAnimationEnd(Animation animation) {
if (mRefreshing) {
// Make sure the progress view is fully visible
mProgress.setAlpha(MAX_ALPHA);
mProgress.start();
if (mNotify) {
if (mListener != null) {
mListener.onRefresh();
}
}
mCurrentTargetOffsetTop = mCircleView.getTop();
} else {
reset();
}
}
};
</pre>
而当你setRefreshing()的时候执行的是以下代码,因为mNotify这个成员变量是赋值为 false的 ,所以不会调用onRefresh()方法。
<pre>
public void setRefreshing(boolean refreshing) {
if (refreshing && mRefreshing != refreshing) {
// scale and show
mRefreshing = refreshing;
int endTarget = 0;
if (!mUsingCustomStart) {
endTarget = (int) (mSpinnerFinalOffset + mOriginalOffsetTop);
} else {
endTarget = (int) mSpinnerFinalOffset;
}
setTargetOffsetTopAndBottom(endTarget - mCurrentTargetOffsetTop,
true /* requires update /);
mNotify = false;
startScaleUpAnimation(mRefreshListener);
} else {
setRefreshing(refreshing, false / notify */);
}
}
</pre>
再往下看一下 重载的两个参数的 setRefreshing(boolean refreshing, final boolean notify) 可以设置mNotify这个值 但是这个方法没有暴露 而只有finishSpinner()方法调用了setRefreshing(boolean refreshing, final boolean notify),并且只有当SwipeRefreshLayout触发自己ontouchevent()方法才会调用finishSpinner()方法。所以onRefresh()的注释的原理就在这里
<pre>
private void setRefreshing(boolean refreshing, final boolean notify) {
if (mRefreshing != refreshing) {
mNotify = notify;
ensureTarget();
mRefreshing = refreshing;
if (mRefreshing) {
animateOffsetToCorrectPosition(mCurrentTargetOffsetTop, mRefreshListener);
} else {
startScaleDownAnimation(mRefreshListener);
}
}
}
</pre>