Android 首頁(yè)輪播圖和list混合(RecyclerView 添加 Header)

使用 convenientBanner 輪播圖給 RecyclerView 添加 header

先來(lái)看看效果圖浙芙。我們要實(shí)現(xiàn)的就是上面輪播圖輪播,下面是列表 list 展示數(shù)據(jù)跪妥。實(shí)際應(yīng)用中很多都會(huì)使用輪播圖去插入廣告趣惠,活動(dòng)等。

那么我們要怎么實(shí)現(xiàn)呢?

思路有很多感昼,可以用 ViewPager 去實(shí)現(xiàn)輪播圖,不過(guò)使用比較麻煩罐脊,要自己實(shí)現(xiàn)無(wú)限循環(huán)輪播定嗓,自動(dòng)輪播等。

還可以用 RecyclerView 去打造一個(gè)輪播圖萍桌。LinearSnapHelper 可以幫助你滑動(dòng)停止的時(shí)候某頁(yè)居中宵溅。設(shè)置定時(shí)器可以實(shí)現(xiàn)自動(dòng)輪播。

這里我們并不使用以上的方法上炎,我們使用外部的框架 convenientBanner 去直接使用輪播圖恃逻。有關(guān)詳細(xì)使用可以搜一下或者點(diǎn)github地址 有說(shuō)明。

效果圖

項(xiàng)目開(kāi)始

新建之后添加依賴(lài)

    //    輪播器
    compile 'com.bigkoo:convenientbanner:2.0.5'
    //圖片加載框架
    compile 'com.github.bumptech.glide:glide:3.7.0'

因?yàn)檫@里的例子是使用網(wǎng)絡(luò)圖片加載的方式,所以還要添加網(wǎng)絡(luò)權(quán)限和存儲(chǔ)權(quán)限辛块。

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

布局

主布局添加 RecyclerView 即可。整個(gè)主界面其實(shí)就是一個(gè) RecyclerView铅碍,輪播圖其實(shí)是 Rv 根據(jù)不同的 Type 添加不同的 View 來(lái)放到 Rv 頭部的润绵。后面會(huì)再提到。

另外就是 rv_header_banner.xml胞谈,這個(gè)就是放到 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" />

還有一個(gè) rv_header_img.xml。這個(gè)是構(gòu)建 Banner 的視圖的時(shí)候用到烦绳。

<?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>

最后一個(gè) rv_item.xml卿捎,這個(gè)是展示 list 數(shù)據(jù)。

<?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>

以上布局基本完成径密。

細(xì)心的人會(huì)發(fā)現(xiàn)午阵,輪播圖那里有幾個(gè)小點(diǎn)指示器。這個(gè)的樣式是我們自定義的享扔。所以我們這里還要在 drawable 下新建兩個(gè)圓點(diǎn)的樣式文件

<?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 配置的核心是根據(jù)不同的 itemType 加載不同的 View。

/**
 * 根據(jù)不同的 ViewType 返回不同的 ViewHolder
 * 通過(guò) setter 方法將不同的 View 注入進(jìn) 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);//插入下標(biāo)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 實(shí)際需要不包括 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 加上算一個(gè) 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 點(diǎn)擊事件接口
        void onItemClick(int position, String data);
    }
}

以上是 Adapter 的代碼籽懦,可見(jiàn)加了兩個(gè) TYPE 來(lái)區(qū)分 header 和 正常的數(shù)據(jù) item。(我們也可以加更多的 Type 去實(shí)現(xiàn)更復(fù)雜的布局)在 onCreateViewHolder 里面判斷 TYPE 如果是 header 則返回 headerView氛魁,否則加載正常的 item 布局暮顺。里面一些函數(shù)還有邏輯都加了注釋說(shuō)明了,這里就不詳細(xì)說(shuō)了秀存。

MainActivity

讓我們回到 MainActivity捶码,這里我們將初始化 Banner,并用 adapter.setmHeaderView 放進(jìn) Adapter 中或链。在使用 LayoutManager 設(shè)置一下 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) //設(shè)置指示器的方向(左株扛、中尤筐、右)
                .setPageIndicator(new int[] { R.drawable.indicator_gray, R.drawable.indicator_red })//設(shè)置指示器樣式
                .setOnItemClickListener(this)//點(diǎn)擊事件
                .setScrollDuration(1500);//滑動(dòng)的時(shí)間

    }

    @Override
    public void onItemClick(int position) {//Banner 點(diǎn)擊事件
        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 的基本設(shè)置就是這樣。通過(guò) setPages 來(lái)配置洞就,上面的配置都加了注釋說(shuō)明盆繁,更多的配置樣式可以查查。

mBanner 綁定的是 rv_header_banner.xml 視圖下的 banner 旬蟋。而 NetWorkImageHolderView 里面的 view 綁定了 rv_header_img.xml.里面的圖片 ImageView 則是綁定了 R.id.iv_head 這個(gè) img 控件油昂。 總的來(lái)說(shuō),Banner 的視圖 View 實(shí)際是 NetWorkImageHolderViewonCreateView 里面返回的。類(lèi)同 Adapter 的使用

UpdateUI 這里用 Glide 加載了網(wǎng)絡(luò)圖片冕碟。Glide 不是這里的主角所以不說(shuō)明了拦惋,用法也很簡(jiǎn)單。placeholder 是設(shè)置占位圖安寺。

以上就基本完成了 Banner 的配置厕妖。在 onResume 里面添加一句 mBanner.startTurning(3000) 就可以自動(dòng)輪播了。

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

View header = LayoutInflater.from(this).inflate(R.layout.rv_header_banner, null);
        mBanner = (ConvenientBanner) header.findViewById(R.id.banner);
        //設(shè)置高度是屏幕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的源碼點(diǎn)這里

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市迎捺,隨后出現(xiàn)的幾起案子举畸,更是在濱河造成了極大的恐慌,老刑警劉巖凳枝,帶你破解...
    沈念sama閱讀 222,378評(píng)論 6 516
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件抄沮,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡岖瑰,警方通過(guò)查閱死者的電腦和手機(jī)合是,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,970評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)锭环,“玉大人聪全,你說(shuō)我怎么就攤上這事「ū纾” “怎么了难礼?”我有些...
    開(kāi)封第一講書(shū)人閱讀 168,983評(píng)論 0 362
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)玫锋。 經(jīng)常有香客問(wèn)我蛾茉,道長(zhǎng),這世上最難降的妖魔是什么撩鹿? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 59,938評(píng)論 1 299
  • 正文 為了忘掉前任谦炬,我火速辦了婚禮,結(jié)果婚禮上节沦,老公的妹妹穿的比我還像新娘键思。我一直安慰自己,他們只是感情好甫贯,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,955評(píng)論 6 398
  • 文/花漫 我一把揭開(kāi)白布吼鳞。 她就那樣靜靜地躺著,像睡著了一般叫搁。 火紅的嫁衣襯著肌膚如雪赔桌。 梳的紋絲不亂的頭發(fā)上供炎,一...
    開(kāi)封第一講書(shū)人閱讀 52,549評(píng)論 1 312
  • 那天,我揣著相機(jī)與錄音疾党,去河邊找鬼音诫。 笑死,一個(gè)胖子當(dāng)著我的面吹牛雪位,可吹牛的內(nèi)容都是我干的竭钝。 我是一名探鬼主播,決...
    沈念sama閱讀 41,063評(píng)論 3 422
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼茧泪,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了聋袋?” 一聲冷哼從身側(cè)響起队伟,我...
    開(kāi)封第一講書(shū)人閱讀 39,991評(píng)論 0 277
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎幽勒,沒(méi)想到半個(gè)月后嗜侮,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,522評(píng)論 1 319
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡啥容,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,604評(píng)論 3 342
  • 正文 我和宋清朗相戀三年锈颗,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片咪惠。...
    茶點(diǎn)故事閱讀 40,742評(píng)論 1 353
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡击吱,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出遥昧,到底是詐尸還是另有隱情覆醇,我是刑警寧澤,帶...
    沈念sama閱讀 36,413評(píng)論 5 351
  • 正文 年R本政府宣布炭臭,位于F島的核電站永脓,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏鞋仍。R本人自食惡果不足惜常摧,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 42,094評(píng)論 3 335
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望威创。 院中可真熱鬧落午,春花似錦、人聲如沸肚豺。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 32,572評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)详炬。三九已至盐类,卻和暖如春寞奸,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背在跳。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,671評(píng)論 1 274
  • 我被黑心中介騙來(lái)泰國(guó)打工枪萄, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人猫妙。 一個(gè)月前我還...
    沈念sama閱讀 49,159評(píng)論 3 378
  • 正文 我出身青樓瓷翻,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親割坠。 傳聞我的和親對(duì)象是個(gè)殘疾皇子齐帚,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,747評(píng)論 2 361

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,317評(píng)論 25 707
  • 內(nèi)容抽屜菜單ListViewWebViewSwitchButton按鈕點(diǎn)贊按鈕進(jìn)度條TabLayout圖標(biāo)下拉刷新...
    皇小弟閱讀 46,791評(píng)論 22 665
  • 昨晚睡覺(jué)前,看了一個(gè)關(guān)于單親母親的故事彼哼,故事的名字叫作《問(wèn)天要條活路对妄!》,非常的勵(lì)志敢朱。講述了在計(jì)劃生育年代剪菱,筆者的...
    七云舒閱讀 681評(píng)論 4 14
  • 一路躊躇,一路波折拴签,我的人生已經(jīng)過(guò)完了四分之一孝常,本該工作穩(wěn)定,愛(ài)情甜蜜蚓哩,可現(xiàn)實(shí)卻不是如此构灸,我還是一個(gè)人走在來(lái)來(lái)往往...
    七七1997閱讀 247評(píng)論 0 0
  • 普通朋友我又不缺你一個(gè),認(rèn)識(shí)你無(wú)非就是想私自占有你岸梨。不行的話只能淪為陌生人冻押。你難過(guò)我也沒(méi)辦法,我也難過(guò)
    橘子姐姐和遠(yuǎn)方閱讀 396評(píng)論 0 2