下拉刷新 一(SwipeRefreshLayout深入理解)

2017-04-06  本文已影响0人  吴振宇

▼ 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>

Paste_Image.png

第一次写这么多的文章,可能写的不够易懂,见谅。

上一篇 下一篇

猜你喜欢

热点阅读