▼ SwipeRefreshLayout簡述
SwipeRefreshLayout為5.0系統后,谷歌官方推出SwipeRefreshLayout控件,在android-support-v4.jar包中,有一股簡潔的風格。
上面的意思就是說,當用戶通過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()方法
▼ 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>