recyclerview完整使用

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的滑动监听方法中不断调用这个方法进行判断:

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();
                }
            }
        });


这里我的加载方法是利用装入更多数据list重新创建adapter然后设置给rv,这样rv就会直接显示第一页。所以使用方法将rv跳到指定位置:

rv.scrollToPosition(currentPosition);

这里传入的position是页面顶部显示的position,自己调试就可以了。


此外还可以在底部放一个初始为gone的加载进度条QAQ



  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值