万能RecyclerViewAdapter

本文介绍了一个简单的万能RecyclerView Adapter实现方案,它简化了适配器的编写过程,使得开发者能够快速地为RecyclerView填充数据并设置点击事件。

很早之前,看到zhy写的万能ListViewAdapter,并且写了篇博客万能Adapter

现在RecyclerView已经逐渐替代了ListView。而RecyclerView的Adapter写起来又比ListView的Adapter要复杂一些,于是我边仿照之前,写了一个RecyclerView的万能Adapter。


使用方法:


public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        RecyclerView recyclerView = (RecyclerView) findViewById(R.id.recyclerView);
        LinearLayoutManager layoutManager = new LinearLayoutManager(this);
        layoutManager.setOrientation(OrientationHelper. VERTICAL);
        recyclerView.setLayoutManager(layoutManager);
        List<Integer> list = new ArrayList<>();
        for(int i = 0; i < 100; i++){
            list.add(i);
        }
        recyclerView.setAdapter(new CommonRecyclerViewAdapter<Integer>(this, list, R.layout.item) {
            @Override
            public void convert(RecyclerViewHolder holder, Integer item, int position) {
                holder.setText(R.id.text, item + "");
            }
        });
    }
}

这样子是不是简单了很多。。。

下面直接贴出代码,一个也就两个类:

CommonRecyclerViewAdapter:

public abstract class CommonRecyclerViewAdapter<T> extends RecyclerView.Adapter<RecyclerViewHolder>{

   public interface OnItemClickListener {
      void onItemClick (View view, int position);
   }

   public interface OnItemLongClickListener {
      void onItemLongClick (View view, int position);
   }

   private LayoutInflater mInflater;
   protected Context mContext;
   private List<T> mData;
   private final int mItemLayoutId;
   private OnItemClickListener mOnItemClickListener;
   private OnItemLongClickListener mOnItemLongClickListener;

   public CommonRecyclerViewAdapter(Context context, List<T> data, int itemLayoutId) {
      this.mContext = context;
      this.mInflater = LayoutInflater.from(mContext);
      this.mItemLayoutId = itemLayoutId;
      this.mData = data;
   }

   public void setOnItemClickListener(OnItemClickListener onItemClickListener){
      this.mOnItemClickListener = onItemClickListener;
   }

   public void setmOnItemLongClickListener(OnItemLongClickListener onItemLongClickListener) {
      this.mOnItemLongClickListener = onItemLongClickListener;
   }

   @Override
   public RecyclerViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
      return new RecyclerViewHolder(mInflater.inflate(mItemLayoutId, parent, false), mOnItemClickListener,
            mOnItemLongClickListener);
   }

   @Override
   public void onBindViewHolder(RecyclerViewHolder holder, int position) {
      convert(holder, mData.get(position), position);
   }

   @Override
   public int getItemCount(){
      return mData.size();
   }

   public abstract void convert(RecyclerViewHolder holder, T item, int position);
}

RecyclerViewHolder:

public class RecyclerViewHolder extends RecyclerView.ViewHolder implements View.OnClickListener,View.OnLongClickListener{

   private final SparseArray<View> mViews;
   private View mItemView;

   private CommonRecyclerViewAdapter.OnItemClickListener mOnItemClickListener;
   private CommonRecyclerViewAdapter.OnItemLongClickListener mOnItemLongClickListener;

   public RecyclerViewHolder(View itemView, CommonRecyclerViewAdapter.OnItemClickListener onItemClickListener,
                             CommonRecyclerViewAdapter.OnItemLongClickListener onItemLongClickListener){
      super(itemView);
      this.mItemView = itemView;
      this.mOnItemClickListener = onItemClickListener;
      this.mOnItemLongClickListener = onItemLongClickListener;
      this.mViews = new SparseArray<>();
   }

   @Override
   public void onClick(View v) {
      if(mOnItemClickListener != null){
         mOnItemClickListener.onItemClick(v,getLayoutPosition());
      }
   }

   @Override
   public boolean onLongClick(View v) {
      if(mOnItemLongClickListener != null){
         mOnItemLongClickListener.onItemLongClick(v,getLayoutPosition());
      }
      return true;
   }

   private <T extends View> T getView(int id){
      View item = mViews.get(id);
      if(item == null){
         item = mItemView.findViewById(id);
         mViews.put(id, item);
      }
      return (T) item;
   }

   public RecyclerViewHolder setText(int id, CharSequence text){
      TextView textView = getView(id);
      textView.setText(text);
      return this;
   }

   public RecyclerViewHolder setTextColor(int id, int color){
      TextView textView = getView(id);
      textView.setTextColor(color);
      return this;
   }

   public RecyclerViewHolder setBackground(int id, int drawable){
      View view = getView(id);
      view.setBackground(ContextCompat.getDrawable(mItemView.getContext(), drawable));
      return this;
   }
}

要使用的人直接复制这两个类即可。

GItHub下载地址:https://github.com/xiaoqiAndroid/CommonRecyclerViewAdapter

通过封装BaseAdapter和RecyclerView.Adapter得到的通用的,简易的Adapter。项目地址:https://github.com/tianzhijiexian/CommonAdapter 效果图:已解决的问题 提升item的独立性,完美支持item被多处复用 item会根据type来做自动复用 支持多种类型的item 一个item仅会调用一次setViews(),避免重复建立监听器 一个item仅会触发一次绑定视图的操作,提示效率 支持dataBinding和其他第三方注入框架 提供了getView()方法来简化findViewById 支持通过item的构造方法来传入Activity对象 支持通过item的构造方法来传入item中事件的回调 提供了getConvertedData(data, type)方法来对item传入的数据做转换,方便拆包和提升item的复用性 支持viewpager的正常加载模式和懒加载 支持快速将listview的适配器切换为recyclerView适配器 viewpager的notifyDataSetChanged可以正常更新界面 支持recyclerView的添加头部和底部 支持适配器的数据自动绑定,只用操作数据便可,adapter会自动notify界面零、重要接口adapter的item必须实现此接口,接口源码如下:public interface AdapterItem<T> {     /**      * @return item布局文件的layoutId      */     @LayoutRes     int getLayoutResId();     /**      * 初始化views      */     void bindViews(final View root);     /**      * 设置view的参数      */     void setViews();     /**      * 根据数据来设置item的内部views      *      * @param model    数据list内部的model      * @param position 当前adapter调用item的位置      */     void handleData(T model, int position); }例子:public class TextItem implements AdapterItem<DemoModel> {     @Override     public int getLayoutResId() {         return R.layout.demo_item_text;     }     TextView textView;     @Override     public void bindViews(View root) {         textView = (TextView) root.findViewById(R.id.textView);     }     @Override     public void setViews() { }     @Override     public void handleData(DemoModel model, int position) {         textView.setText(model.content);     } }一、ListView GridView的通用适配器——CommonAdapter只需继承CommonAdapter便可实现适配器:listView.setAdapter(new CommonAdapter<DemoModel>(data, 1) {     public AdapterItem<DemoModel> createItem(Object type) {         return new TextItem();     } });二、RecyclerView的通用适配器——CommonRcvAdapter通过继承CommonRcvAdapter来实现适配器:mAdapter = new CommonRcvAdapter<DemoModel>(data) {  public AdapterItem createItem(Object type) {         return new TextItem();   } };三、ViewPager的通用适配器——CommonPagerAdapter通过继承CommonPagerAdapter来实现适配器:viewPager.setAdapter(new CommonPagerAdapter<DemoModel>() {     public AdapterItem createItem(Object type) {         return new TextItem();     } });设计思路1. Adapter如果用adapter常规写法,你会发现代码量很大,可读性低。如果adapter中有多个类型的Item,我们还得在getView()中写很多if-else语句,很乱。 而现在我让adapter的代码量减少到一个8行的内部类,如果你需要更换item只需要动一行代码,真正实现了可插拔化。最关键的是item现在作为了一个独立的对象,可以方便的进行复用。2. AdapterItem和原来方式最为不同的一点就是我把adapter的item作为了一个实体,这种方式借鉴了RecyclerView中ViewHolder的设计。把item作为实体的好处有很多,比如复用啊,封装啊,其余的就不细说了。3. 分层在使用过程中,我发现如果adapter放在view层,那就会影响到view层的独立性。此外adapter中经常有很多数据处理的操作,比如通过type选择item,数据的拆包、转换等操作。于是我还是推荐把adapter放在mvp的p层,或者是mvvm的m层。通过在实际的项目中使用来看,放在m或p层的效果较好,view的复用也比较好做。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值