RecycleView的Item点击事件

文章转自 冥萧阳大神 原作地址http://blog.csdn.net/guxiao1201/article/details/40423361


RecyclerView不再负责Item视图的布局及显示,所以RecyclerView也没有为Item开放OnItemClick等点击事件,这就需要开发者自己实现。博客最下面有Demo程序运行动画。

奉上Demo的Github链接

在调研过程中,发现有同学修改RecyclerView源码来实现Item的点击监听,但认为这不是一个优雅的解决方案,最终决定在RecyclerView.ViewHolder上做文章。

思路是:因为ViewHolder我们可以拿到每个Item的根布局,所以如果我们为根布局设置单独的OnClick监听并将其开放给Adapter,那不就可以在组装RecyclerView时就能够设置ItemClickListener,只不过这个Listener不是设置到RecyclerView上而是设置到Adapter。

我们首先看ViewHolder的代码:

[java]  view plain  copy
  1. public class MyViewHolder extends ViewHolder implements OnClickListener,OnLongClickListener{  
  2.   
  3.     public ImageView iv;  
  4.     public TextView tv;  
  5.     private MyItemClickListener mListener;  
  6.     private MyItemLongClickListener mLongClickListener;  
  7.       
  8.     public MyViewHolder(View rootView,MyItemClickListener listener,MyItemLongClickListener longClickListener) {  
  9.         super(rootView);  
  10.         iv = (ImageView)rootView.findViewById(R.id.item_iv);  
  11.         tv = (TextView)rootView.findViewById(R.id.item_tv);  
  12.         this.mListener = listener;  
  13.         this.mLongClickListener = longClickListener;  
  14.         rootView.setOnClickListener(this);  
  15.         rootView.setOnLongClickListener(this);  
  16.     }  
  17.   
  18.     /** 
  19.      * 点击监听 
  20.      */  
  21.     @Override  
  22.     public void onClick(View v) {  
  23.         if(mListener != null){  
  24.             mListener.onItemClick(v,getPosition());  
  25.         }  
  26.     }  
  27.   
  28.     /** 
  29.      * 长按监听 
  30.      */  
  31.     @Override  
  32.     public boolean onLongClick(View arg0) {  
  33.         if(mLongClickListener != null){  
  34.             mLongClickListener.onItemLongClick(arg0, getPosition());  
  35.         }  
  36.         return true;  
  37.     }  
  38.   
  39. }  
因为在构造ViewHolder时,rootView将作为一个必传参数传递进来,所以我们只需要拿到rootView并给其绑定点击监听事件即可。

下面要考虑的就是怎样把listener传递进来。Demo中设定了监听点击事件的Interface:MyItemClickListener:

[java]  view plain  copy
  1. public interface MyItemClickListener {  
  2.     public void onItemClick(View view,int postion);  
  3. }  
MyItemClickListener模仿ListView的OnItemClickListener,开放了view和position两个参数,这对习惯使用ListView的开发者们使用起来更得心应手。从ViewHolder的代码中可以看到,执行onClick方法时会调用getPosition()将当前Item的位置回调给listener。getPosition()是ViewHolder的内置方法,可直接使用。

上面提到过,listener是设定到Adapter上的,所以Adapter就需要对外开放相关方法:

[java]  view plain  copy
  1. @Override  
  2.     public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {  
  3.         View itemView = LayoutInflater.from(parent.getContext()).inflate(R.layout.item, parent,false);  
  4.         MyViewHolder vh = new MyViewHolder(itemView,mItemClickListener,mItemLongClickListener);  
  5.         return vh;  
  6.     }  
  7.   
  8.     /** 
  9.      * 设置Item点击监听 
  10.      * @param listener 
  11.      */  
  12.     public void setOnItemClickListener(MyItemClickListener listener){  
  13.         this.mItemClickListener = listener;  
  14.     }  
  15.       
  16.     public void setOnItemLongClickListener(MyItemLongClickListener listener){  
  17.         this.mItemLongClickListener = listener;  
  18.     }  
上篇博客(Android-RecylerView初识)提到过,Adapter的onCreateViewHolder是负责实例化每个Item的视图,所以我在实例化视图时就将listener传递给ViewHolder。

最后就是组装RecyclerView时根据需求设定点击监听了:

[java]  view plain  copy
  1. /** 
  2.      * 初始化RecylerView 
  3.      */  
  4.     private void initView(){  
  5.         mRecyclerView = (RecyclerView)findViewById(R.id.recyclerView);  
  6.         MyLayoutManager manager = new MyLayoutManager(this);  
  7.         manager.setOrientation(LinearLayout.HORIZONTAL);//默认是LinearLayout.VERTICAL  
  8.         mRecyclerView.setLayoutManager(manager);  
  9.         mRecyclerView.setItemAnimator(new DefaultItemAnimator());  
  10.     }  
  11.       
  12.     private void initData(){  
  13.         this.mData = new ArrayList();  
  14.         for(int i=0;i<20;i++){  
  15.             MyItemBean bean = new MyItemBean();  
  16.             bean.tv = "Xmy"+i;  
  17.             mData.add(bean);  
  18.         }  
  19.         this.mAdapter = new MyAdapter(mData);  
  20.         this.mRecyclerView.setAdapter(mAdapter);  
  21.         RecyclerView.ItemDecoration decoration = new MyDecoration(this);  
  22.         this.mRecyclerView.addItemDecoration(decoration);  
  23.         this.mAdapter.setOnItemClickListener(this);  
  24.         this.mAdapter.setOnItemLongClickListener(this);  
  25.     }  
Demo为ViewHolder设置了OnClick和OnLongClickListener,在Activity中我们实现了接口方法并在里面打印Toast提示:

[java]  view plain  copy
  1. @Override  
  2.     public void onItemClick(View view, int postion) {  
  3.         MyItemBean bean = mData.get(postion);  
  4.         if(bean != null){  
  5.             Toast.makeText(this, bean.tv, Toast.LENGTH_SHORT).show();  
  6.         }  
  7.     }  
  8.   
  9.     @Override  
  10.     public void onItemLongClick(View view, int postion) {  
  11.         MyItemBean bean = mData.get(postion);  
  12.         if(bean != null){  
  13.             Toast.makeText(this"LongClick "+bean.tv, Toast.LENGTH_SHORT).show();  
  14.         }  
  15.     }  

你可能感兴趣的:(RecycleView的Item点击事件)