1.和listview一样,创建数据模型和子项布局文件
2.自定义适配器:
首先在适配器类中自定义ViewHolder内部类,然后将其作为泛型传入类的声明中:
class MyViewHolder extends RecyclerView.ViewHolder{
TextView tvTitle,tvNumber,tvTime;
ImageView iv;
public MyViewHolder(View itemView) {
super(itemView);
tvTitle= (TextView) itemView.findViewById(R.id.tvTitle);
tvNumber= (TextView) itemView.findViewById(R.id.tvNumber);
tvTime= (TextView) itemView.findViewById(R.id.tvTime);
iv= (ImageView) itemView.findViewById(R.id.image);
}
}
public class ClassAdapter extends RecyclerView.Adapter<ClassAdapter.MyViewHolder> {
2.实现核心方法:
要实现的三个核心方法是getItemCount,onCreateViewHolder,onBindViewHolder。第一个是获取列表项数,一般取传入的数组长度。
第二个和第三个表示创建Viewholder和将其与数据模型对应起来。也就是说,我们只需要正确创建viewholder和把数据传给他,recyclerview就可以自动帮我们实现viewholder的应用。创建viewholder就是将相应的布局文件转化成view传入其构造器,这里用到的context在适配器的构造器传入:
@Override
public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
MyViewHolder holder=new MyViewHolder(LayoutInflater.from(context)
.inflate(R.layout.class_item,parent,false));
return holder;
}
@Override
public void onBindViewHolder(MyViewHolder holder, final int position) {
ClassItem item=list.get(position);
holder.tvTitle.setText(item.getTitle());
holder.tvNumber.setText(item.getNumber()+"人已参加");
holder.tvTime.setText("开课时间:"+item.getTime());
if (item.getImgUrl().length()>1){
Glide.with(context).load(item.getImgUrl()).into(holder.iv);
}
if (listener!=null){
holder.itemView.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
listener.onItemClick(view,position);
}
});
holder.itemView.setOnLongClickListener(new View.OnLongClickListener() {
@Override
public boolean onLongClick(View view) {
listener.onItemLongClick(view,position);
return false;
}
});
}
}
@Override
public int getItemCount() {
return list.size();
}
3.增加监听事件:
因为rv自身是没有设置监听的方法的,所以要自定义监听事件并直接设置在viewholder的view上(见上面的onBindViewHolder中的设置)。这里的思路是在适配器内部自定义一个监听接口传入:
public interface OnItenClickListener {
public void onItemClick(View view, int position);
public void onItemLongClick(View view,int position);
}
public void setLisenter(ClassAdapter.OnItenClickListener lisenter){
this.listener=lisenter;
}
4.使用:
rv使用的时候除了要设置适配器外还必须设置layoutmanager:
rv.setLayoutManager(new LinearLayoutManager(this));
这个线性布局设置还有一个多个参数的方法,可以指定布局方向。
5.设置分割线:
分割线也没有现成的,必须自定义一个类,这个我没有仔细研究,直接从网上找的一个分割线:
public class KopItemDecoration 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 KopItemDecoration(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;
}
@Override
public void onDraw(Canvas c, RecyclerView parent) {
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);
}
}
@Override
public 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);
}
}
}
然后设置:
rv.addItemDecoration(new KopItemDecoration(this,KopItemDecoration.VERTICAL_LIST));
可以看到,这个分割线还可以设置方向。
6.底部加载:
有一个需求,列表对数据的获取是分页的,第一页滑到底部后加载下一页,首先看判断rv滑到底部的方法:
private boolean isSlideToBottom(RecyclerView recyclerView) {
if (recyclerView == null) return false;
if (recyclerView.computeVerticalScrollExtent() + recyclerView.computeVerticalScrollOffset() >= recyclerView.computeVerticalScrollRange())
return true;
return false;
}
rv.addOnScrollListener(new RecyclerView.OnScrollListener() {
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
}
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
if (isSlideToBottom(recyclerView)) {
loadMore();
}
}
});
rv.scrollToPosition(currentPosition);
这里传入的position是页面顶部显示的position,自己调试就可以了。
此外还可以在底部放一个初始为gone的加载进度条QAQ