Android 首页轮播图和list混合(RecyclerView 添加 Header)

使用 convenientBanner 轮播图给 RecyclerView 添加 header

先来看看效果图。我们要实现的就是上面轮播图轮播,下面是列表 list 展示数据。实际应用中很多都会使用轮播图去插入广告,活动等。

那么我们要怎么实现呢?

思路有很多,可以用 ViewPager 去实现轮播图,不过使用比较麻烦,要自己实现无限循环轮播,自动轮播等。

还可以用 RecyclerView 去打造一个轮播图。LinearSnapHelper 可以帮助你滑动停止的时候某页居中。设置定时器可以实现自动轮播。

这里我们并不使用以上的方法,我们使用外部的框架 convenientBanner 去直接使用轮播图。有关详细使用可以搜一下或者点github地址 有说明。

效果图

项目开始

新建之后添加依赖

    //    轮播器
    compile 'com.bigkoo:convenientbanner:2.0.5'
    //图片加载框架
    compile 'com.github.bumptech.glide:glide:3.7.0'

因为这里的例子是使用网络图片加载的方式,所以还要添加网络权限和存储权限。

 <uses-permission android:name="android.permission.INTERNET" />
 <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />

布局

主布局添加 RecyclerView 即可。整个主界面其实就是一个 RecyclerView,轮播图其实是 Rv 根据不同的 Type 添加不同的 View 来放到 Rv 头部的。后面会再提到。

另外就是 rv_header_banner.xml,这个就是放到 Rv 头部的。

<?xml version="1.0" encoding="utf-8"?>
<com.bigkoo.convenientbanner.ConvenientBanner
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/banner"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    app:canLoop="true" />

还有一个 rv_header_img.xml。这个是构建 Banner 的视图的时候用到。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" android:layout_width="match_parent"
    android:layout_height="match_parent">

    <ImageView
        android:id="@+id/iv_head"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:scaleType="centerCrop"/>

</LinearLayout>

最后一个 rv_item.xml,这个是展示 list 数据。

<?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="wrap_content">

    <TextView
        android:id="@+id/tv_item"
        android:layout_marginBottom="20dp"
        android:textSize="20sp"
        android:text="item"
        android:layout_alignParentBottom="true"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

</RelativeLayout>

以上布局基本完成。

细心的人会发现,轮播图那里有几个小点指示器。这个的样式是我们自定义的。所以我们这里还要在 drawable 下新建两个圆点的样式文件

<?xml version="1.0" encoding="utf-8"?>
<shape
    android:shape="oval"
    xmlns:android="http://schemas.android.com/apk/res/android">
    <solid android:color="#BDBDBD"/>
    <size android:height="10dp" android:width="10dp"/>
</shape>

<?xml version="1.0" encoding="utf-8"?>
<shape
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:shape="oval">
    <solid android:color="#D32F2F"/>
    <size android:width="10dp" android:height="10dp"/>
</shape>

Adapter

在这里,Adapter 配置的核心是根据不同的 itemType 加载不同的 View。

/**
 * 根据不同的 ViewType 返回不同的 ViewHolder
 * 通过 setter 方法将不同的 View 注入进 Adapter
 */

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

    public static final int TYPE_HEADER = 0;

    public static final int TYPE_NOMAL = 1;

    private View mHeaderView;

    private List<String> mDatas = new ArrayList<>();

    private OnItemClickListener mListener;

    public View getmHeaderView() {
        return mHeaderView;
    }

    public void setmHeaderView(View mHeaderView) {
        this.mHeaderView = mHeaderView;
        notifyItemInserted(0);//插入下标0位置
    }

    public void addDatas(List<String> datas){
        mDatas.addAll(datas);
        notifyDataSetChanged();
    }

    @Override
    public int getItemViewType(int position) {
        if(mHeaderView == null){
            return TYPE_NOMAL;
        }
        if(position == 0){
            return TYPE_HEADER;
        }
        return TYPE_NOMAL;
    }

    @Override
    public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        if(mHeaderView != null && viewType == TYPE_HEADER){
            return new MyViewHolder(mHeaderView);
        }

        View itemView = LayoutInflater.from(parent.getContext()).inflate(R.layout.rv_item, parent, false);
        return new MyViewHolder(itemView);

    }

    @Override
    public void onBindViewHolder(MyViewHolder holder, int position) {
        if (getItemViewType(position) == TYPE_HEADER) {
            return;
        }

        final int pos = getRealPosition(holder);//这里的 position 实际需要不包括 header
        final String data = mDatas.get(pos);

        holder.textView.setText(data);
        if(mListener == null) return;
        holder.itemView.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                mListener.onItemClick(pos, data);
            }
        });
    }

    /**
     * 添加头部布局后的位置
     * headerView 不为空则 position - 1
     */
    private int getRealPosition(MyViewHolder holder) {
        int position = holder.getLayoutPosition();
        return mHeaderView == null ? position : position - 1;
    }

    @Override
    public int getItemCount() {
        //header 不为空,则 rv 的总 Count 需要 +1(把 Header 加上算一个 item)
        return mHeaderView == null ? mDatas.size() : mDatas.size() + 1;
    }

    public class MyViewHolder extends RecyclerView.ViewHolder {
        TextView textView;

        public MyViewHolder(View itemView) {
            super(itemView);
            if(itemView == mHeaderView){ return; }
            textView = (TextView) itemView.findViewById(R.id.tv_item);
        }
    }

    public void setOnItemClickListener(OnItemClickListener listener){
            mListener = listener;
    }

    public interface OnItemClickListener{//item 点击事件接口
        void onItemClick(int position, String data);
    }
}

以上是 Adapter 的代码,可见加了两个 TYPE 来区分 header 和 正常的数据 item。(我们也可以加更多的 Type 去实现更复杂的布局)在 onCreateViewHolder 里面判断 TYPE 如果是 header 则返回 headerView,否则加载正常的 item 布局。里面一些函数还有逻辑都加了注释说明了,这里就不详细说了。

MainActivity

让我们回到 MainActivity,这里我们将初始化 Banner,并用 adapter.setmHeaderView 放进 Adapter 中。在使用 LayoutManager 设置一下 Rv 的布局,然后就基本完成了。


    private void initBanner(){
        networkImage = Arrays.asList(images);

        mBanner.setPages(new CBViewHolderCreator<NetWorkImageHolderView>() {
            @Override
            public NetWorkImageHolderView createHolder() {
                return new NetWorkImageHolderView();
            }
        }, networkImage)
                .setPageIndicatorAlign(ConvenientBanner.PageIndicatorAlign.ALIGN_PARENT_RIGHT) //设置指示器的方向(左、中、右)
                .setPageIndicator(new int[] { R.drawable.indicator_gray, R.drawable.indicator_red })//设置指示器样式
                .setOnItemClickListener(this)//点击事件
                .setScrollDuration(1500);//滑动的时间

    }

    @Override
    public void onItemClick(int position) {//Banner 点击事件
        Toast.makeText(MainActivity.this, "Banner:"+position, Toast.LENGTH_SHORT).show();
    }


    public class NetWorkImageHolderView implements Holder<String>{

        private ImageView imageView;

        @Override
        public View createView(Context context) {
            View view = LayoutInflater.from(context).inflate(R.layout.rv_header_img, null);
            imageView = (ImageView) view.findViewById(R.id.iv_head);
            imageView.setScaleType(ImageView.ScaleType.CENTER_CROP);
            return view;
        }

        @Override
        public void UpdateUI(Context context, int position, String data) {
            //Glide.with(context).load(data.getImgUrl()).into(imageView);
            Log.d("imgUrl", "UpdateUI: "+data);
            Glide.with(context).load(data).placeholder(R.mipmap.ic_launcher_round).into(imageView);
        }
    }

}

convenientBanner 的基本设置就是这样。通过 setPages 来配置,上面的配置都加了注释说明,更多的配置样式可以查查。

mBanner 绑定的是 rv_header_banner.xml 视图下的 banner 。而 NetWorkImageHolderView 里面的 view 绑定了 rv_header_img.xml.里面的图片 ImageView 则是绑定了 R.id.iv_head 这个 img 控件。 总的来说,Banner 的视图 View 实际是 NetWorkImageHolderViewonCreateView 里面返回的。类同 Adapter 的使用

UpdateUI 这里用 Glide 加载了网络图片。Glide 不是这里的主角所以不说明了,用法也很简单。placeholder 是设置占位图。

以上就基本完成了 Banner 的配置。在 onResume 里面添加一句 mBanner.startTurning(3000) 就可以自动轮播了。

剩下的就是 Rv 的基本用法了。

View header = LayoutInflater.from(this).inflate(R.layout.rv_header_banner, null);
        mBanner = (ConvenientBanner) header.findViewById(R.id.banner);
        //设置高度是屏幕1/4
        mBanner.setLayoutParams(new RecyclerView.LayoutParams(
                ViewGroup.LayoutParams.MATCH_PARENT, getWindowManager().getDefaultDisplay().getHeight()/3));

        mRecyclerView = (RecyclerView) findViewById(R.id.rv_content);
        mLayoutManager = new LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false);
        mRecyclerView.setLayoutManager(mLayoutManager);
        mRecyclerView.setItemAnimator(new DefaultItemAnimator());

        initBanner();

        myAdapter = new MyAdapter();
        myAdapter.addDatas(mData);
        myAdapter.setmHeaderView(mBanner);
        mRecyclerView.setAdapter(myAdapter);

        myAdapter.setOnItemClickListener(new MyAdapter.OnItemClickListener() {
            @Override
            public void onItemClick(int position, String data) {
                Toast.makeText(MainActivity.this, data, Toast.LENGTH_SHORT).show();
            }
        });

还有需要的可以看一下我的源码。

这次demo的源码点这里

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 225,529评论 6 524
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 96,683评论 3 406
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 172,905评论 0 370
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 61,318评论 1 303
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 70,325评论 6 401
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 53,754评论 1 316
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 42,081评论 3 431
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 41,088评论 0 280
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 47,653评论 1 327
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 39,651评论 3 347
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 41,766评论 1 355
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 37,359评论 5 351
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 43,083评论 3 341
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 33,491评论 0 25
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 34,654评论 1 278
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 50,367评论 3 383
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 46,836评论 2 367

推荐阅读更多精彩内容