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,219评论 25 708
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,229评论 4 61
  • App推广渠道分类解析 1)各大应用市场 首先要发布我们的APP应用,在各大手机厂商市场、各大网络运营商、独立商店...
    Rijkaa阅读 524评论 0 3
  • 看完《七月与安生》,突然想起18岁以前的日子,那些青春期的萌动,青春期的矫情和青春期特有的暗恋。 1...
    鱼洛洛阅读 324评论 0 0
  • 进度:184--271 灵与肉、轻与重之疑惑 ‘灵与肉’‘轻与重’~寓意深远的哲学命题。简单的阅读引发了思索:脑海...
    般若自在阅读 330评论 0 0