Android之RecyclerView

来源:互联网 发布:js给div设置高度 编辑:程序博客网 时间:2024/06/18 17:37

本讲内容:RecyclerView控件


一、RecyclerView控件概述

该控件用于在有限的窗口中展示大量数据集,例如:ListView、GridView。

1、可以通过布局管理器LayoutManager控制其显示的方式
2、可以通过ItemDecoration控制Item间的间隔
3、可以通过ItemAnimator控制Item增删的动画


示例一:

  

下面是res/layout/activity_main.xml 布局文件:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"    android:layout_height="match_parent" >    <android.support.v7.widget.RecyclerView        android:id="@+id/id_recyclerview"        android:layout_width="match_parent"        android:layout_height="match_parent" /></RelativeLayout>


下面是res/layout/item_textview.xml 布局文件:

<?xml version="1.0" encoding="utf-8"?><FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:layout_width="match_parent"    android:background="#44ff0000"    android:layout_height="72dp" >    <TextView        android:id="@+id/id_item"        android:layout_width="match_parent"        android:layout_height="match_parent"        android:gravity="center" /></FrameLayout>


下面是SimpleAdapter.java文件:

public class SimpleAdapter extends RecyclerView.Adapter<MyViewHolder> {private LayoutInflater mInflater;private Context mContext;private List<String> mDatas;public SimpleAdapter(Context context, List<String> datas) {this.mContext = context;this.mDatas = datas;mInflater = LayoutInflater.from(context);// 加载布局管理器}public int getItemCount() {return mDatas.size();}/** * 绑定ViewHolder */public void onBindViewHolder(MyViewHolder holder, int position) {holder.tv.setText(mDatas.get(position));}/** * 创建ViewHolder */public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {//将xml布局转换为view对象View view = mInflater.inflate(R.layout.item_textview, parent, false);MyViewHolder viewHolder = new MyViewHolder(view);return viewHolder;}class MyViewHolder extends ViewHolder {TextView tv;public MyViewHolder(View view) {super(view);tv = (TextView) view.findViewById(R.id.id_item);}}}


下面是DividerItemDecoration.java文件:(如果要设置分隔线则要用到)

public class DividerItemDecoration extends RecyclerView.ItemDecoration {private static final int[] ATTRS = new int[] { android.R.attr.listDivider };public static final int HORIZONTAL_LIST = LinearLayoutManager.HORIZONTAL;public static final int VERTICAL_LIST = LinearLayoutManager.VERTICAL;private Drawable mDivider;private int mOrientation;public DividerItemDecoration(Context context, int orientation) {final TypedArray a = context.obtainStyledAttributes(ATTRS);mDivider = a.getDrawable(0);a.recycle();setOrientation(orientation);}public void setOrientation(int orientation) {if (orientation != HORIZONTAL_LIST && orientation != VERTICAL_LIST) {throw new IllegalArgumentException("invalid orientation");}mOrientation = orientation;}@Overridepublic void onDraw(Canvas c, RecyclerView parent) {Log.v("recyclerview - itemdecoration", "onDraw()");if (mOrientation == VERTICAL_LIST) {drawVertical(c, parent);} else {drawHorizontal(c, parent);}}public void drawVertical(Canvas c, RecyclerView parent) {final int left = parent.getPaddingLeft();final int right = parent.getWidth() - parent.getPaddingRight();final int childCount = parent.getChildCount();for (int i = 0; i < childCount; i++) {final View child = parent.getChildAt(i);android.support.v7.widget.RecyclerView v = new android.support.v7.widget.RecyclerView(parent.getContext());final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child.getLayoutParams();final int top = child.getBottom() + params.bottomMargin;final int bottom = top + mDivider.getIntrinsicHeight();mDivider.setBounds(left, top, right, bottom);mDivider.draw(c);}}public void drawHorizontal(Canvas c, RecyclerView parent) {final int top = parent.getPaddingTop();final int bottom = parent.getHeight() - parent.getPaddingBottom();final int childCount = parent.getChildCount();for (int i = 0; i < childCount; i++) {final View child = parent.getChildAt(i);final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child.getLayoutParams();final int left = child.getRight() + params.rightMargin;final int right = left + mDivider.getIntrinsicHeight();mDivider.setBounds(left, top, right, bottom);mDivider.draw(c);}}@Overridepublic void getItemOffsets(Rect outRect, int itemPosition,RecyclerView parent) {if (mOrientation == VERTICAL_LIST) {outRect.set(0, 0, 0, mDivider.getIntrinsicHeight());} else {outRect.set(0, 0, mDivider.getIntrinsicWidth(), 0);}}}

下面是MainActivity.java主界面文件:

public class MainActivity extends Activity {private RecyclerView mRecyclerView;private List<String> mDatas;private SimpleAdapter mAdapter;protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);initViews();initDatas();mAdapter = new SimpleAdapter(this, mDatas);mRecyclerView.setAdapter(mAdapter);// 设置RecyclerView的布局管理:1、创建一个线性布局管理器2、设置布局管理器LinearLayoutManager linearLayoutManager = new LinearLayoutManager(this,LinearLayoutManager.VERTICAL, false);mRecyclerView.setLayoutManager(linearLayoutManager);// 设置RecyclerView的Item间分隔线mRecyclerView.addItemDecoration(new DividerItemDecoration(this,DividerItemDecoration.VERTICAL_LIST));}private void initViews() {mRecyclerView = (RecyclerView) findViewById(R.id.id_recyclerview);}private void initDatas() {mDatas = new ArrayList<String>();for (int i = 'A'; i <= 'z'; i++) {mDatas.add("" + (char) i);}}}

该白色分割线是系统默认的,可以在theme.xml中找到该属性的使用情况。那么,使用系统的listDivider有什么好处呢?就是方便我们去随意的改变,该属性我们可以直接声明在:(注意在不同版本values-v11和values-v14也写上

<style name="AppTheme" parent="AppBaseTheme">        <!-- All customizations that are NOT specific to a particular API-level can go here. -->    <item name="android:listDivider">@drawable/divider</item>      </style>

下面是res/drawable/divider.xml 自定义分隔线文件:

<?xml version="1.0" encoding="utf-8"?><shape xmlns:android="http://schemas.android.com/apk/res/android"    android:shape="rectangle" >    <gradient        android:centerColor="#ff00ff00"        android:endColor="#ff0000ff"        android:startColor="#ffff0000"        android:type="linear" />    <size android:height="4dp" /></shape>


二、LayoutManager

上面实现了类似ListView样子的Demo,通过使用其默认的LinearLayoutManager。

RecyclerView.LayoutManager吧,这是一个抽象类,好在系统提供了3个实现类:

  1. LinearLayoutManager 现行管理器,支持横向、纵向。
  2. GridLayoutManager 网格布局管理器
  3. StaggeredGridLayoutManager 瀑布就式布局管理器

示例二:实现类似GridView


//mRecyclerView.setLayoutManager(new LinearLayoutManager(this));mRecyclerView.setLayoutManager(new GridLayoutManager(this,4));

下面是res/layout/item_textview.xml 布局文件:(去掉设置分隔线,通过背景色设置分隔线)

<span style="color:#333333;"><?xml version="1.0" encoding="utf-8"?><FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:layout_width="match_parent"    android:background="#44ff0000"    </span><span style="color:#ff0000;">android:layout_margin="3dp"</span><span style="color:#333333;">    android:layout_height="72dp" >    <TextView        android:id="@+id/id_item"        android:layout_width="</span><span style="color:#ff0000;">72dp</span><span style="color:#333333;">"        android:layout_height="match_parent"        android:gravity="center" /></FrameLayout></span>

示例三:StaggeredGridLayoutManager 瀑布就式布局管理器

mRecyclerView.setLayoutManager(new StaggeredGridLayoutManager(4,StaggeredGridLayoutManager.HORIZONTAL));
第二个参数传一个orientation,如果传入的是StaggeredGridLayoutManager.VERTICAL代表有多少列;那么传入的如果是StaggeredGridLayoutManager.HORIZONTAL就代表有多少行

示例四:


下面是res/layout/activity_main.xml 布局文件:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    xmlns:tools="http://schemas.android.com/tools"    android:layout_width="match_parent"    android:layout_height="match_parent" >    <android.support.v7.widget.RecyclerView        android:id="@+id/id_recyclerview"        android:layout_width="match_parent"        android:layout_height="match_parent" /></RelativeLayout>

下面是res/layout/masonry_item.xml 布局文件:

<?xml version="1.0" encoding="utf-8"?><LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"    android:layout_width="wrap_content"    android:layout_height="wrap_content"    android:layout_margin="2dp"    android:background="#ffffff"    android:orientation="vertical" >    <ImageView        android:id="@+id/masonry_item_img"        android:layout_width="match_parent"        android:layout_height="wrap_content"        android:adjustViewBounds="true"        android:scaleType="centerCrop" />    <TextView        android:id="@+id/masonry_item_title"        android:layout_width="match_parent"        android:layout_height="wrap_content"        android:gravity="center" /></LinearLayout>


为了简单起见,我们假设每个item的数据是一个产品信息

public class Product {private int img;private String title;public Product(int img, String title) {this.img = img;this.title = title;}public int getImg() {return img;}public void setImg(int img) {this.img = img;}public String getTitle() {return title;}public void setTitle(String title) {this.title = title;}}


下面是SimpleAdapter.java文件:

public class SimpleAdapter extends RecyclerView.Adapter<MyViewHolder> {private LayoutInflater mInflater;private Context mContext;private List<Product> mDatas;public interface OnItemClickListener{void onItemClick(View view,int position);void onItemLongClick(View view,int position);}/** * 自定义接口点击事件 */private OnItemClickListener mOnItemClickListener;public void setOnItemClickListener(OnItemClickListener listener){this.mOnItemClickListener=listener;}public SimpleAdapter(Context context, List<Product> datas) {this.mContext = context;this.mDatas = datas;mInflater = LayoutInflater.from(context);// 加载布局管理器}public int getItemCount() {return mDatas.size();}/** * 绑定ViewHolder */public void onBindViewHolder(final MyViewHolder holder, final int position) {holder.mImageView.setImageResource(mDatas.get(position).getImg());holder.mTextView.setText(mDatas.get(position).getTitle());if(mOnItemClickListener!=null){holder.mImageView.setOnClickListener(new OnClickListener() {public void onClick(View v) {mOnItemClickListener.onItemClick(holder.mImageView, position);}});holder.mImageView.setOnLongClickListener(new OnLongClickListener() {public boolean onLongClick(View v) {mOnItemClickListener.onItemLongClick(holder.mImageView, position);return false;}});}}/** * 创建ViewHolder */public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {//将xml布局转换为view对象View view = mInflater.inflate(R.layout.masonry_item, parent, false);MyViewHolder viewHolder = new MyViewHolder(view);return viewHolder;}class MyViewHolder extends ViewHolder {ImageView mImageView;TextView mTextView;public MyViewHolder(View view) {super(view);mImageView=(ImageView) view.findViewById(R.id.masonry_item_img);mTextView=(TextView) view.findViewById(R.id.masonry_item_title);}}}

下面是MainActivity.java主界面文件:

public class MainActivity extends Activity {private RecyclerView mRecyclerView;private List<Product> productList;private SimpleAdapter mAdapter;protected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mRecyclerView = (RecyclerView) findViewById(R.id.id_recyclerview);mRecyclerView.setLayoutManager(new StaggeredGridLayoutManager(2,StaggeredGridLayoutManager.VERTICAL));initDatas();mAdapter = new SimpleAdapter(this, productList);mRecyclerView.setAdapter(mAdapter);mAdapter.setOnItemClickListener(new OnItemClickListener() {public void onItemLongClick(View view, int position) {Toast.makeText(MainActivity.this, "long click:"+position, Toast.LENGTH_LONG).show();}public void onItemClick(View view, int position) {Toast.makeText(MainActivity.this, "click:"+position, Toast.LENGTH_LONG).show();}});}private void initDatas() {productList = new ArrayList<Product>();Product p1 = new Product(R.drawable.p1, "我是照片1");productList.add(p1);Product p2 = new Product(R.drawable.p2, "我是照片2");productList.add(p2);Product p3 = new Product(R.drawable.p3, "我是照片3");productList.add(p3);Product p4 = new Product(R.drawable.p4, "我是照片4");productList.add(p4);Product p5 = new Product(R.drawable.p5, "我是照片5");productList.add(p5);Product p6 = new Product(R.drawable.p6, "我是照片6");productList.add(p6);Product p7 = new Product(R.drawable.p2, "我是照片7");productList.add(p7);Product p8 = new Product(R.drawable.p1, "我是照片8");productList.add(p8);Product p9 = new Product(R.drawable.p4, "我是照片9");productList.add(p9);Product p10 = new Product(R.drawable.p6, "我是照片10");productList.add(p10);Product p11 = new Product(R.drawable.p3, "我是照片11");productList.add(p11);}}


参考文章http://blog.csdn.net/lmj623565791/article/details/45059587


0 0
原创粉丝点击