本著授人以魚嘹朗,不如授人以漁的道理蔬啡,從源頭出發(fā)诲侮,為大家說明RecyclerView在多列顯示(多行同理)的情況下,如何添加分割線箱蟆;
首先沟绪,要想給RecyclerView添加分隔線,需要調(diào)用
RecyclerView.addItemDecoration(ItemDecoration decor);
這里講的就是空猜,如何定義我們需要的 ItemDecoration
萬變不離其宗
繼承
創(chuàng)建一個類绽慈,繼承 RecyclerView.ItemDecoration,并重寫第一個關(guān)鍵方法
public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state)
public class RecycleGridDivider extends RecyclerView.ItemDecoration{
@Override
public void getItemOffsets(Rect outRect, View view,
RecyclerView parent, RecyclerView.State state) {
}
}
從字面上我們理解這個方法的簡單含義
getItemOffsets
我習(xí)慣性的拆分理解
獲取辈毯、列表項坝疼、偏移量、多個
那么我初步認(rèn)為這個方法是拿到所有列表項的偏移量
所以我對方法內(nèi)的Rect(經(jīng)常自定義View的朋友應(yīng)該知道谆沃,這個說白了揍是個區(qū)域钝凶,定義上下左右的邊界)進(jìn)行各種設(shè)定參數(shù);
擼了這么一段代碼
outRect.set(10,10,10,10);
至于什么效果唁影,有沒有作用耕陷,管它呢,跑一遍程序看看長啥樣据沈,試試又不會懷孕S茨(其實很多不會的東西,我都是這樣一步一步摸索著過來的锌介,擼一段嗜诀,看看效果,再改一改孔祸,再看看效果)隆敢;
可以看到,界面邊緣有10像素的分割線融击,Item中間是20像素
ps(我說是這么多就是這么多爸雳窟!你別跟我杠)
為什么中間是20呢尊浪?因為有左右兩個Item分別給了10像素
假如這正是你想要的效果,那么很好封救,差不多是一行代碼搞定拇涤,但很遺憾的,這不是我想要的效果誉结,我的RecyclerView需要設(shè)置Padding鹅士,同時,我的Item兩邊不需要分割線惩坑,怎么搞定呢掉盅?
我想到了for循環(huán)也拜,你別管對不對,想到就干趾痘,就擼慢哈,您猜怎么著?
有作用永票!
這時就用到了 getItemOffsets 方法中的第三個參數(shù) RecyclerView
RecyclerView.getChildCount();獲取Item數(shù)量
RecyclerView.getLayoutManager();轉(zhuǎn)成GridLayoutManager
GridLayoutManager.getSpanCount();獲取列數(shù)/行數(shù)
多少列有了卵贱,多少個Item有了,盤他 侣集!輸入 “fori”回車键俱!
GridLayoutManager manager = (GridLayoutManager) parent.getLayoutManager();
int childSize = parent.getChildCount();
int span = manager.getSpanCount();
for (int i = 0; i < childSize; i++) {
}
遍歷了,開始盤他啊世分,怎么盤呢编振?廢話,當(dāng)然是根據(jù)自己需求來啦臭埋!
思路
靠邊的Item党觅,邊上不加偏移量
判斷Item是不是第一行
計算Item是在左邊還是右邊
腦瓜子轉(zhuǎn)轉(zhuǎn),怎么判斷他是第一行呢斋泄?
//為了Item大小均勻杯瞻,將設(shè)定分割線平均分給左右兩邊Item各一半
int offset = space / 2;
for (int i = 0; i < childSize; i++) {
//第一排,頂部不畫
if (i < span) {
//最左邊的炫掐,左邊不畫
if (i % span == 0) {
outRect.set(0, 0, offset, 0);
//最右邊魁莉,右邊不畫
} else if (i % span == span - 1) {
outRect.set(offset, 0, 0, 0);
} else {
outRect.set(offset, 0, offset, 0);
}
} else {
//上下的分割線,就從第二排開始募胃,每個區(qū)域的頂部直接添加設(shè)定大小旗唁,不用再均分了
if (i % span == 0) {
outRect.set(0, space, offset, 0);
} else if (i % span == span - 1) {
outRect.set(offset, space, 0, 0);
} else {
outRect.set(offset, space, offset, 0);
}
}
}
同時給RecyclerView加了Padding 15dp
設(shè)置分割線 20dp
效果圖如下
完事了嗎?沒有痹束,做到這里检疫,我們也基本明白了該如何添加分割線,但是在使用中會發(fā)現(xiàn) getItemOffsets 是多次執(zhí)行的祷嘶,它本身就遍歷了RecycleView屎媳,如果我們再次遍歷,很不友好论巍,效率也不高烛谊,這時候就又用到了getItemOffsets的第二個參數(shù) View,根據(jù)經(jīng)驗嘉汰,用屁股想想都知道丹禀,這個極有可能就是單個子Item,繼續(xù)盤它
//得到View的位置
int childPosition = parent.getChildAdapterPosition(view);
利用獲取的childPosition 代替for循環(huán)內(nèi)的下標(biāo),并干掉for循環(huán)
//得到View的位置
int childPosition = parent.getChildAdapterPosition(view);
//第一排双泪,頂部不畫
if (childPosition < span) {
//最左邊的持搜,左邊不畫
if (childPosition % span == 0) {
outRect.set(0, 0, offset, 0);
//最右邊,右邊不畫
} else if (childPosition % span == span - 1) {
outRect.set(offset, 0, 0, 0);
} else {
outRect.set(offset, 0, offset, 0);
}
} else {
//上下的分割線焙矛,就從第二排開始朵诫,每個區(qū)域的頂部直接添加設(shè)定大小,不用再均分了
if (childPosition % span == 0) {
outRect.set(0, space, offset, 0);
} else if (childPosition % span == span - 1) {
outRect.set(offset, space, 0, 0);
} else {
outRect.set(offset, space, offset, 0);
}
}
根據(jù)需求薄扁,將Item區(qū)域均勻分配剪返,如果需求是垂直方向的分割線和水平方向分割線規(guī)格不一樣,就分別定義吧邓梅!
至于分割線顏色脱盲,需要重寫 onDraw 來自己畫,我們上一步已經(jīng)把分割線區(qū)域搞定了(此時看上去是透明的日缨,實際是壓根就沒有分割線的)
怎么畫分割線钱反?通過計算上一步Item的區(qū)域(也就是大小)以及其他Padding 或者 Margin匣距,剩下的區(qū)域進(jìn)行繪制面哥,同時也可以繪制特定圖形Drawable,大概思路就是這樣毅待,分割線的繪制抽空單獨(dú)再寫一份經(jīng)驗貼尚卫;
分割線經(jīng)驗貼連接《RecyclerView之自定義分割線onDraw的用法》
我是Android界的小學(xué)生,經(jīng)驗分享的同時也是一份記錄尸红,俗話說好記性不如爛筆頭吱涉,保不齊哪天我給忘了,翻翻就想起來了外里,寫的不好怎爵,或者您有更好的實現(xiàn)方式,請及時盤我盅蝗,我好給你盤圓溜滴鳖链,嘿嘿嘿!下面附上完整的代碼
/**
* Created by dzh on 02.22.022.
* 網(wǎng)格布局分割線
*/
public class RecycleGridDivider extends RecyclerView.ItemDecoration {
private int space;
private int color;
private Paint mPaint;
/**
* 默認(rèn)的墩莫,垂直方向 橫縱1px 的分割線 顏色透明
*/
public RecycleGridDivider() {
this(1);
}
/**
* 自定義寬度的透明分割線
*
* @param space 指定寬度
*/
public RecycleGridDivider(int space) {
this(space, Color.TRANSPARENT);
}
/**
* 自定義寬度芙委,并指定顏色的分割線
*
* @param space 指定寬度
* @param color 指定顏色
*/
public RecycleGridDivider(int space, int color) {
this.space = space;
this.color = color;
initPaint();
}
private void initPaint() {
mPaint = new Paint(Paint.ANTI_ALIAS_FLAG);
mPaint.setColor(color);
mPaint.setStyle(Paint.Style.FILL);
mPaint.setStrokeWidth(space);
}
@Override
public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
GridLayoutManager manager = (GridLayoutManager) parent.getLayoutManager();
int childSize = parent.getChildCount();
int span = manager.getSpanCount();
//為了Item大小均勻,將設(shè)定分割線平均分給左右兩邊Item各一半
int offset = space / 2;
//得到View的位置
int childPosition = parent.getChildAdapterPosition(view);
//第一排贼穆,頂部不畫
if (childPosition < span) {
//最左邊的题山,左邊不畫
if (childPosition % span == 0) {
outRect.set(0, 0, offset, 0);
//最右邊兰粉,右邊不畫
} else if (childPosition % span == span - 1) {
outRect.set(offset, 0, 0, 0);
} else {
outRect.set(offset, 0, offset, 0);
}
} else {
//上下的分割線故痊,就從第二排開始,每個區(qū)域的頂部直接添加設(shè)定大小玖姑,不用再均分了
if (childPosition % span == 0) {
outRect.set(0, space, offset, 0);
} else if (childPosition % span == span - 1) {
outRect.set(offset, space, 0, 0);
} else {
outRect.set(offset, space, offset, 0);
}
}
}
@Override
public void onDraw(Canvas c, RecyclerView parent, RecyclerView.State state) {
super.onDraw(c, parent, state);
}
}