前言
關(guān)于adapter的封裝煤禽,網(wǎng)上有很多開(kāi)源庫(kù)液样,開(kāi)發(fā)的時(shí)候可以直接拿來(lái)用振亮,省了很多事。
最近閑來(lái)無(wú)事鞭莽,想著自己動(dòng)手封裝一個(gè)adapter坊秸,主要參考了這篇文章:http://www.reibang.com/p/3e60147523e1 感謝
問(wèn)題
-
通常我們封裝的時(shí)候,可以簡(jiǎn)化到這一步:
BaseRecyclerViewAdapter adapter = new BaseRecyclerViewAdapter() { private static final int TYPE_FIR = 1; private static final int TYPE_SEC = 2; private static final int TYPE_THR = 3; @Override public int getLayoutId(int viewType) { if (viewType == TYPE_FIR) { return R.layout.test_activity_recyclerview_item; } else if (viewType == TYPE_SEC){ return R.layout.test_activity_recyclerview_item_two; } else { return R.layout.test_activity_recyclerview_item_three; } } @Override public int getItemViewType(int position) { if (position % 3 == 1) { return TYPE_FIR; } else if (position % 3 == 2) { return TYPE_SEC; } else { return TYPE_THR; } } @Override public void onBindRecyclerViewHolder(BaseViewHolder holder, int position) { if (getItemVIewType(position) == TYPE_FIR) { // TODO 數(shù)據(jù)處理及綁定 } else if (getItemVIewType(position) == TYPE_SEC) { // TODO 數(shù)據(jù)處理及綁定 } else { // TODO 數(shù)據(jù)處理及綁定 } } };
-
從上面代碼我們可以看到澎怒,當(dāng)處理多布局類(lèi)型的時(shí)候褒搔,我們需要解決的是:
- 多布局類(lèi)型定義
- 返回什么類(lèi)型處理
- 根據(jù)指定類(lèi)型,加載對(duì)應(yīng)布局Layout
- 根據(jù)類(lèi)型喷面,處理及綁定數(shù)據(jù)
當(dāng)我們迭代更新的時(shí)候星瘾,我們只能通過(guò)修改adapter,這樣耦合性稍微有點(diǎn)強(qiáng)乖酬。
不過(guò)呢死相,簡(jiǎn)化到這一步的時(shí)候,其實(shí)在項(xiàng)目中是可以使用的
解決
上面的問(wèn)題咬像,我們知道算撮,如果需要迭代更新,我們只能通過(guò)修改viewType县昂、layout肮柜、onBind等具體數(shù)據(jù)來(lái)實(shí)現(xiàn)。既然這樣倒彰,我們可以將這些具體數(shù)據(jù)抽取封裝起來(lái)审洞,adapter不直接處理數(shù)據(jù),而是交給這個(gè)封裝類(lèi),邏輯交給封裝類(lèi)來(lái)處理芒澜,這樣adapter只需要負(fù)責(zé)維護(hù)這個(gè)封裝類(lèi)即可仰剿。
-
這個(gè)封裝類(lèi),我們暫且稱(chēng)作Item
interface Item { @LayoutRes int getLayoutResource(); int getItemViewType(); BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType); void onBindViewHolder(BaseViewHolder holder, int position); }
adpater中痴晦,需要?jiǎng)討B(tài)調(diào)整的就是
- viewType
- layout布局
- onCreateViewHolder
- onBindViewHolder
所以就將這些方法抽取封裝起來(lái)南吮,然后adapter維護(hù)Item列表即可
上面這個(gè)接口Item
第一:沒(méi)有數(shù)據(jù),我們需要自己在實(shí)現(xiàn)類(lèi)中提供數(shù)據(jù)設(shè)置獲取方法誊酌;
第二:我們需要同時(shí)實(shí)現(xiàn)四個(gè)方法
既然這樣部凑,我們可以通過(guò)一個(gè)基類(lèi)BaseItem,來(lái)簡(jiǎn)化操作-
在Item接口中碧浊,只有onCreateViewHolder()方法涂邀,是不需要每次都實(shí)現(xiàn)的,因?yàn)樗枪潭ú蛔兊牡南淙瘢珺aseItem就可以這樣寫(xiě):
public abstract class BaseItem<T> implements Item { public T mData; public void setData(T t) { this.mData = t; } @Override public BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { int resourceId = getLayoutResource(); View itemView = LayoutInflater.from(parent.getContext()).inflate(resourceId, parent, false); return new BaseViewHolder(itemView); } }
-
現(xiàn)在比勉,我們只需要實(shí)現(xiàn)三個(gè)方法就可以了。我們?cè)倏磄etItemViewType瑞躺,這個(gè)返回的是當(dāng)前布局類(lèi)型敷搪,int類(lèi)型,唯一且不能重復(fù) 幢哨。Android中的Layout維護(hù)著一個(gè)int型的唯一標(biāo)識(shí)ID赡勘,我們是不是可以拿這個(gè)標(biāo)識(shí)ID,作為viewType捞镰?這樣就可以進(jìn)一步簡(jiǎn)化實(shí)現(xiàn)類(lèi)闸与,現(xiàn)在我們只需實(shí)現(xiàn)兩個(gè)方法即可。
public abstract class BaseItem<T> implements Item { public T mData; public void setData(T t) { this.mData = t; } @Override public int getItemViewType() { return getLayoutResource(); } @Override public BaseViewHolder onCreateViewHolder(ViewGroup parent, int viewType) { int resourceId = getLayoutResource(); View itemView = LayoutInflater.from(parent.getContext()).inflate(resourceId, parent, false); return new BaseViewHolder(itemView); } }
-
接下來(lái)岸售,我們要構(gòu)造Adapter践樱,使之不再維護(hù)具體數(shù)據(jù),而是BaseItem列表
public class BaseAdapter extends RecyclerView.Adapter<BaseViewHolder> { private List<BaseItem> mData; public BaseRecyclerViewAdapter(List<BaseItem> data) { mData = new ArrayList<>(); if (data != null) { mData.addAll(data); } } @Override public BaseViewHolder onCreateViewHolder(ViewGroup parent, final int viewType) { for (BaseItem item : mData) { if (viewType == item.getItemViewType()) { return item.onCreateViewHolder(parent, viewType);; } } } @Override public void onBindViewHolder(BaseViewHolder holder, int position) { mData.get(dataPosition).onBindViewHolder(holder, position); } @Override public int getItemCount() { return mData.size(); } @Override public int getItemViewType(int position) { return mData.get(position).getItemViewType(); } }
結(jié)語(yǔ)
以上簡(jiǎn)單介紹了一下封裝思路凸丸,至于像頭布局拷邢、腳布局、空布局屎慢、加載更多布局等等瞭稼,其實(shí)都是Adapter中的一種vietType布局,具體實(shí)現(xiàn)代碼腻惠,還請(qǐng)有勞移駕到GitHub:BaseAdapter查看环肘。歡迎star,提意見(jiàn)集灌。
下圖是一adapter中數(shù)據(jù)構(gòu)造: