美文网首页Android知识Android进阶之旅
RecyclerView,ListView实现加载更多

RecyclerView,ListView实现加载更多

作者: SevChen | 来源:发表于2016-09-02 15:09 被阅读258次

    分页是Android性能优化和提升用户体验的一个重要手段,因此,几乎在所有项目中,都存在上拉加载更多的功能。目前第三方上拉加载更多的项目五花百门,但是都存在一个缺点,就是,往往我们只需要一个小功能,但是却不得不导入大量无用代码,所以何不借助RecyclerView和ListView来实现自己的加载更多控件呢。

    需求

    1. 滑动到最底部时,执行加载更多操作。
    2. 临界点问题:当页面数据没有超出屏幕时,加载更多功能自动禁止。
    3. 临界点问题:当最后一个item可见时就执行加载更多。

    RecyclerView 实现加载更多

    加载进度条作为recyclerview的最后一项,通过一个封装AdapterWrapper将实际的Adapter和加载更多的Item结合起来。

    // 如果count的大小没变化,那么AdapterWrapper实际上就是一个mAdapter的代理。
    private class AdapterWrapper extends RecyclerView.Adapter{
    
            private AdapterWrapper() {
            }
    
            @Override
            public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
                if (viewType <= 0) {
                    return loadMoreAdapter.onCreateViewHolder(parent, viewType);
                } else {
                    return mAdapter.onCreateViewHolder(parent, viewType);
                }
            }
    
            @Override
            public void onBindViewHolder(ViewHolder holder, int position) {
                if (position < mAdapter.getItemCount()) {
                    mAdapter.onBindViewHolder(holder, position);
                }
                else {
                    loadMoreAdapter.onBindViewHolder(holder, position);
                }
            }
    
            @Override
            public int getItemCount() {
                // 如果开启加载更多功能,那么多增加一项作为加载的进度状态
                if (loadMoreEnable) {
                    return mAdapter.getItemCount() + 1;
                }
                else {
                    return mAdapter.getItemCount();
                }
            }
    
            @Override
            public int getItemViewType(int position) {
                if (position < mAdapter.getItemCount()) {
                    return mAdapter.getItemViewType(position);
                }
                else {
                    return loadMoreType;
                }
            }
    
            @Override
            public long getItemId(int position) {
                if (position < mAdapter.getItemCount()) {
                    return mAdapter.getItemId(position);
                }
                else {
                    return position;
                }
            }
    
            @Override
            public void onBindViewHolder(ViewHolder holder, int position, List payloads) {
                if (position < mAdapter.getItemCount()) {
                    mAdapter.onBindViewHolder(holder, position, payloads);
                }
                else {
                    super.onBindViewHolder(holder, position, payloads);
                }
            }
        }
    

    LoadMoreAdapter模拟RecyclerView.Adapter的2个核心回调,通过实现LoadMoreAdapter定制加载进度的显示样式,如果开启加载更多,必须设置LoadMoreAdapter。

        public static abstract class LoadMoreAdapter {
            public abstract ViewHolder onCreateViewHolder(ViewGroup parent, int viewType);
            public abstract void onBindViewHolder(ViewHolder holder, int position);
        }
    

    RecyclerView.OnScrollListener有2个核心回调

    • void onScrolled(RecyclerView recyclerView, int dx, int dy)
      每次RecyclerView滑动的时候都会触发这个函数,注意,如果已经滑动到底部,那么这个函数就不会再被触发。
    • void onScrollStateChanged(RecyclerView recyclerView, int newState)
      每次滑动过程,会触发3次这个函数,先后顺序分别是
      1. SCROLL_STATE_DRAGGING 手指接触屏幕
      2. SCROLL_STATE_SETTLING 手指离开屏幕
      3. SCROLL_STATE_IDLE RecyclerView停止滑动
    private RecyclerView.OnScrollListener onScrollListener = new OnScrollListener() {
    
            /**
             * 手指滑动过程,recyclerView是否发生滑动
             */
            private boolean hasScrolled;
            /**
             * 是否已经初始化
             */
            private boolean inited;
            /**
             * 数据是否超出屏幕,如果不超出屏幕,那么无论loadMoreEnable是true或者false,都不能加载更多。
             */
            private boolean dataOutOfScreen;
    
            @Override
            public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
                super.onScrollStateChanged(recyclerView, newState);
    
                if (newState == RecyclerView.SCROLL_STATE_IDLE) {
                    // recyclerView 停止滑动时
                    LinearLayoutManager llm = (LinearLayoutManager) recyclerView.getLayoutManager();
    
                    // 只要最后一个item可见,即可开始加载。
                    boolean scrollToEnd = llm.findLastVisibleItemPosition() == llm.getItemCount() - 1;
    
    
                    if ((dataOutOfScreen ||hasScrolled) && scrollToEnd && loadMoreEnable) {
                        // 开始加载,并设置只能同时存在一次加载
                        if (loadMoreListener != null && !isLoading) {
                            isLoading = true;
    
                            loadMoreType = TYPE_LOADMORE_LOADING;
                            adapterWrapper.notifyDataSetChanged();
    
                            loadMoreListener.onLoadMore();
                        }
                    }
                } else if (newState == RecyclerView.SCROLL_STATE_DRAGGING) {
                    // 手指刚刚触摸屏幕开始滑动时
                    hasScrolled = false;
                }
            }
    
            @Override
            public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
                super.onScrolled(recyclerView, dx, dy);
    
                hasScrolled = true;
    
                if (!inited) {
                    init(recyclerView);
                }
            }
    
            /**
             * 如果数据没有超出屏幕,那么,不会发生加载更多的事件
             * @param recyclerView
             */
            private void init(RecyclerView recyclerView) {
                inited = true;
    
                LinearLayoutManager llm = (LinearLayoutManager) recyclerView.getLayoutManager();
                boolean scrollToEnd = llm.findLastCompletelyVisibleItemPosition() == llm.getItemCount() - 1;
    
                if (!scrollToEnd) {
                    dataOutOfScreen = true;
                }
            }
        };
    

    在setAdapter中初始化AdapterWrapper和RecyclerView.OnScrollListener

        public void setAdapter(Adapter adapter) {
            mAdapter = adapter;
            if (mAdapter == null) {
                throw new IllegalArgumentException("adapter不能为null");
            }
    
            adapterWrapper = new AdapterWrapper();
            super.setAdapter(adapterWrapper);
    
            removeOnScrollListener(onScrollListener);
            addOnScrollListener(onScrollListener);
        }
    

    注意,由于设置的Adapter被包装了一层,所以类似notifyDataSetChanged()这样的接口将没有效果。

    相关文章

      网友评论

        本文标题:RecyclerView,ListView实现加载更多

        本文链接:https://www.haomeiwen.com/subject/phvhettx.html