Android 5.X新特性之為RecyclerView添加HeaderView和FooterView

来源:http://www.cnblogs.com/guanmanman/archive/2016/11/23/6092378.html
-Advertisement-
Play Games

針對RecyclerView的頭部和底部,官方並沒有給我們提供像listView一樣可以直接通過addHeaderView()/addFooterView()的方法,所以只能靠我們自己去實現了,那怎麼實現呢?大家都知道RecyclerView已經為我們封裝好了Adapter和ViewHolder,在... ...


上一節我們講到了 Android 5.X新特性之RecyclerView基本解析及無限復用 相信大家也應該熟悉了RecyclerView的基本使用,這一節我們來學習下,為RecyclerView添加HeaderView和FooterView。

針對RecyclerView的頭部和底部,官方並沒有給我們提供像listView一樣可以直接通過addHeaderView()/addFooterView()的方法,所以只能靠我們自己去實現了,那怎麼實現呢?大家都知道RecyclerView已經為我們封裝好了Adapter和ViewHolder,在Adapter中我們需要重寫onCreateViewHolder(ViewGroup parent, int viewType)這個方法方便我們把ItemView佈局文件或是自定義View傳遞到ViewHolder中,從而達到ItemView的重覆使用和回收等。而我們今天要講的添加頭部和底部和該方法有密不可分的關係。

大家仔細觀察onCreateViewHolder(ViewGroup parent, int viewType)這個方法,在它的參數中,含有一個viewType,它就代表了每一個子列表中的ItemView的類型,而該類型我們又可以通過Adapter中封裝好的getItemViewType()方法來定義。因此,我們可以根據這兩個方法來完成我們今天的學習。

首先,我們也需要在BaseRecyclerAdapter中添加一個addHeaderView的方法,用於接受Activity中傳遞過來的HeaderView,並且把HeaderView添加到第0個ItemView中。

    private View mHeaderView;
    public void addHeaderView(View headerView){
        mHeaderView = headerView;
        notifyItemInserted(0);
    }

然後,我們在我們自定義的BaseRecyclerAdapter中重寫getItemViewType()方法,並且定義兩個靜態變數來區分我們的viewType的類型,如下:

    public final static int TYPE_HEADER = 0;
    public final static int TYPE_BODY = 1;
    @Override
    public int getItemViewType(int position) {
        return super.getItemViewType(position);
    }

ok,現在我們可以針對getItemViewType()方法來為我們的ItemView設置viewType類型了。我們知道,getItemViewType()預設返回的是0這個類型,所以我們重載該方法,在沒有mHeaderView時,我們讓它返回TYPE_BODY這個類型,而當有mHeaderView,由於把它添加到第0個ItemView中了,所以我們可以根據position等於0的時候讓它返回TYPE_HEADER這個類型。定義好的類型將會在onCreateViewHolder()中使用到。

所以我們的getItemViewType方法可以這樣設計:

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

到這裡大家已經很明確的知道了每個ItemView的viewType類型了,那麼我們就可以在onCreateViewHolder()方法中根據ItemView的viewType來做出不同的判斷了,如下:

    @Override
    public BaseViewHolderHelper onCreateViewHolder(ViewGroup parent, int viewType) {
        if(viewType == TYPE_HEADER && mHeaderView != null){
            return new BaseViewHolderHelper(mHeaderView);
        }
        View view = LayoutInflater.from(parent.getContext()).inflate(mLayoutResId, parent, false);
        return new BaseViewHolderHelper(view);
    }

代碼一目瞭然,就是根據viewType判斷是否為TYPE_HEADER,如果是,則添加不同的佈局或View,否則,載入正常的佈局,其他的不變。

然後,我們就可以在onBindViewHolder(BaseViewHolderHelper holder, int position)方法中根據當前的position位置來綁定我們要顯示數據了。

    @Override
    public void onBindViewHolder(BaseViewHolderHelper holder, int position) {
        if(getItemViewType(position) == TYPE_HEADER){
            return;
        }else{
            if(mHeaderView != null){
                position-- ;
            }
            holder.itemView.setTag(position);
            holder.itemView.setOnClickListener(this);
            holder.itemView.setOnLongClickListener(this);
            T itemData = mDatas.get(position);
            displayContents(holder,itemData);
        }
    }

代碼解釋:如果當前position位置的類型是TYPE_HEADER,也就是說用來顯示mHeaderView的,這裡我們就直接返回mHeaderView的佈局,不做事件處理了;如果不是,並且RecyclerView是有帶mHeaderView頭部的,那麼由於它占去第0個itemView,所以我們的position是從第一個開始計算的,所以我們必須得到當前真實position位置,並通過position位置來獲取當前的真實數據,如果不帶mHeaderView頭部,則可直接根據position獲取顯示數據,其他的邏輯不變。

還有註意的是當mHeaderView不為空時,我們的數據量大小也有一定的變化,請看:

    @Override
    public int getItemCount() {
        return mHeaderView != null ? mDatas.size() + 1 : mDatas.size();
    }

ok,在RecycerActivity的onCreate方法中添加一下兩句:

View headerView = LayoutInflater.from(this).inflate(R.layout.item_view1, null);
mBaseRecyclerAdapter.addHeaderView(headerView);

來看看運行結果吧

這裡寫圖片描述

ok,已經完成了mHeaderView 的添加。但是有個小問題,當你把RecyclerView的佈局設置為GridLayoutManager時,如:mRecyclerView.setLayoutManager(new GridLayoutManager(this,2));就會出現這種情況:

這裡寫圖片描述

這種情況也很好解決,在GridLayoutManager中我們可以在SpanSizeLookup中重新設置顯示的列數。

  @Override
    public void onAttachedToRecyclerView(RecyclerView recyclerView) {
        super.onAttachedToRecyclerView(recyclerView);
        RecyclerView.LayoutManager manager = recyclerView.getLayoutManager();
        if(manager instanceof GridLayoutManager){
            final GridLayoutManager gridLayoutManager = ((GridLayoutManager) manager);
            gridLayoutManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
                @Override
                public int getSpanSize(int position) {
                    //是HeaderView則占所有列,否則只占自己列
                    return getItemViewType(position) == TYPE_HEADER ? gridLayoutManager.getSpanCount() : 1;
                }
            });
        }
    }

最主要的就是在getSpanSize方法中根據我們的需要重新設置就ok了。看看吧
這裡寫圖片描述

好了,mHeaderView 已基本搞定,現在來看看怎麼添加FooterView了,其實原理是一樣的,也是根據getItemViewType()返回的ViewType類型來載入不同的佈局了。

首先我們需要定義一個內部類FooterViewHolder繼承我們的BaseViewHolderHelper,它主要是用來綁定FooterView佈局文件:

    private class FooterViewHolder extends BaseViewHolderHelper{
        private TextView footView;
        public FooterViewHolder(View itemView) {
            super(itemView);
            footView = (TextView) itemView.findViewById(R.id.tv_addFooter);
        }
    }

然後在getItemViewType中獲取到最後的ItemView的位置並返回TYPE_FOOTER類型:

 @Override
    private View mFooterView;
    public final static int TYPE_FOOTER = 2;
    ......
    
    public int getItemViewType(int position) {
        if(position + 1 == getItemCount()){
            return TYPE_FOOTER;
        }
        if(mHeaderView == null)
            return TYPE_BODY;
        if(position == 0) {
            return TYPE_HEADER;
        }
        return TYPE_BODY;
    }

另外在getItemCount()方法中我們因為添加個一個FooterView所以需要在原來的基礎上再加 1 ;

    @Override
    public int getItemCount() {
        return mHeaderView != null ? mDatas.size() + 2 : mDatas.size() + 1;
    }

再次在onCreateViewHolder方法中根據類型載入不同的佈局文件:

@Override
    public BaseViewHolderHelper onCreateViewHolder(ViewGroup parent, int viewType) {
        ...
        
        if(viewType == TYPE_FOOTER){
            mFooterView = LayoutInflater.from(mContext).inflate(R.layout.custom_footerview, parent,false);
            return new FooterViewHolder(mFooterView);
        }
        View view = LayoutInflater.from(parent.getContext()).inflate(mLayoutResId, parent, false);
        return new BaseViewHolderHelper(view);
    }

最後在onBindViewHolder方法中來展示我們的數據吧

    @Override
    public void onBindViewHolder(BaseViewHolderHelper holder, int position) {
        if(getItemViewType(position) == TYPE_HEADER){
            return;
        }else if(getItemViewType(position) == TYPE_FOOTER){
            FooterViewHolder footViewHolder=(FooterViewHolder)holder;
            footViewHolder.footView.setText("上拉載入更多...");
        } else{
            ...
        }
    }

ok,完成,來看看結果吧

這裡寫圖片描述

總結下,在給RecyclerView添加HeaderView和FooterView時,只要利用好getItemViewType這個方法,返回相對應的ViewType,並且在onCreateViewHolder方法中根據ViewType類型載入不同的佈局就完全可是實現我們的需求了。說起來就是這麼簡單。好了,今天就講到這裡吧,祝大家學習愉快。

更多資訊請關註微信平臺,有博客更新會及時通知。愛學習愛技術。

這裡寫圖片描述


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title></title> <style type="text/css"> #table { width: 400px; border-collapse: collapse; } </sty ...
  • 接觸面向對象許久了,對於繼承這一塊也確實琢磨了一段時間,搜集各種資料和網站,也未能得到使自己滿意的,並能快速理解的繼承知識,於是對他們歸類,並得出自己的總結。 先說說繼承是什麼吧?也許我們最早接觸有關繼承的應該是“遺產”??也許電視劇看多了,總有家族為了這玩意兒整的你死我活,確實聽看不下去的哈,但是 ...
  • 前言:今天想寫個頁面常用到的【點擊回到頁面頂部或是首頁的功能】,生活和職場一樣,總會有低谷的時候,這個時候咱也別慫、別慫、別慫,說三遍!那都不是事,工作沒了,再找唄,就像我上周五,團隊解散那天,我是笑著走的,還給小白揮了揮手,微笑一個。那句話叫什麼來著,佛祖雖給你關了一扇門,說不定會再給你開一扇窗。 ...
  • 最近在重新學習JQuery 轉載自http://www.cnblogs.com/onlys/articles/jQuery.html 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 ...
  • Bootstrap按鈕學習筆記,包含按鈕的樣式,按鈕的大小及狀態 ...
  • {“ web前端開發 ”是什麼? } {“ web前端開發 ”有前途嗎?}{“ web前端開發 ”到底怎麼學?}這3個對象,是你入【前端】的初戀,對!沒錯同時和 3個對象談戀愛。 web 前端篇:web前端到底是什麼?有前途嗎 ?請聽:微信jingfeng18分享 NO 1: web前端開發 ”是什 ...
  • 居中佈局 <div class=”parent”> <div class=”child”>demo</div> </div> 1.水平居中 1> 方案一 inlink-block+text-align .child {display:inlink-block;} .parent {text-alig ...
  • 設置導航欄nav全透明 - (void)viewDidLoad {裡面添加 swift:版本 // 1、設置視圖背景顏色 // self.view.backgroundColor = UIColor(white: 0.25, alpha: 1.0) // // // 2、設置導航欄標題屬性:設置標題 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...