在上一篇博客Android-RecylerView初识中提到,RecyclerView不再负责Item视图的布局及显示,所以RecyclerView也没有为Item开放OnItemClick等点击事件,这就需要开发者自己实现。博客最下面有Demo程序运行动画。
奉上Demo的Github链接。
在调研过程中,发现有同学修改RecyclerView源码来实现Item的点击监听,但认为这不是一个优雅的解决方案,最终决定在RecyclerView.ViewHolder上做文章。
思路是:因为ViewHolder我们可以拿到每个Item的根布局,所以如果我们为根布局设置单独的OnClick监听并将其开放给Adapter,那不就可以在组装RecyclerView时就能够设置ItemClickListener,只不过这个Listener不是设置到RecyclerView上而是设置到Adapter。
我们首先看ViewHolder的代码:
- public class MyViewHolder extends ViewHolder implements OnClickListener,OnLongClickListener{
- public ImageView iv;
- public TextView tv;
- private MyItemClickListener mListener;
- private MyItemLongClickListener mLongClickListener;
- public MyViewHolder(View rootView,MyItemClickListener listener,MyItemLongClickListener longClickListener) {
- super(rootView);
- iv = (ImageView)rootView.findViewById(R.id.item_iv);
- tv = (TextView)rootView.findViewById(R.id.item_tv);
- this.mListener = listener;
- this.mLongClickListener = longClickListener;
- rootView.setOnClickListener(this);
- rootView.setOnLongClickListener(this);
- }
- /**
- * 点击监听
- */
- @Override
- public void onClick(View v) {
- if(mListener != null){
- mListener.onItemClick(v,getPosition());
- }
- }
- /**
- * 长按监听
- */
- @Override
- public boolean onLongClick(View arg0) {
- if(mLongClickListener != null){
- mLongClickListener.onItemLongClick(arg0, getPosition());
- }
- return true;
- }
- }</span>
下面要考虑的就是怎样把listener传递进来。Demo中设定了监听点击事件的Interface:MyItemClickListener:
- public interface MyItemClickListener {
- public void onItemClick(View view,int postion);
- }
上面提到过,listener是设定到Adapter上的,所以Adapter就需要对外开放相关方法:
- @Override
- public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
- View itemView = LayoutInflater.from(parent.getContext()).inflate(R.layout.item, parent,false);
- MyViewHolder vh = new MyViewHolder(itemView,mItemClickListener,mItemLongClickListener);
- return vh;
- }
- /**
- * 设置Item点击监听
- * @param listener
- */
- public void setOnItemClickListener(MyItemClickListener listener){
- this.mItemClickListener = listener;
- }
- public void setOnItemLongClickListener(MyItemLongClickListener listener){
- this.mItemLongClickListener = listener;
- }
最后就是组装RecyclerView时根据需求设定点击监听了:
- /**
- * 初始化RecylerView
- */
- private void initView(){
- mRecyclerView = (RecyclerView)findViewById(R.id.recyclerView);
- MyLayoutManager manager = new MyLayoutManager(this);
- manager.setOrientation(LinearLayout.HORIZONTAL);//默认是LinearLayout.VERTICAL
- mRecyclerView.setLayoutManager(manager);
- mRecyclerView.setItemAnimator(new DefaultItemAnimator());
- }
- private void initData(){
- this.mData = new ArrayList<MyItemBean>();
- for(int i=0;i<20;i++){
- MyItemBean bean = new MyItemBean();
- bean.tv = "Xmy"+i;
- mData.add(bean);
- }
- this.mAdapter = new MyAdapter(mData);
- this.mRecyclerView.setAdapter(mAdapter);
- RecyclerView.ItemDecoration decoration = new MyDecoration(this);
- this.mRecyclerView.addItemDecoration(decoration);
- this.mAdapter.setOnItemClickListener(this);
- this.mAdapter.setOnItemLongClickListener(this);
- }
- @Override
- public void onItemClick(View view, int postion) {
- MyItemBean bean = mData.get(postion);
- if(bean != null){
- Toast.makeText(this, bean.tv, Toast.LENGTH_SHORT).show();
- }
- }
- @Override
- public void onItemLongClick(View view, int postion) {
- MyItemBean bean = mData.get(postion);
- if(bean != null){
- Toast.makeText(this, "LongClick "+bean.tv, Toast.LENGTH_SHORT).show();
- }
- }