Android RecyclerView控件

1. RecyclerView

RecyclerView是5.0版本出现的控件,用来替代传统的ListView,更加强大和灵活。需要添加Design依赖库,并且使用Theme.AppCompat主题。

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <android.support.v7.widget.RecyclerView
        android:id="@+id/recycler_view"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />
</RelativeLayout>

在使用RecyclerView时候,必须指定一个适配器Adapter和一个布局管理器LayoutManager

RecyclerView recyclerView = findViewById(R.id.recycler_view);
// 设置布局方式
RecyclerView.LayoutManager layoutManager = new LinearLayoutManager(this);
recyclerView.setLayoutManager(layoutManager);
recyclerView.setAdapter(new RecyclerViewAdapter(this));

显示如下
在这里插入图片描述

2. 适配器

RecyclerView的适配器必须继承RecyclerView.Adapter,并且封装了ViewHolder的回收复用。

需要实现三个方法

  • VH onCreateViewHolder(ViewGroup parent, int viewType),为每个Item创建一个新的ViewHolder
  • void onBindViewHolder(VH holder, int position),为每个Item赋值
  • int getItemCount(),返回数量

基本适配器实现如下

public final class RecyclerViewAdapter extends RecyclerView.Adapter<RecyclerViewAdapter.ItemViewHolder> {
    private Context mContext;
    private List<String> mContent = new ArrayList<>();

    public RecyclerViewAdapter(Context context) {
        this.mContext = context;

        for (int position = 1; position <= 40; position++) {
            mContent.add("This is " + position + " item");
        }
    }

    @NonNull
    @Override
    public ItemViewHolder onCreateViewHolder(@NonNull ViewGroup viewGroup, int viewType) {
        return new ItemViewHolder(LayoutInflater.from(mContext)
                .inflate(R.layout.list_item_text, viewGroup, false));
    }

    @Override
    public void onBindViewHolder(@NonNull ItemViewHolder viewHolder, int position) {
        viewHolder.textView.setText(mContent.get(position));
    }

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

    class ItemViewHolder extends RecyclerView.ViewHolder {
        private TextView textView;

        public ItemViewHolder(@NonNull View itemView) {
            super(itemView);
            textView = itemView.findViewById(R.id.tv_content);
        }
    }

}

3. 布局管理器

RecyclerView提供了三种布局管理器LayoutManager

  • LinerLayoutManager,以垂直或者水平列表方式展示Item
  • GridLayoutManager,以网格方式展示Item
  • StaggeredGridLayoutManager,以瀑布流方式展示Item

3.1 LinerLayoutManager

默认的LinerLayoutManager布局是垂直方向的,调用setOrientation()方法可以改变方向。

LinearLayoutManager layoutManager = new LinearLayoutManager(this);
layoutManager.setOrientation(RecyclerView.HORIZONTAL);

效果如下
在这里插入图片描述

3.2 GridLayoutManager

GridLayoutManager布局需要指定网格的数量

LinearLayoutManager layoutManager = new GridLayoutManager(this, 4);

效果如下
在这里插入图片描述
还可以通过setSpanSizeLookup()方法来动态控制。

GridLayoutManager gridSpanLayoutManager = new GridLayoutManager(this, 4);
gridSpanLayoutManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
        @Override
        public int getSpanSize(int position) {
            if (position % 5 == 0) {
                return 4;
            }
            return 1;
        }
});

效果如下
在这里插入图片描述

3.3 StaggeredGridLayoutManager

StaggeredGridLayoutManager是瀑布流布局,根据布局的列数来自动适配宽度。

LinearLayoutManager layoutManager = new StaggeredGridLayoutManager(4, RecyclerView.VERTICAL);

效果如下
在这里插入图片描述

4. 添加列表头和列表尾

RecyclerView默认没有提供类似addHeaderView()addFooterView()这样的操作,我们只能自己实现。

首先我们需要准备两个ViewHolder,分别存放列表内容和表头表尾,拥有共同的父类ItemViewHolder

class ItemViewHolder extends RecyclerView.ViewHolder {

    public ItemViewHolder(@NonNull View itemView) {
        super(itemView);
    }

    // 为每个Item赋值
    protected void bindViewHolder(int position) {
    }

}

private class ContentViewHolder extends ItemViewHolder {
    private TextView textView;

    public ContentViewHolder(@NonNull View itemView) {
        super(itemView);
        textView = itemView.findViewById(R.id.tv_content);
    }

    @Override
    protected void bindViewHolder(int position) {
        textView.setText(mContent.get(position));
    }

}

private class HeaderFooterView extends ItemViewHolder {

    public HeaderFooterView(@NonNull View itemView) {
        super(itemView);
    }

}

然后我们在getItemViewType()里面,根据不同的行号返回不同的ViewType。将ViewType分为列表内容和表头表尾,对应不同的ItemViewHolder。调用bindViewHolder()方法更新内容。

public final class RecyclerViewAdapter extends RecyclerView.Adapter<RecyclerViewAdapter.ItemViewHolder> {
    private static final int VIEW_TYPE_HEADER_VIEW      = 100;
    private static final int VIEW_TYPE_CONTENT_VIEW     = 200;
    private static final int VIEW_TYPE_FOOTER_VIEW      = 300;

    private Context mContext;
    private List<String> mContent = new ArrayList<>();
    private List<View> mHeaderViewList = new ArrayList<>();
    private List<View> mFooterViewList = new ArrayList<>();

    public RecyclerViewAdapter(Context context) {
        this.mContext = context;

        for (int position = 1; position <= 20; position++) {
            mContent.add("This is " + position + " item");
        }
    }

    @Override
    public int getItemViewType(int position) {
        if (position < getHeaderViewSize()) {
            // 表头的ViewType为1xx
            return VIEW_TYPE_HEADER_VIEW + position;
        } else if (position >= getHeaderViewSize() + mContent.size()) {
            // 表尾的ViewType为3xx
            return VIEW_TYPE_FOOTER_VIEW + position - getHeaderViewSize() - mContent.size();
        } else {
            // 内容的ViewType为200
            return VIEW_TYPE_CONTENT_VIEW;
        }
    }

    @NonNull
    @Override
    public RecyclerViewAdapter.ItemViewHolder onCreateViewHolder(@NonNull ViewGroup viewGroup, int viewType) {
        if (viewType == VIEW_TYPE_CONTENT_VIEW) {
            // 返回内容ViewHolder
            return new ContentViewHolder(LayoutInflater.from(mContext)
                    .inflate(R.layout.list_item_text, viewGroup, false));
        } else if (viewType < VIEW_TYPE_CONTENT_VIEW) {
            // 返回表头ViewHolder
            return new HeaderFooterView(mHeaderViewList.get(viewType - VIEW_TYPE_HEADER_VIEW));
        } else {
            // 返回表尾ViewHolder
            return new HeaderFooterView(mFooterViewList.get(viewType - VIEW_TYPE_FOOTER_VIEW));
        }
    }

    @Override
    public void onBindViewHolder(@NonNull RecyclerViewAdapter.ItemViewHolder viewHolder, int position) {
        if (position >= getHeaderViewSize()) {
            // 更新Item内容
            viewHolder.bindViewHolder(position - getHeaderViewSize());
        }
    }

    @Override
    public int getItemCount() {
        return getHeaderViewSize() + mContent.size() + getFooterViewSize();
    }

    public void addHeaderView(View headerView) {
        this.mHeaderViewList.add(headerView);
    }

    public void addFooterView(View footerView) {
        this.mFooterViewList.add(footerView);
    }

    private int getHeaderViewSize() {
        return mHeaderViewList.size();
    }

    private int getFooterViewSize() {
        return mFooterViewList.size();
    }

}

效果如下
在这里插入图片描述

相关文章
Android RecyclerView控件
Android RecyclerView刷新和加载
Android RecyclerView实现简易聊天工具

  • 0
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值