ISwipeRefreshLayout

Jake大神都去Google搞Kotlin了~

ISwipeRefreshLayout是基于SwipeRefreshLayout源碼基礎上修改,便于使用自定義loading樣式的下拉刷新組件。

為什么寫這個組件?

原生的SwipeRefreshLayout好歸好,但它不能自定義動畫效果,只能簡單改下color、alpha等,往往實際開發中都是自家的loading效果。所以就有了ISwipeRefreshLayout

這里貼一個現在項目使用的loading。

錄屏的問題動畫太快了

還不錯吧,說實話雀氏有點low。但它不是畫出來了,是幀動畫在切換,而且內存控制的很好,推薦下這個組件FrameAnimDrawable

用法

和SwipeRefreshLayout幾乎一樣,并且適用于所有view。

  • xml布局
<io.jiantao.android.uikit.refresh.ISwipeRefreshLayout
        android:id="@+id/refresh_layout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <android.support.v7.widget.RecyclerView
            android:id="@+id/recyclerView"
            android:layout_width="match_parent"
            android:layout_height="match_parent"/>
    </io.jiantao.android.uikit.refresh.ISwipeRefreshLayout>
  • 代碼中
refreshLayout = (ISwipeRefreshLayout) findViewById(R.id.refresh_layout);
        refreshLayout.setOnRefreshListener(new ISwipeRefreshLayout.OnRefreshListener() {
            @Override
            public void onRefresh() {
                // do refresh ...
                handler.sendEmptyMessageDelayed(1, 3000);
            }
        });
        refreshLayout.setRefreshHeaderView(new AvLoadingRefreshView(this));
//------------------------------
//手動刷新和取消(ture/false)
ISwipeRefreshLayout.setRefreshing(true);

//另外一種方式,當然就是下拉刷新了,內部實現下文會提到。

實現過程

  • 剔除原有效果
    //  這兩個類就是原生那個轉的圈圈
    CircleImageView mCircleView;
    MaterialProgressDrawable mProgress;

把SwipeRefreshLayout代碼爬出來,發現就這兩個依賴類,剔除簡單方便。

  • 自定義loading
//定義成員變量view
private View mRefreshView;
...

//并添加設置入口
public void setRefreshHeaderView(View view) {
        if(view == null){//為了安全,這里還應該判斷下是否正在執行動畫
            return;
        }
        removeView(mRefreshView);//移除舊的
        this.mRefreshView = view;
        view.setMinimumHeight(HEADER_VIEW_MIN_HEIGHT);
        addView(view);//添加新的
        getRefreshTrigger().init();
    }

以上代碼很簡單,暴露自定義view方法。

// onMeasure 方法計算view高度
@Override
    public void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);
        ... 省略代碼 ...
        if(mRefreshView != null){
            //計算view寬高
            measureChild(mRefreshView, widthMeasureSpec, heightMeasureSpec);
            //緩存一些變量
            updateBaseValues(mRefreshView.getMeasuredHeight());
            mRefreshViewIndex = -1;
            // Get the index of the mRefreshView. 這里是用于改變view的繪制順序的。
            for (int index = 0; index < getChildCount(); index++) {
                if (getChildAt(index) == mRefreshView) {
                    mRefreshViewIndex = index;
                    break;
                }
            }
        }
    }
// onLayout 方法布局
@Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        final int width = getMeasuredWidth();
        final int childTop = getPaddingTop();
         ... 省略代碼 ...
        if(mRefreshView != null){
            int refreshWidth = mRefreshView.getMeasuredWidth();
            //使mRefreshView相對父控件水平方向居中,豎直方向在父控件的上面位置,剛好看不見。mRefreshViewHeight就是onMeasure得到的值。
            mRefreshView.layout((width/2 - refreshWidth/2), childTop - mRefreshViewHeight, (width/2 + refreshWidth/2), childTop);
        }
    }

以上為計算和布局過程修改,下面說下滑動過程處理,更新百分比,view滑動,觸發回調刷新等等。

//手指move事件觸發此方法
@SuppressLint("NewApi")
    private void moveSpinner(float overscrollTop) {
       ... 省略代碼 ...
        if (mScale) {//大小縮放動畫
            setAnimationProgress(Math.min(1f, overscrollTop / mTotalDragDistance));
        }
        float progress = overscrollTop / mTotalDragDistance;
        if (progress < 1.0f) {//下拉至觸發點的百分比
            getRefreshTrigger().onPullDownState(progress);
        } else {//釋放即可觸發刷新
            getRefreshTrigger().onReleaseToRefresh();
        }
        //下拉過程view的移動距離
        final float tranlationY = overscrollTop > mTotalDragDistance ? mTotalDragDistance : overscrollTop;
        //執行移動動畫
        translateContentViews(tranlationY);
    }

畢竟是在源碼基礎上改動,整個過程還是比較簡單。

感興趣的朋友可參考修改后的代碼

最后,推薦一個動畫庫,效果很棒,代碼簡潔,本文ISwipeRefreshLayout組件可直接使用。代碼在手,想怎么搞都行。

avi.gif

loading_test.gif

文中難免有錯誤之處,還望指出,歡迎評論交流。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 173,536評論 25 708
  • 發現 關注 消息 iOS 第三方庫、插件、知名博客總結 作者大灰狼的小綿羊哥哥關注 2017.06.26 09:4...
    肇東周閱讀 12,252評論 4 61
  • App推廣渠道分類解析 1)各大應用市場 首先要發布我們的APP應用,在各大手機廠商市場、各大網絡運營商、獨立商店...
    Rijkaa閱讀 524評論 0 3
  • 看完《七月與安生》,突然想起18歲以前的日子,那些青春期的萌動,青春期的矯情和青春期特有的暗戀。 1...
    魚洛洛閱讀 324評論 0 0
  • 進度:184--271 靈與肉、輕與重之疑惑 ‘靈與肉’‘輕與重’~寓意深遠的哲學命題。簡單的閱讀引發了思索:腦海...
    般若自在閱讀 340評論 0 0