RecycleView的UP和Down加载

由于项目中使用到了协调者 布局,需要与其协调配合的可滑动的View,开始用到了ListView+NestedScrollView的方案,但是各种问题出来了,解决上拉刷新和下拉加载的方案:重写NestedScrollView,判断其是否滑动到底部,这样的思路下去就实现了这个功能,但是ListView+ScrollVIew的组合是很不好的,Google的建议是这两者是不能一起用的,实测getView()方法调用了2倍以上,新能很差,so问题来了,那么用什么代替呢?RecycleView。
这里写图片描述
这是上述的要达到的效果.
思路:
1.继承RecycleView.
2.内部实现ViewHolder和Adapter.
3.抓取符合刷新的时间点.
4.动态改变item的数量,完成刷新效果。
5.刷新结束,动态移除刷新的item布局。

/**
  * BaseAdapter,控制头尾的刷新视图
 * <p/>
 * RecyclerView.INVALID_TYPE:是-1,这里作为头部视图
 */

class Adapter extends RecyclerView.Adapter<ViewHolder> {

    @Override
    public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        //当是普通的状态时,就直接交由用户的Adapter处理
        if (mState==STATE_NOMAL){
            return mAdapter.onCreateViewHolder(parent, viewType);
        }
        //flase
        if (viewType == INVALID_TYPE && isNeedHeadFresh) {
            return new ViewHolder(head);
        } else if (viewType == INVALID_TYPE - 1 && isNeedFootFresh) {
            return new ViewHolder(foot);
        }
        return mAdapter.onCreateViewHolder(parent, viewType);
    }

    @Override
    public void onBindViewHolder(ViewHolder holder, int position) {
    //普通状态交给用户的Adapter
        if (mState==STATE_NOMAL){
            mAdapter.onBindViewHolder(holder,position);
            return;
        }
        //这里Adapter为空的话也不会进行任何才做
        if (position >= 0 && position < getHeadCount()) {
            return;
        }
        //将真实的position交给用户
        if (mAdapter != null) {
            int real = position - getHeadCount();
            if (real < mAdapter.getItemCount()) {
                mAdapter.onBindViewHolder(holder, real);
            }
        }
    }

    @Override
    public int getItemCount() {
    //普通状态,就返回用户的count
        if (mState==STATE_NOMAL){
            return mAdapter.getItemCount();
        }
        //返回所有的item,包括刷新的item数量
        return mAdapter == null ? getHeadCount() + getFootCount() : +getHeadCount() + getFootCount() + mAdapter.getItemCount();
    }

    @Override
    public int getItemViewType(int position) {
        if (mState==STATE_NOMAL){
            return mAdapter.getItemViewType(position);
        }
        //讲位置0设为INVALID_TYPE-->代表head刷新
        if (getHeadCount() > 0&&position<1) {
            return INVALID_TYPE;
        }
        //内容item的type由用户决定
        int real = position - getHeadCount();
        if (real < mAdapter.getItemCount()) {
            return mAdapter.getItemViewType(real);
        }
        //else就是尾部刷新视图
        return INVALID_TYPE - 1;
    }
}

//内部实现ViewHolder,目的是为了能让YzzRecycleView只是负责刷新效果的实现,其他item的数据填充和初始化都是依赖用户自己操作的,和普通的步骤一致。  
 public static class ViewHolder extends    RecyclerView.ViewHolder    {
    public ViewHolder(View itemView) {
        super(itemView);
    }
}

好了,现在我们已经实现了Adapter和ViewHolder了,在这里讲刷新和呈现conntentItem分开,进一步增强了程序的可扩展性。

接下来我们要找到刷新的时机:onTuchEvent()方法,当item布局设置了监听之后,该方法是接收不到ACTION_DOWN这个事件的,原因很简单,RecycleView在收到该事件的时候,不会拦截的,会分发到下层View中,因为点击事件的设定,那么就要活得ACTION_DOWN事件,父容器是不可以拦截的,如果不设置的话,RecycleView会在ACTION_DOWN拦截一次,后续的一系列动作都会在父容器中处理。那么点击的初始位子就要在dispatchTouchEvent或者onInterceptTouchEvent这两个方法中获取。

当确定了到达顶部或者底部时并不是立马就刷新,是有一个滑动的最小距离的,当滑动超过这个距离,就会触发刷新。

 @Override
public boolean onTouchEvent(MotionEvent e) {
    switch (e.getAction()){
        case MotionEvent.ACTION_DOWN:
            mFy = e.getY();
            break;
        case MotionEvent.ACTION_UP:
        //获得移动的距离(y轴)
            float y = e.getY()-mFy;
            //加载
            loadMore(y);
            //标志位复位
            opPenHead = false;
            opFoot = false;
            break;
    }
    return super.onTouchEvent(e);
}
//50是最小的滑动距离
private void loadMore(float x) {
    if (x>50&&opPenHead){
        mState = STATE_FRESHING;
        if (mOnLoadMore!=null){
            mOnLoadMore.onLoadMore();
            notifyData();
        }
    }
    if (x<-50&&opFoot){
        mState = STATE_FRESHING;
        if (mOnLoadMore!=null){
            mOnLoadMore.onLoadMore();
            notifyData();
            scrollBy(0,STATIC_HEIGHT);
        }
    }
}

//这里我舒心的内部实现的Adapter,如果用户刷新用户的Adapter的话,我这里Adapter的不会执行,所以,用户在跟新完数据后调用此方法即可。
public void notifyData(){
mInnerAdapter.notifyDataSetChanged();
}

接下来是刷新的接口了
public interface OnLoadMoreListener{
void onLoadMore();
}

//init方法,设置了滑动监听,判断是否达到底部

private void init() {
    mInnerAdapter = new Adapter();
    head = LayoutInflater.from(getContext()).inflate(R.layout.refresh, null);
    foot = LayoutInflater.from(getContext()).inflate(R.layout.refresh, null);
    addOnScrollListener(new OnScrollListener() {
        @Override
        public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
            super.onScrollStateChanged(recyclerView, newState);
            //这里要判断是否是到达顶部和尾部
            //当在刷新的时候就不做任何处理
            if (mState == STATE_FRESHING) {
                return;
            }
            if (mAdapter==null){
                return;
            }
            int visibleCount = getLayoutManager().getChildCount();
            if (visibleCount > 0 &&newState==RecyclerView.SCROLL_STATE_IDLE&&isNeedFresh()){
                //判断是否到达顶部
                if (getLayoutManager() instanceof LinearLayoutManager){
                    int first = ((LinearLayoutManager)getLayoutManager()).findFirstVisibleItemPosition();
                    if (first==0&&isNeedHeadFresh) {
                        opPenHead = true;
                    }
                    int last = ((LinearLayoutManager)getLayoutManager()).findLastVisibleItemPosition();

                    if (last==mAdapter.getItemCount()-1){
                        opFoot = true;
                    }
                }
            }
        }

        @Override
        public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
            super.onScrolled(recyclerView, dx, dy);

        }
    });
}`` 

好了基本上,已经处理完了,在外部使用时候,也很简单和普通使用一样,但是ViewHolder必须是集成YzzRecycleView.ViewHolder,这样方可setMAdapter()进行初始化。

RecyclerView.LayoutManager manager = new LinearLayoutManager(this);
recyclerView.setLayoutManager(manager);
adapter = new RecycleAdapter(this);
list = new ArrayList<>();
for (int i = 0; i <20 ; i++) {
list.add("=="+i+"==");
}
adapter.setList(list);
recyclerView.setNeedFootFresh(true);
recyclerView.setNeedHeadFresh(true);
recyclerView.setMAdapter(adapter);

这样就ok了,基本实现了刷新的功能,ok。
GitHub地址:https://github.com/yzzAndroid/MyRecycleView

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值