1. 程式人生 > >RecyclerView的五個小例項

RecyclerView的五個小例項

一、垂直的RecyclerView,類似listview的效果。

1.首先匯入庫:compile 'com.android.support:appcompat-v7:25.1.0'

2.在佈局中引入RecyclerView.

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

3.初始化RecyclerView。

 recycler_view = (RecyclerView)findViewById(R.id.recycler_view);
//設定為垂直的樣式
recycler_view.setLayoutManager(new LinearLayoutManager(mContext));
//使用的是系統預設的分割線
recycler_view.addItemDecoration(new DividerItemDecoration(mContext, DividerItemDecoration.VERTICAL));
//設定介面卡
recycler_view.setAdapter(adapter=new BaseAdapter(mContext,mData));
//設定預設動畫
recycler_view.setItemAnimator(new DefaultItemAnimator());

4.編寫介面卡BaseAdapter

public class BaseAdapter extends RecyclerView.Adapter<BaseAdapter.MyViewHolder> {

    private Context mContext;
    private List<String> mData;

    //自定義點選事件和長按事件
    private OnItemClickListener mOnItemClickListener;

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

    //構造器
    public BaseAdapter(Context mContext, List<String> mData) {
    this.mContext = mContext;
    this.mData = mData;
    }
    //載入佈局
    @Override
    public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
    MyViewHolder holder = new MyViewHolder(LayoutInflater.from(mContext).inflate(R.layout.item_base, parent, false));
    return holder;
    }
    //為佈局載入資料
    @Override
    public void onBindViewHolder(MyViewHolder holder, int position) {
    holder.tv_text.setText(mData.get(position));

    setClickListener(holder,position);
    }
    //設定點選事件
    private void setClickListener(final MyViewHolder holder, int position) {
    if(mOnItemClickListener!=null){
    holder.itemView.setOnClickListener(new View.OnClickListener() {
    @Override
    public void onClick(View view) {
    int pos = holder.getLayoutPosition();
    mOnItemClickListener.onItemClick(holder.itemView, pos);
    }
    });
    }

    holder.itemView.setOnLongClickListener(new View.OnLongClickListener()
    {
    @Override
    public boolean onLongClick(View v)
    {
    int pos = holder.getLayoutPosition();
    mOnItemClickListener.onItemLongClick(holder.itemView, pos);
    return true;//返回true可以遮蔽點選監聽的響應
    }
    });
    }

    //總共多少個項
    @Override
    public int getItemCount() {
    return mData.size();
    }
    //初始化佈局資訊
    class MyViewHolder extends RecyclerView.ViewHolder{
    TextView tv_text;
    public MyViewHolder(View itemView) {
    super(itemView);
    tv_text = (TextView) itemView.findViewById(R.id.tv_text);
    }
    }

    //新增
    public void addData(int position) {
    mData.add(position, "Insert One");
    notifyItemInserted(position);
    }
    //刪除
    public void removeData(int position) {
    mData.remove(position);
    notifyItemRemoved(position);
    }
}

5.新增點選監聽,長按監聽。寫一個介面類。

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

6.BaseAdapter中已經使用了這個介面,在activity呼叫。

adapter.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(View view, int position) {
    view.startAnimation(animation);
    Toast.makeText(mContext, position + " click",
    Toast.LENGTH_SHORT).show();
}

@Override
public void onItemLongClick(View view, int position) {
    view.startAnimation(animation);
    Toast.makeText(mContext, position + " long click",
    Toast.LENGTH_SHORT).show();
}
});

7.效果圖

這裡寫圖片描述

二、網格的RecyclerView。

1.只需要修改兩處程式碼

 //設定為水平網格樣式
recycler_view.setLayoutManager(new StaggeredGridLayoutManager(4,StaggeredGridLayoutManager.HORIZONTAL));//如果需要垂直,改為垂直即可
////使用的是自定義的網格分割線
recycler_view.addItemDecoration(new DividerGridItemDecoration(mContext));

2.有一個DividerGridItemDecoration這個類需要自定義,網格分割線,程式碼如下:

public class DividerGridItemDecoration extends RecyclerView.ItemDecoration{
    private static final int[] ATTRS = new int[] { android.R.attr.listDivider };
    private Drawable mDivider;

    public DividerGridItemDecoration(Context context)
    {
        final TypedArray a = context.obtainStyledAttributes(ATTRS);
        mDivider = a.getDrawable(0);
        a.recycle();
    }

    @Override
    public void onDraw(Canvas c, RecyclerView parent, RecyclerView.State state)
    {

        drawHorizontal(c, parent);
        drawVertical(c, parent);

    }

    private int getSpanCount(RecyclerView parent)
    {
        // 列數
        int spanCount = -1;
        RecyclerView.LayoutManager layoutManager = parent.getLayoutManager();
        if (layoutManager instanceof GridLayoutManager)
        {

            spanCount = ((GridLayoutManager) layoutManager).getSpanCount();
        } else if (layoutManager instanceof StaggeredGridLayoutManager)
        {
            spanCount = ((StaggeredGridLayoutManager) layoutManager)
            .getSpanCount();
        }
        return spanCount;
        }

        public void drawHorizontal(Canvas c, RecyclerView parent)
        {
            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.getLeft() - params.leftMargin;
                final int right = child.getRight() + params.rightMargin
                + mDivider.getIntrinsicWidth();
                final int top = child.getBottom() + params.bottomMargin;
                final int bottom = top + mDivider.getIntrinsicHeight();
                mDivider.setBounds(left, top, right, bottom);
                mDivider.draw(c);
            }
    }

    public void drawVertical(Canvas c, RecyclerView parent)
    {
        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 top = child.getTop() - params.topMargin;
            final int bottom = child.getBottom() + params.bottomMargin;
            final int left = child.getRight() + params.rightMargin;
            final int right = left + mDivider.getIntrinsicWidth();

            mDivider.setBounds(left, top, right, bottom);
            mDivider.draw(c);
        }
    }

    private boolean isLastColum(RecyclerView parent, int pos, int spanCount,
    int childCount)
    {
        RecyclerView.LayoutManager layoutManager = parent.getLayoutManager();
        if (layoutManager instanceof GridLayoutManager)
        {
            if ((pos + 1) % spanCount == 0)// 如果是最後一列,則不需要繪製右邊
        {
        return true;
        }
        } else if (layoutManager instanceof StaggeredGridLayoutManager)
        {
        int orientation = ((StaggeredGridLayoutManager) layoutManager)
        .getOrientation();
        if (orientation == StaggeredGridLayoutManager.VERTICAL)
        {
            if ((pos + 1) % spanCount == 0)// 如果是最後一列,則不需要繪製右邊
        {
        return true;
        }
        } else
        {
        childCount = childCount - childCount % spanCount;
        if (pos >= childCount)// 如果是最後一列,則不需要繪製右邊
        return true;
        }
        }
        return false;
    }

    private boolean isLastRaw(RecyclerView parent, int pos, int spanCount,
      int childCount)
    {
    RecyclerView.LayoutManager layoutManager = parent.getLayoutManager();
    if (layoutManager instanceof GridLayoutManager)
    {
    childCount = childCount - childCount % spanCount;
    if (pos >= childCount)// 如果是最後一行,則不需要繪製底部
    return true;
    } else if (layoutManager instanceof StaggeredGridLayoutManager)
    {
    int orientation = ((StaggeredGridLayoutManager) layoutManager)
    .getOrientation();
    // StaggeredGridLayoutManager 且縱向滾動
    if (orientation == StaggeredGridLayoutManager.VERTICAL)
    {
    childCount = childCount - childCount % spanCount;
    // 如果是最後一行,則不需要繪製底部
    if (pos >= childCount)
    return true;
    } else
    // StaggeredGridLayoutManager 且橫向滾動
    {
    // 如果是最後一行,則不需要繪製底部
    if ((pos + 1) % spanCount == 0)
    {
    return true;
    }
    }
    }
    return false;
    }

    @Override
    public void getItemOffsets(Rect outRect, int itemPosition,
       RecyclerView parent)
    {
    int spanCount = getSpanCount(parent);
    int childCount = parent.getAdapter().getItemCount();
    if (isLastRaw(parent, itemPosition, spanCount, childCount))// 如果是最後一行,則不需要繪製底部
    {
    outRect.set(0, 0, mDivider.getIntrinsicWidth(), 0);
    } else if (isLastColum(parent, itemPosition, spanCount, childCount))// 如果是最後一列,則不需要繪製右邊
    {
    outRect.set(0, 0, 0, mDivider.getIntrinsicHeight());
    } else
    {
    outRect.set(0, 0, mDivider.getIntrinsicWidth(),
    mDivider.getIntrinsicHeight());
    }
    }
}

3.效果圖

這裡寫圖片描述

三、用RecyclerView做的四級聯動。這部分程式碼比較多,也比較複雜,請看demo吧

1.效果圖

這裡寫圖片描述

四、帶重新整理的RecyclerView,這個需要引入一個新的庫,XRecyclerView,用github搜尋一下,用第一個就行。

1.引入XRecyclerView的庫,compile 'com.jcodecraeer:xrecyclerview:1.3.2',下載地址https://github.com/jianghejie/XRecyclerView

2.在佈局中使用XRecyclerView.

<?xml version="1.0" encoding="utf-8"?>
<com.jcodecraeer.xrecyclerview.XRecyclerView xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:id="@+id/recycler_view"
android:layout_width="match_parent"
android:layout_height="wrap_content"
>

</com.jcodecraeer.xrecyclerview.XRecyclerView>

3.其他的都跟一類似,重新整理時,使用

 recycler_view.setLoadingListener(new XRecyclerView.LoadingListener() {
@Override
public void onRefresh() {
refresh();
}
@Override
public void onLoadMore() {
loadMore();
}
});

4.注意重新整理完成,無論上拉還是下拉,完成之後都要執行recycler_view.refreshComplete();,如果下拉載入更多,已經沒有更多了,那麼,需要執行recycler_view.loadMoreComplete();

5.效果圖:

這裡寫圖片描述

五、用RecyclerView做瀑布流,

1.只需要更改兩處即可

 //設定為垂直3列
recycler_view.setLayoutManager(new StaggeredGridLayoutManager(3,StaggeredGridLayoutManager.VERTICAL));
////item間隔16
recycler_view.addItemDecoration(new SpacesItemDecoration(16));

2.其中SpacesItemDecoration這個類的是為了新增間距,程式碼如下。

public class SpacesItemDecoration extends RecyclerView.ItemDecoration {

private int space;

public SpacesItemDecoration(int space) {
this.space=space;
}

@Override
public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
outRect.left=space;
outRect.right=space;
outRect.bottom=space;
if(parent.getChildAdapterPosition(view)==0){
outRect.top=space;
}
}
}

3.這裡還添加了,增加還刪除單項操作,點選增加第一項,長按刪除第二項。因為新增過預設動畫,所以會有效果。

 adapter.setOnItemClickListener(new OnItemClickListener() {
@Override
public void onItemClick(View view, int position) {
view.startAnimation(animation);
Toast.makeText(mContext, position + " 新增A",
Toast.LENGTH_SHORT).show();
adapter.addData(0);
}

@Override
public void onItemLongClick(View view, int position) {
view.startAnimation(animation);
Toast.makeText(mContext, position + " 刪除B",
Toast.LENGTH_SHORT).show();
adapter.removeData(2);
}
});

4.效果圖

這裡寫圖片描述

六、多個item的RecyclerView。

1.新建一個DifferentItemAdapter,程式碼如下

    public class DifferentItemAdapter extends RecyclerView.Adapter {
private Context mContext;
private List<String> mData;

private static final int CASE1=1;
private static final int CASE2=2;

public DifferentItemAdapter(Context mContext, List<String> mData) {
this.mContext = mContext;
this.mData = mData;
}
//自定義點選事件和長按事件
private OnItemClickListener mOnItemClickListener;

public void setOnItemClickListener(OnItemClickListener mOnItemClickListener)
{
this.mOnItemClickListener = mOnItemClickListener;
}
@Override
public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
switch (viewType){
case CASE1:
   return new MyViewHolder(LayoutInflater.from(mContext).inflate(R.layout.item_base, parent, false));
case CASE2:
return new DifferentHolder(LayoutInflater.from(mContext).inflate(R.layout.item_different,parent,false));
}
return null;
}

@Override
public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {
if(position%2==0){
DifferentHolder holder1= (DifferentHolder) holder;
holder1.tv_text.setText(mData.get(position));
}else{
MyViewHolder holder2= (MyViewHolder) holder;
holder2.tv_text.setText(mData.get(position));
}

setClickListener(holder,position);
}

@Override
public int getItemCount() {
return mData.size();
}
//這個非常重要,根據這個選擇不同的佈局
@Override
public int getItemViewType(int position) {
if(position%2==0){
return CASE2;
}else{
return CASE1;
}
}

//初始化佈局資訊
class MyViewHolder extends RecyclerView.ViewHolder{
TextView tv_text;
public MyViewHolder(View itemView) {
super(itemView);
tv_text = (TextView) itemView.findViewById(R.id.tv_text);
}
}

class DifferentHolder extends RecyclerView.ViewHolder{
TextView tv_text;
public DifferentHolder(View itemView) {
super(itemView);
tv_text = (TextView) itemView.findViewById(R.id.tv_text);
}
}


//設定點選事件
private void setClickListener(final RecyclerView.ViewHolder holder, int position) {
if(mOnItemClickListener!=null){
holder.itemView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
int pos = holder.getLayoutPosition();
mOnItemClickListener.onItemClick(holder.itemView, pos);
}
});
}

holder.itemView.setOnLongClickListener(new View.OnLongClickListener()
{
@Override
public boolean onLongClick(View v)
{
int pos = holder.getLayoutPosition();
mOnItemClickListener.onItemLongClick(holder.itemView, pos);
return true;//返回true可以遮蔽點選監聽的響應
}
});
}
}`

裡面只是更改了繼承的方式,繼承RecyclerView.Adapter,不需要泛型了,同時多了一個public int getItemViewType(int position) 這個函式,根據不同的型別載入不同的佈局。只需要改Adapter其他的不用變。

2.效果圖

這裡寫圖片描述