Android實戰(zhàn)RecyclerView頭部尾部添加方法示例
最近開啟SDK Manager,突然發(fā)現(xiàn)android7.0的都有了,這迭代升級還真快。不過國內(nèi)普遍手機還是停留在4.4+,多則是是處于5.0版本的。Android5.0變化非常大,引入material design,加強權(quán)限管理、減少功耗...好像扯遠了0 0?,F(xiàn)在直接進入主題。在這里先感謝讀者的支持!!
ListView是有addHeaderView和 addFooterView兩個方法的.
但是作為官方推薦的ListView的升級版RecyclerView缺無法實現(xiàn)這兩個方法。
那么如果使用RecyclerView實現(xiàn)這兩個方法的效果該怎么做呢?
網(wǎng)上查詢了很久,試過各種各樣的實現(xiàn)方式,終于讓我發(fā)現(xiàn)一個還不錯的實現(xiàn)方法,那么就給大家推薦一下。
筆者前陣子寫了一個萬能適配器,提供了上拉加載、上拉刷新的基礎(chǔ)功能,重要的是一個基礎(chǔ)baseAdapter能夠支持ListView與RecyclerView,后期提供傳送門,現(xiàn)在我打算一步驟一步驟講下我的實現(xiàn)思路。
實戰(zhàn)RecyclerView頭部尾部添加方法
效果圖如下:
一、前提
首先ListView與RecyclerView兩者非常相似,兩者提供view都是依賴適配器。只不過就是5.0版本推出RecyclerView后,Google將adapter和viewHolder做了一系列的優(yōu)化和封裝。不像之前為了復用Listview里面的converView,要類似在getView里面實現(xiàn)下列的代碼:
上面代碼看起來挺眼熟吧~
二、對比RecyclerView,google進行的優(yōu)化
在RecyclerView依賴的適配器中,無論是適配器還是ViewHolder,從源碼我們可以看出,都存在RecyclerView的匿名內(nèi)部類。相對于Listview,RecyclerView內(nèi)置了多級緩存、RecyclerViewPool(從線程的角度,可以理解成類似線程池的東西,即多個RecyclerView可以公用一個view)、ViewHolder(已經(jīng)實現(xiàn)了復用,相對于Listview的BaseAdapter中g(shù)etView方法需要開發(fā)者自己引入復用問題方便很多)等等。這里我們簡單說下兩個方法:
public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType)
public void onBindViewHolder(ViewHolder holder, int position)
在以前的BaseAdapter中,所有視圖加載、數(shù)據(jù)綁定以及復用,都需要我們直接在getView里面進行操作。onCreateViewHolder負責視圖加載并且內(nèi)部完成復用,onBindViewHolder負責數(shù)據(jù)綁定并且內(nèi)部完成一系列的緩存機制。這里滿足了視圖層與邏輯層的分離,典型的mvp模式。
三、RecyclerView的頭部與尾部實現(xiàn)
RecyclerView不像ListView擁有addHeaderView()與addFooterView()的方法簡單添加頭部尾部即可,而且RecyclerView也沒有像ListView的列表點擊監(jiān)聽方法(setItemOnclickListener),這里我也不明白為什么官方會取消了這些獨有的屬性,不過我們依然可以在onBindViewHolder方法中進行事件綁定!
具體頭部與尾部實現(xiàn)方法,這里有個訣竅,這里先看一個方法:
public int getItemViewType(int position)
getItemViewType方法是在執(zhí)行onCreateViewHolder(ViewGroup parent, int viewType)前回調(diào)用viewType,目的是為了根據(jù)viewType不同創(chuàng)建不同的視圖。我們可以通過在onCreateViewHolder創(chuàng)建視圖的時候,對viewType進行判斷,如果添加了頭部,在position = 0的時候回調(diào)頭部的viewType給onCreateViewHolder,從而創(chuàng)建頭部。尾部創(chuàng)建方法于此類同,直接看下代碼,適配器的實現(xiàn):
package cn.wsy.recyclerdemo; import android.content.Context; import android.support.v7.widget.GridLayoutManager; import android.support.v7.widget.RecyclerView; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.TextView; import java.util.ArrayList; import java.util.List; /** * Created by wsy on 2016/8/4. */ public class MyAdapter extends RecyclerView.Adapter<MyAdapter.MyHolder> { private RecyclerView mRecyclerView; private List<String> data = new ArrayList<>(); private Context mContext; private View VIEW_FOOTER; private View VIEW_HEADER; //Type private int TYPE_NORMAL = 1000; private int TYPE_HEADER = 1001; private int TYPE_FOOTER = 1002; public MyAdapter(List<String> data, Context mContext) { this.data = data; this.mContext = mContext; } @Override public MyAdapter.MyHolder onCreateViewHolder(ViewGroup parent, int viewType) { if (viewType == TYPE_FOOTER) { return new MyHolder(VIEW_FOOTER); } else if (viewType == TYPE_HEADER) { return new MyHolder(VIEW_HEADER); } else { return new MyHolder(getLayout(R.layout.item_list_layout)); } } @Override public void onBindViewHolder(MyHolder holder, int position) { if (!isHeaderView(position) && !isFooterView(position)) { if (haveHeaderView()) position--; TextView content = (TextView) holder.itemView.findViewById(R.id.item_content); TextView time = (TextView) holder.itemView.findViewById(R.id.item_time); content.setText(data.get(position)); time.setText("2016-1-1"); } } @Override public int getItemCount() { int count = (data == null ? 0 : data.size()); if (VIEW_FOOTER != null) { count++; } if (VIEW_HEADER != null) { count++; } return count; } @Override public int getItemViewType(int position) { if (isHeaderView(position)) { return TYPE_HEADER; } else if (isFooterView(position)) { return TYPE_FOOTER; } else { return TYPE_NORMAL; } } @Override public void onAttachedToRecyclerView(RecyclerView recyclerView) { try { if (mRecyclerView == null && mRecyclerView != recyclerView) { mRecyclerView = recyclerView; } ifGridLayoutManager(); } catch (Exception e) { e.printStackTrace(); } } private View getLayout(int layoutId) { return LayoutInflater.from(mContext).inflate(layoutId, null); } public void addHeaderView(View headerView) { if (haveHeaderView()) { throw new IllegalStateException("hearview has already exists!"); } else { //避免出現(xiàn)寬度自適應(yīng) ViewGroup.LayoutParams params = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT); headerView.setLayoutParams(params); VIEW_HEADER = headerView; ifGridLayoutManager(); notifyItemInserted(0); } } public void addFooterView(View footerView) { if (haveFooterView()) { throw new IllegalStateException("footerView has already exists!"); } else { ViewGroup.LayoutParams params = new ViewGroup.LayoutParams(ViewGroup.LayoutParams.MATCH_PARENT, ViewGroup.LayoutParams.WRAP_CONTENT); footerView.setLayoutParams(params); VIEW_FOOTER = footerView; ifGridLayoutManager(); notifyItemInserted(getItemCount() - 1); } } private void ifGridLayoutManager() { if (mRecyclerView == null) return; final RecyclerView.LayoutManager layoutManager = mRecyclerView.getLayoutManager(); if (layoutManager instanceof GridLayoutManager) { final GridLayoutManager.SpanSizeLookup originalSpanSizeLookup = ((GridLayoutManager) layoutManager).getSpanSizeLookup(); ((GridLayoutManager) layoutManager).setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() { @Override public int getSpanSize(int position) { return (isHeaderView(position) || isFooterView(position)) ? ((GridLayoutManager) layoutManager).getSpanCount() : 1; } }); } } private boolean haveHeaderView() { return VIEW_HEADER != null; } public boolean haveFooterView() { return VIEW_FOOTER != null; } private boolean isHeaderView(int position) { return haveHeaderView() && position == 0; } private boolean isFooterView(int position) { return haveFooterView() && position == getItemCount() - 1; } public static class MyHolder extends RecyclerView.ViewHolder { public MyHolder(View itemView) { super(itemView); } } }
四、實現(xiàn)方法
簡單的初始化RecycerView,以及設(shè)置適配器,如下:
private void initRecyc() { // mRecyclerView.setLayoutManager(new GridLayoutManager(this,2)); mRecyclerView.setLayoutManager(new LinearLayoutManager(this)); adapter = new MyAdapter(data, this); mRecyclerView.setAdapter(adapter); adapter.addFooterView(LayoutInflater.from(this).inflate(R.layout.item_footer_layout,null)); adapter.addHeaderView(LayoutInflater.from(this).inflate(R.layout.item_header_layout,null)); }
五、注意的問題
筆者在添加頭部尾部的時候,發(fā)現(xiàn)在配置RecyclerView,如果模式是配置GridLayoutManager的時候,發(fā)現(xiàn)頭部會跑到第一格,也就是不是自己想要獨立一行的效果,這里貼上關(guān)鍵代碼,可以解決(簡單數(shù)學問題啦哈~):
private void ifGridLayoutManager() { if (mRecyclerView == null) return; final RecyclerView.LayoutManager layoutManager = mRecyclerView.getLayoutManager(); if (layoutManager instanceof GridLayoutManager) { final GridLayoutManager.SpanSizeLookup originalSpanSizeLookup = ((GridLayoutManager) layoutManager).getSpanSizeLookup(); ((GridLayoutManager) layoutManager).setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() { @Override public int getSpanSize(int position) { return (isHeaderView(position) || isFooterView(position)) ? ((GridLayoutManager) layoutManager).getSpanCount() : 1; } }); } }
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
android實現(xiàn)程序自動升級到安裝示例分享(下載android程序安裝包)
這篇文章主要介紹了android實現(xiàn)下載android程序安裝包自動升級的示例,大家參考使用吧2014-01-01Android自定義View實現(xiàn)shape圖形繪制
這篇文章主要為大家詳細介紹了Android使用自定義View實現(xiàn)shape圖形繪制,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-01-01Android 使用ViewPager實現(xiàn)圖片左右循環(huán)滑動自動播放
這篇文章主要介紹了Android 使用ViewPager實現(xiàn)圖片左右循環(huán)滑動自動播放的相關(guān)資料,非常不錯,具有參考解決價值,需要的朋友可以參考下2016-08-08淺談android性能優(yōu)化之啟動過程(冷啟動和熱啟動)
本篇文章主要介紹了淺談android性能優(yōu)化之啟動過程(冷啟動和熱啟動) ,具有一定的參考價值,感興趣的小伙伴們可以參考一下。2017-08-08