RecyclerView的封装以实现上拉加载更多
下拉加载更多功能是怎么实现的呢?
当实现recyclerview的时候,通常需要实现adapter跟viewholder,首先我们要明白adapter里面各个方法的调用顺序。
首先调用getItemCount(),作为recyclerview里的item数量
调用getItemViewType(int position),该方法返回一个int值作为onCreateViewHolder中的viewtype参数
调用onCreateViewHolder(ViewGroup parent, int viewType)
调用onBindViewHolder(BaseViewHolder holder, int position)
recyclerview的设计思路是这样的,
如上,adapter负责把数据传给viewholder,viewhoder就相当于一个item,recyclerview通过布局策略layout这些item。
adapter与recyclerview是一个观察者模式,当adpater发生变化通知recyclerview触发重新布局,这就是我们调用adapter的notify时会发生的改变。
简单讲完recyclerview的原理,接下来就可以开始说说我们的上拉加载更多了。
1.首先我们需要下拉加载更多的进度条,因此当我们当我们返回item数量时要多返回一个item,作为显示加载更多的item。如下,我们返回data的数量+1的item数。
@Override public int getItemCount() { if (mNewsData.size() > 0) { return mNewsData.size()+1; } return 0; }
2.当加载到最后一个数据时,这个item为加载更多的那个item,返回不同的viewtype给onCreateViewHolder
@Override public int getItemViewType(int position) { if (position>=mNewsData.size()){ return TYPE_FOOT; } return TYPE_NORMAL; }
3.onCreateViewHolder根据不同的viewtype返回不同的布局。
@Override public BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { if (viewType==TYPE_FOOT){ View itemView = LayoutInflater.from(mContext).inflate(R.layout.listview_footer, parent, false); return new BaseViewHolder(itemView, mContext); }else { View itemView = LayoutInflater.from(mContext).inflate(R.layout.item_news, parent, false); return new BaseViewHolder(itemView, mContext); } }
到此我们就实现了加载更多的布局,但是我们还需要去加载数据。我们可以封装recyclerview,当它滑动到数据底部的时候回调接口,触发下载跟多数据。并notify adapter。
recyclerview中有个onScrollStateChanged(int state)方法,State主要有:
SCROLL_STATE_IDLE表示当前并不处于滑动状态
SCROLL_STATE_DRAGGING表示当前RecyclerView处于滑动状态(手指在屏幕上)
SCROLL_STATE_SETTLING表示当前RecyclerView处于滑动状态,(手已经离开屏幕)。
我们只要判断当recyclerview没有在滑动的时候,这时是否已经滑动到了最后一个item。
@Override public void onScrollStateChanged(int state){ if (state==RecyclerView.SCROLL_STATE_IDLE){ Log.d("LoadMoreRecyclerView","run in onScrollStateChanged"); LayoutManager layoutManager=getLayoutManager(); int lastVisiblePosition; if (layoutManager instanceof GridLayoutManager){ lastVisiblePosition=((GridLayoutManager) layoutManager).findLastVisibleItemPosition(); }else if (layoutManager instanceof StaggeredGridLayoutManager){ int into[]=new int[((StaggeredGridLayoutManager) layoutManager).getSpanCount()]; ((StaggeredGridLayoutManager)layoutManager).findLastVisibleItemPositions(into); lastVisiblePosition=findMax(into); }else { lastVisiblePosition= ((LinearLayoutManager) layoutManager).findLastVisibleItemPosition(); } Log.d("LoadMoreRecyclerView","ChildCount: "+layoutManager.getChildCount()+" lastvisiblePosition: " +lastVisiblePosition+" ItemCount: "+layoutManager.getItemCount()); if (layoutManager.getChildCount()>0 //当当前显示的item数量>0 &&lastVisiblePosition>=layoutManager.getItemCount()-1 //当当前屏幕最后一个加载项位置>=所有item的数量 &&layoutManager.getItemCount()>layoutManager.getChildCount()) { // 当当前总Item数大于可见Item数 Log.d("LoadMoreRecyclerView","run onLoadMore"); if (mLoadMoreListener!=null){ mLoadMoreListener.onLoadMore(); } } } }
通过回调LoadMoreListener的onLoadMore()我们就可以去加载数据,并通过adapter更新recyclerview了。