RecycleView的UP和Down加载
来源:互联网 发布:select to SQL 编辑:程序博客网 时间:2024/06/06 16:34
由于项目中使用到了协调者 布局,需要与其协调配合的可滑动的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这两个方法中获取。
当确定了到达顶部或者底部时并不是立马就刷新,是有一个滑动的最小距离的,当滑动超过这个距离,就会触发刷新。
@Overridepublic 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
- RecycleView的UP和Down加载
- Extjs4中up()和down()的用法
- Extjs4 中up()和down()的用法
- Extjs4中up()和down()的用法
- Extjs4中up()和down()的用法
- Extjs4中up()和down()的用法
- Extjs4中up()和down()的用法
- mouse down和mouse up的合用
- Extjs4中up()和down()的用法
- Extjs4中up()和down()的用法
- Extjs4中up()和down()的用法
- Extjs4中up()和down()的用法
- JAVA中泛型的up-case 和 down-case
- vc中处理按钮的UP和DOWN事件
- touch up inside 和 touch down 的区别
- ifup/ifdown ethX 和 ifconfig ehtX up/down的区别
- 端口的UP与down
- RecycleView的头部加载
- VC1和VS的对应关系
- 内存溢出,main线程与虚拟机的关系猜测,待验证
- UML9张图总结下--机房收费系统
- Linux内核分析(一):计算机工作原理
- 不敢死队问题
- RecycleView的UP和Down加载
- redis使用教程(安装和连接)
- Codeforces Round #401 (Div. 2) D. Cloud of Hashtags
- 高精度算法加法
- sys/time.h 是GCC的编译标准写法
- hdu2521 反素数(求约数个数)
- android studio 中 java.lang.NoClassDefFoundError:retrofit2.Retrofit$Builder错误解决
- warning: Error opening file (../../modules/highgui/src/cap_ffmpeg_impl.hpp:545)
- TensorFlow数据处理方法