优雅实现RecyclerView多种布局

要优雅就要符合 开闭原则,单一职责,当增加新的类型事只能扩展不能修改源代码。


package com.multitypeitem.adapter;

import android.support.v7.widget.RecyclerView;
import android.view.ViewGroup;

import java.util.ArrayList;
import java.util.List;

/**
 * Created by wanjian on 2017/1/12.
 */

public class MultiTypeRecyclerAdapter extends RecyclerView.Adapter {
    public static String TAG = MultiTypeRecyclerAdapter.class.getName();
    private List<ItemViewFactory> mItemViewList = new ArrayList<>();
    private TypeMapPolicy mTypeMapPolicy = new DefaultTypeMapPolicy();

    @Override
    public int getItemViewType(int position) {
        ItemViewFactory item = mItemViewList.get(position);
        return mTypeMapPolicy.toType(item);
    }

    public void setTypeMapPolicy(TypeMapPolicy typeMapPolicy) {
        mTypeMapPolicy = typeMapPolicy;
    }

    public void setData(List<ItemViewFactory> datas) {
        if (datas == null || datas.isEmpty()) {
            return;
        }
        mItemViewList.clear();

        appendData(datas);
    }

    public void appendData(List<ItemViewFactory> datas) {
        if (datas == null || datas.isEmpty()) {
            return;
        }
        for (ItemViewFactory item : datas) {
            if (item == null) {
                continue;
            }
            mItemViewList.add(item);
            item.attachAdapter(this);
        }
        notifyDataSetChanged();
    }

    @Override
    public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        return mTypeMapPolicy.toItemView(viewType).innerCreateVH(parent);
    }

    @Override
    public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
        ItemViewFactory viewFactory = mItemViewList.get(position);
        viewFactory.innerBindVH(holder, position);
    }

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


}



itemviewfactory

package com.multitypeitem.adapter;

import android.content.Context;
import android.support.v7.widget.RecyclerView;
import android.view.ViewGroup;

/**
 * Created by wanjian on 2017/1/23.
 */

public abstract class ItemViewFactory<D, H extends RecyclerView.ViewHolder> {
    D mData;
    private RecyclerView.Adapter mAdapter;
    private Context mContext;
    private int mPosition;

    public ItemViewFactory(Context context, D data) {
        this.mData = data;
        this.mContext = context;
    }

    //不加修饰符,只供内部使用,禁止外部调用
    H innerCreateVH(ViewGroup parent) {
        return onCreateViewHolder(mContext, parent);
    }

    void innerBindVH(RecyclerView.ViewHolder holder, int position) {
        mPosition = position;
        onBindViewHolder(mContext, ((H) holder), mData);
    }

    void attachAdapter(RecyclerView.Adapter adapter) {
        mAdapter = adapter;
    }

    protected void refresh(D data) {
        resetData(data);
        if (mAdapter != null) {
            mAdapter.notifyItemChanged(mPosition);
        }
    }

    protected void resetData(D data) {
        this.mData = data;
    }

    public abstract H onCreateViewHolder(Context context, ViewGroup parent);

    public abstract void onBindViewHolder(Context context, H holder, D data);
}

package com.multitypeitem.adapter;

/**
 * Created by wanjian on 2017/1/23.
 */

public interface TypeMapPolicy {
    int toType(ItemViewFactory viewFactory);

    ItemViewFactory toItemView(int type);
}

package com.multitypeitem.adapter;


import java.util.ArrayList;
import java.util.List;

/**
 * Created by wanjian on 2017/1/23.
 */

public class DefaultTypeMapPolicy implements TypeMapPolicy {

    private List<ItemViewFactory> mTypesMapping = new ArrayList<>();

    @Override
    public int toType(ItemViewFactory item) {
        Class<?> clz = item.getClass();

        for (int i = mTypesMapping.size() - 1; i > -1; i--) {
            ItemViewFactory itemViewFactory = mTypesMapping.get(i);
            if (itemViewFactory.getClass() == clz) {
                return i;
            }
        }
        mTypesMapping.add(item);
        return mTypesMapping.size() - 1;
    }

    @Override
    public ItemViewFactory toItemView(int viewType) {
        return mTypesMapping.get(viewType);
    }
}

各种类型view
package com.multitypeitem;

import android.content.Context;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

import com.multitypeitem.adapter.ItemViewFactory;

import static com.multitypeitem.adapter.MultiTypeRecyclerAdapter.TAG;

/**
 * Created by wanjian on 2017/1/12.
 */

public class ItemView1 extends ItemViewFactory<Integer, ItemView1.Item1VH> {

    public ItemView1(Context context, Integer data) {
        super(context, data);
    }

    @Override
    public Item1VH onCreateViewHolder(Context context, ViewGroup parent) {
        Log.d(TAG, "onCreateViewHolder:..... 1");
        return new Item1VH(LayoutInflater.from(context).inflate(R.layout.item1, parent, false));
    }

    @Override
    public void onBindViewHolder(Context context, Item1VH holder, final Integer data) {
        holder.mTextView.setText(data + "");
        Log.w(TAG, "onBindViewHolder: " + data);
        holder.mTextView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                refresh(data + 10000);
            }
        });
    }


    static class Item1VH extends RecyclerView.ViewHolder {

        TextView mTextView;

        public Item1VH(View itemView) {
            super(itemView);
            mTextView = (TextView) itemView.findViewById(R.id.tv);
        }
    }
}

package com.multitypeitem;

import android.content.Context;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;


import com.multitypeitem.adapter.ItemViewFactory;
import com.multitypeitem.adapter.MultiTypeRecyclerAdapter;

/**
 * Created by wanjian on 2017/1/12.
 */

public class ItemView2 extends ItemViewFactory<String, ItemView2.Item2VH> {

    public ItemView2(Context context, String data) {
        super(context, data);
    }

    @Override
    public Item2VH onCreateViewHolder(Context context, ViewGroup parent) {
        Log.d(MultiTypeRecyclerAdapter.TAG, "onCreateViewHolder:----------- 2");
        return new ItemView2.Item2VH(LayoutInflater.from(context).inflate(R.layout.item2, parent, false));
    }

    @Override
    public void onBindViewHolder(Context context, Item2VH holder, final String data) {
        holder.mTextView.setText(data);

        holder.mTextView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                refresh("change..." + data);
            }
        });
    }

    static class Item2VH extends RecyclerView.ViewHolder {
        TextView mTextView;

        public Item2VH(View itemView) {
            super(itemView);
            mTextView = (TextView) itemView.findViewById(R.id.tv);
        }
    }
}

其他view类似


main布局

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:orientation="vertical"
    tools:context="com.multitypeitem.MainActivity">

    <Button
        android:id="@+id/but"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="ok"/>

    <android.support.v7.widget.RecyclerView
        android:id="@+id/recycler"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        >

    </android.support.v7.widget.RecyclerView>

</LinearLayout>


item布局

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
              android:orientation="vertical"
              android:layout_width="match_parent"
              android:layout_height="wrap_content"
              android:padding="5dp">

    <TextView
        android:id="@+id/tv"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="#e1e1e1"
        android:textSize="20sp"
        android:textColor="#000"
        android:padding="10dp"/>
</LinearLayout>

使用

package com.multitypeitem;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.view.View;

import java.util.ArrayList;
import java.util.Date;
import java.util.List;

import com.multitypeitem.adapter.ItemViewFactory;
import com.multitypeitem.adapter.MultiTypeRecyclerAdapter;

public class MainActivity extends AppCompatActivity {

    RecyclerView mRecyclerView;

    MultiTypeRecyclerAdapter mAdapter;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);


        mRecyclerView = (RecyclerView) findViewById(R.id.recycler);

        RecyclerView.LayoutManager layoutManager = new LinearLayoutManager(this);

        mRecyclerView.setLayoutManager(layoutManager);

        RecyclerView.RecycledViewPool recycledViewPool = new RecyclerView.RecycledViewPool();
        //设置每种布局最大缓存数量,可以不设置
        recycledViewPool.setMaxRecycledViews(0, 15);
        recycledViewPool.setMaxRecycledViews(1, 15);
        recycledViewPool.setMaxRecycledViews(2, 15);
        recycledViewPool.setMaxRecycledViews(3, 15);
        recycledViewPool.setMaxRecycledViews(4, 15);
        mRecyclerView.setRecycledViewPool(recycledViewPool);

        mAdapter = new MultiTypeRecyclerAdapter();
        mRecyclerView.setAdapter(mAdapter);

        List<ItemViewFactory> data = getData();
        mAdapter.setData(data);

        findViewById(R.id.but).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                mAdapter.setData(getData2());
            }
        });
    }


    public List<ItemViewFactory> getData() {
        List<ItemViewFactory> data = new ArrayList<>();

        for (int i = 0; i < 10; i++) {
            data.add(new ItemView1(this, i));
        }

        for (int i = 0; i < 1; i++) {
            data.add(new ItemView2(this, "view2---------- " + i));
        }

//        for (int i = 0; i < 10; i++) {
//            data.add(new ItemView3(this, new Date(2017-1900, 1, 1 + i)));
//        }
//
        for (int i = 0; i < 8; i++) {
            data.add(new ItemView1(this, 800 + i));
        }
//
        for (int i = 0; i < 2; i++) {
            data.add(new ItemView4(this, i));
        }
//
//        for (int i = 0; i < 50; i++) {
//            data.add(new ItemView5(this, "view5 " + i));
//        }

        return data;
    }

    public List<ItemViewFactory> getData2() {
        List<ItemViewFactory> data = new ArrayList<>();

        for (int i = 0; i < 3; i++) {
            data.add(new ItemView3(this, new Date(2017 - 1900, 1, 1 + i)));
        }

        for (int i = 0; i < 3; i++) {
            data.add(new ItemView2(this, "view2---------- " + i));
        }


//        for (int i = 0; i < 5; i++) {
//            data.add(new ItemView1(this, i));
//        }
//


        for (int i = 0; i < 2; i++) {
            data.add(new ItemView4(this, i));
        }

        for (int i = 0; i < 5; i++) {
            data.add(new ItemView5(this, "view5 " + i));
        }


        for (int i = 0; i < 8; i++) {
            data.add(new ItemView1(this, 800 + i));
        }

        return data;
    }
}

效果




git

https://github.com/android-notes/multityperecyclerview



实现 RecyclerView 多种布局,可以通过重写 getItemViewType 方法,根据不同的数据类型返回不同的布局类型,然后在 onCreateViewHolder 方法中根据 viewType 加载不同的布局文件,最后在 onBindViewHolder 方法中根据数据类型绑定数据到不同的 ViewHolder 上。 代码示例: ``` public class MyAdapter extends RecyclerView.Adapter<RecyclerView.ViewHolder> { private List<Object> mDataList; // 枚举不同的布局类型 private static final int TYPE_ONE = 1; private static final int TYPE_TWO = 2; public MyAdapter(List<Object> dataList) { mDataList = dataList; } @Override public int getItemViewType(int position) { if (mDataList.get(position) instanceof TypeOneBean) { return TYPE_ONE; } else if (mDataList.get(position) instanceof TypeTwoBean) { return TYPE_TWO; } return super.getItemViewType(position); } @Override public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { switch (viewType) { case TYPE_ONE: View view1 = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_type_one, parent, false); return new TypeOneViewHolder(view1); case TYPE_TWO: View view2 = LayoutInflater.from(parent.getContext()).inflate(R.layout.item_type_two, parent, false); return new TypeTwoViewHolder(view2); default: throw new IllegalArgumentException("invalid viewType"); } } @Override public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) { Object data = mDataList.get(position); if (holder instanceof TypeOneViewHolder) { // 绑定 TypeOneViewHolder 的数据 ((TypeOneViewHolder) holder).bindData((TypeOneBean) data); } else if (holder instanceof TypeTwoViewHolder) { // 绑定 TypeTwoViewHolder 的数据 ((TypeTwoViewHolder) holder).bindData((TypeTwoBean) data); } } @Override public int getItemCount() { return mDataList.size(); } static class TypeOneViewHolder extends RecyclerView.ViewHolder { public TypeOneViewHolder(View itemView) { super(itemView); // 初始化 TypeOneViewHolder 的控件 } public void bindData(TypeOneBean data) { // 绑定 TypeOneViewHolder 的数据 } } static class TypeTwoViewHolder extends RecyclerView.ViewHolder { public TypeTwoViewHolder(View itemView) { super(itemView); // 初始化 TypeTwoViewHolder 的控件 } public void bindData(TypeTwoBean data) { // 绑定 TypeTwoViewHolder 的数据 } } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值