RecyclerView Adapter 简易封装 支持 添加头、尾 布局支持多布局类型

RecyclerView Adapter 简易封装 支持 添加头、尾 布局支持多布局类型

 

import android.content.Context;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

import androidx.annotation.NonNull;
import androidx.annotation.Nullable;
import androidx.recyclerview.widget.RecyclerView;

import com.dapeng.base_lib.interfaces.IMulTiType;

import java.util.List;

/**
 * RecyclerView Adapter 基本的封装 添加头尾可以再优化,这边只是图简单举个例子,没有复杂需求可以直接用 原理一样的 可根据需要自由修改
 * <p>
 * Created by dp.
 * <p>
 * Date: 2020-12-18
 * <p>
 * @param <T> item 数据类型 model 数据Bean
 * @param <V> ViewHolder 如果无需自定义 可直接传入{@link BaseRecyclerViewHolder}
 * <p>
 * Summary: recyclerView 通用 适配器,
 *          同时  1、支持多布局
 *                2、支持添加多个头尾布局
 * <p>
 * Use: 1、一般的列表 初始化{@link #BaseRecyclerViewAdapter(Context, List )} 复写 {@link #getItemLayResId()} 即可
 *      2、多种item类型 初始化{@link #BaseRecyclerViewAdapter(Context, List,IMulTiType)} 外部实现IMulTiType 接口传入布局类型 即可
 *      3、添加头尾布局 {@link #addHeadView(View, Object)} {@link #addFootView(View, Object)} object为设置一个tag 区分添加的view
 */
public abstract class BaseRecyclerViewAdapter<T, V extends BaseRecyclerViewHolder> extends RecyclerView.Adapter<V> {

    /**
     * 全局维持一个数据 list
     */
    protected final List<T> list;
    /**
     * 上下文 Context
     */
    protected final Context context;
    /**
     * 多布局类型实现接口 返回对应的布局类型
     */
    private final IMulTiType<T> iMulTiType;
    /**
     * headViews缓存
     */
    private SparseArray<View> headViews;
    /**
     * footViews缓存
     */
    private SparseArray<View> footViews;
    /**
     * 头布局类型
     */
    public static final int TYPE_HEADER = -1;
    /**
     * 尾布局类型
     */
    public static final int TYPE_FOOTER = -2;

    /**
     * SparseArray tag 按照添加顺序 0 ↑
     */
    private int headViewTag = 0;
    private int footViewTag = 0;
    private int headIndex = 0;
    private int footIndex = 0;

    /**
     * 单类型条目布局 初始化方法
     */
    public BaseRecyclerViewAdapter(Context context, List<T> list) {
        this(context, list, null);
    }

    /**
     * 多类型条目布局 初始化方法
     */
    public BaseRecyclerViewAdapter(Context context, List<T> list, IMulTiType<T> iMulTiType) {
        this.list = list;
        this.context = context;
        this.iMulTiType = iMulTiType;
    }

    /**
     * item 数据
     */
    public T getItem(int position) {
        return list.get(position);
    }

    @Override
    public long getItemId(int position) {
        return super.getItemId(position);
    }

    /**
     * 创建 布局
     */
    @SuppressWarnings("unchecked")
    @NonNull
    @Override
    public V onCreateViewHolder(@NonNull ViewGroup parent, int viewType) {
        //------------------------- 头尾布局(如果有)----------------------
        if (getHeadCount() > 0 && viewType == TYPE_HEADER) {
            return (V) new BaseRecyclerViewHolder(getHeadView(headIndex++));
        } else if (getFootCount() > 0 && viewType == TYPE_FOOTER) {
            return (V) new BaseRecyclerViewHolder(getFootView(footIndex++));
        }
        //------------------------------------------------------------------

        // ------------------------- 多布局(如果设置了)--------------------
        if (iMulTiType != null) {
            if (viewType == 0) {
                return ((V) new BaseRecyclerViewHolder(getErrorView()));
            }
            View view = LayoutInflater.from(context).inflate(viewType, parent, false);
            return (V) new BaseRecyclerViewHolder(view);
        }
        //------------------------------------------------------------------


        //------------------------------一般的列表 item----------------------
        if (getItemLayResId() == 0) {
            return ((V) new BaseRecyclerViewHolder(getErrorView()));
        }
        View view = LayoutInflater.from(context).inflate(getItemLayResId(), parent, false);
        return ((V) new BaseRecyclerViewHolder(view));
        //-------------------------------------------------------------------
    }

    /**
     * 容错 防止view找不到 崩溃
     */
    private View getErrorView() {
        return new View(context);
    }

    /**
     * 数据绑定
     */
    @Override
    public void onBindViewHolder(@NonNull final V holder, final int position) {
        if (holder.getItemViewType() == TYPE_HEADER) {
            onBindHeadViewsItemData(holder, position, holder.itemView.getTag());
            return;
        } else if (holder.getItemViewType() == TYPE_FOOTER) {
            onBindFootViewsItemData(holder, position, holder.itemView.getTag());
            return;
        }
        final int dataRealPosition = position - getHeadCount();
        holder.itemView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                onNormalItemClick(holder, getItem(dataRealPosition), dataRealPosition);
            }
        });
        onBindNormalItemData(holder, getItem(dataRealPosition), position);
    }

    /**
     * item 点击事件
     */
    protected void onNormalItemClick(V holder, T item, int position) {

    }

    @Override
    public int getItemCount() {
        return getNormalItemCount() + getHeadCount() + getFootCount();
    }

    @Override
    public int getItemViewType(int position) {
        if (position < getHeadCount())
            return TYPE_HEADER;
        int realPosition = position - getHeadCount();
        if (realPosition < list.size()) {
            if (iMulTiType != null) {
                return iMulTiType.getLayoutType(getItem(realPosition), realPosition);
            }
            return super.getItemViewType(position);
        }
        return TYPE_FOOTER;
    }

    /**
     * 单条目布局 必须复写
     */
    protected int getItemLayResId() {
        return 0;
    }

    /**
     * 除了头部 尾部布局 其他item 数据绑定
     */
    protected abstract void onBindNormalItemData(V holder, T itemData, int position);

    /**
     * 头部数据绑定  按需要复写
     */
    protected void onBindHeadViewsItemData(V holder, int position, Object tag) {
    }

    /**
     * 尾部数据绑定  按需要复写
     */
    protected void onBindFootViewsItemData(V holder, int position, Object tag) {
    }


    /**
     * 添加头布局 可添加 多个 以 tag做标记 区分
     */
    public void addHeadView(@NonNull View view, Object viewTag) {
        if (view == null)
            return;
        if (headViews == null) {
            headViews = new SparseArray<>();
        }
        view.setTag(viewTag);
        headViews.put(headViewTag++, view);
        notifyItemInserted(getHeadCount() - 1);
    }


    /**
     * 添加尾布局 可添加多个 以 tag 做标记 区分
     */
    public void addFootView(@NonNull View view, Object viewTag) {
        if (view == null)
            return;
        if (footViews == null) {
            footViews = new SparseArray<>();
        }
        view.setTag(viewTag);
        footViews.put(footViewTag++, view);
        notifyItemInserted(getItemCount());
    }

    @Nullable
    private View getHeadView(int headViewTag) {
        if (headViews == null) {
            return null;
        }
        return headViews.get(headViewTag, getErrorView());
    }

    @Nullable
    private View getFootView(int footViewTag) {
        if (footViews == null) {
            return null;
        }
        return footViews.get(footViewTag, getErrorView());
    }

    public int getHeadCount() {
        return headViews != null ? headViews.size() : 0;
    }

    public int getFootCount() {
        return footViews != null ? footViews.size() : 0;
    }

    public int getNormalItemCount() {
        return list != null ? list.size() : 0;
    }

}
import android.graphics.drawable.Drawable;
import android.util.SparseArray;
import android.view.View;
import android.widget.ImageView;
import android.widget.TextView;

import androidx.annotation.ColorInt;
import androidx.annotation.DrawableRes;
import androidx.annotation.NonNull;
import androidx.recyclerview.widget.RecyclerView;

/**
 * 通用 RecyclerView ViewHolder
 *
 * 链式调用
 */
public class BaseRecyclerViewHolder extends RecyclerView.ViewHolder {

    public static final int COLOR_DEFAULT = -1;
    public static final int SIZE_DEFAULT = -1;
    private final SparseArray<View> views;

    public BaseRecyclerViewHolder(@NonNull View itemView) {
        super(itemView);
        views = new SparseArray<>();
    }


    @SuppressWarnings("unchecked")
    public <T extends View> T getView(int viewId) {
        View view = views.get(viewId);
        if (view == null) {
            view = itemView.findViewById(viewId);
            views.put(viewId, view);
        }
        return (T) view;
    }

    /**
     * 设置 tetView 文字
     */
    public BaseRecyclerViewHolder setText(int viewId, CharSequence text) {
        return setText(viewId, text, COLOR_DEFAULT, SIZE_DEFAULT);
    }

    /**
     * 设置 tetView 文字 + 颜色
     */
    public BaseRecyclerViewHolder setText(int viewId, CharSequence text, @ColorInt int textColor) {
        return setText(viewId, text, textColor, SIZE_DEFAULT);
    }

    /**
     * 设置 tetView 文字 + 大小
     */
    public BaseRecyclerViewHolder setText(int viewId, CharSequence text, float textSize) {
        return setText(viewId, text, COLOR_DEFAULT, textSize);
    }

    /**
     * 设置 tetView 文字 + 颜色 + 大小
     */
    public BaseRecyclerViewHolder setText(int viewId, CharSequence text, @ColorInt int textColor, float textSize) {
        TextView textView = getView(viewId);
        if (textView != null) {
            textView.setText(text);
            if (COLOR_DEFAULT != textColor) {
                textView.setTextColor(textColor);
            }
            if (SIZE_DEFAULT != textSize) {
                textView.setTextSize(textSize);
            }
        }
        return this;
    }

    /**
     * view 设置背景
     */
    public BaseRecyclerViewHolder setViewBackground(int viewId, Drawable background) {
        View view = getView(viewId);
        if (view != null) {
            view.setBackground(background);
        }
        return this;
    }

    /**
     * 点击事件
     */
    public BaseRecyclerViewHolder setViewClickListener(int viewId, View.OnClickListener listener) {
        View view = getView(viewId);
        if (view != null && listener != null) {
            view.setOnClickListener(listener);
        }
        return this;
    }

    /**
     * 长按事件
     */
    public BaseRecyclerViewHolder setViewLongClickListener(int viewId, View.OnLongClickListener longClickListener) {
        View view = getView(viewId);
        if (view != null && longClickListener != null) {
            view.setOnLongClickListener(longClickListener);
        }
        return this;
    }

    /**
     * 设置图片资源
     */
    public BaseRecyclerViewHolder setImageRes(int viewId, @DrawableRes int resId) {
        ImageView imageView = getView(viewId);
        if (imageView != null) {
            imageView.setImageResource(resId);
        }
        return this;
    }

    public BaseRecyclerViewHolder setVisibility(int viewId, int visibility) {
        View view = getView(viewId);
        if (view != null) {
            view.setVisibility(visibility);
        }
        return this;
    }

    public BaseRecyclerViewHolder setViewGone(int viewId, boolean isGone) {
        View view = getView(viewId);
        if (view != null) {
            view.setVisibility(isGone ? View.GONE : View.VISIBLE);
        }
        return this;
    }

    public BaseRecyclerViewHolder setViewVisible(int viewId, boolean isVisible) {
        View view = getView(viewId);
        if (view != null) {
            view.setVisibility(isVisible ? View.VISIBLE : View.INVISIBLE);
        }
        return this;
    }
}

 

/**
 * Created by ldp.
 * <p>
 * Date: 2020-12-18
 * <p>
 * Summary: 多类型布局实现接口
 */
public interface IMulTiType<T> {

    /**
     * @param itemData item 数据bean
     * @param position 索引,位置
     * @return 根据数据 或者索引 返回需要展示的 layout id 形如 R.layout.item_xxx
     */
    int getLayoutType(T itemData,int position);
}

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,以下是一个基本的多布局 RecyclerView Adapter 的示例代码。假设我们有两种不同的布局类型类型1和类型2。 首先,我们需要创建一个 ViewHolder 的基类,它将存储我们的布局中的所有视图。 ```java public abstract class BaseViewHolder extends RecyclerView.ViewHolder { public BaseViewHolder(View itemView) { super(itemView); } public abstract void bindData(Object data); } ``` 然后,我们需要创建两个具体的 ViewHolder 类,分别对应我们的两种不同的布局类型。 ViewHolder 类型1: ```java public class Type1ViewHolder extends BaseViewHolder { private TextView titleTextView; private TextView subtitleTextView; public Type1ViewHolder(View itemView) { super(itemView); titleTextView = itemView.findViewById(R.id.titleTextView); subtitleTextView = itemView.findViewById(R.id.subtitleTextView); } public void bindData(Object data) { Type1Data type1Data = (Type1Data) data; titleTextView.setText(type1Data.getTitle()); subtitleTextView.setText(type1Data.getSubtitle()); } } ``` ViewHolder 类型2: ```java public class Type2ViewHolder extends BaseViewHolder { private ImageView imageView; public Type2ViewHolder(View itemView) { super(itemView); imageView = itemView.findViewById(R.id.imageView); } public void bindData(Object data) { Type2Data type2Data = (Type2Data) data; Picasso.get().load(type2Data.getImageUrl()).into(imageView); } } ``` 接下来,我们需要创建一个 RecyclerView.Adapter 类,并在其中实现 onCreateViewHolder、getItemViewType 和 onBindViewHolder 方法。 ```java public class MultiLayoutAdapter extends RecyclerView.Adapter<BaseViewHolder> { private List<Object> dataList; public MultiLayoutAdapter(List<Object> dataList) { this.dataList = dataList; } @Override public int getItemViewType(int position) { // 根据数据类型返回对应的布局类型 Object data = dataList.get(position); if (data instanceof Type1Data) { return 0; } else if (data instanceof Type2Data) { return 1; } return -1; } @Override public BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { // 根据布局类型返回对应的 ViewHolder if (viewType == 0) { View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.layout_type1, parent, false); return new Type1ViewHolder(view); } else if (viewType == 1) { View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.layout_type2, parent, false); return new Type2ViewHolder(view); } return null; } @Override public void onBindViewHolder(BaseViewHolder holder, int position) { // 绑定数据 holder.bindData(dataList.get(position)); } @Override public int getItemCount() { return dataList.size(); } } ``` 最后,在您的 Activity 或 Fragment 中,您可以像这样使用这个适配器: ```java List<Object> dataList = new ArrayList<>(); dataList.add(new Type1Data("Title 1", "Subtitle 1")); dataList.add(new Type2Data("https://example.com/image.jpg")); MultiLayoutAdapter adapter = new MultiLayoutAdapter(dataList); recyclerView.setAdapter(adapter); ``` 这样,您就可以使用 RecyclerView.Adapter 和多个 ViewHolder 来实现一个多布局RecyclerView Adapter
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值