RecyclerView GridLayoutManager 添加分割線

本著授人以魚嘹朗,不如授人以漁的道理蔬啡,從源頭出發(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茨(其實很多不會的東西,我都是這樣一步一步摸索著過來的锌介,擼一段嗜诀,看看效果,再改一改孔祸,再看看效果)隆敢;


微信圖片_20190222173945.jpg

可以看到,界面邊緣有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
效果圖如下


b9325eac97ea080231e9b991e1efa2f.jpg

完事了嗎?沒有痹束,做到這里检疫,我們也基本明白了該如何添加分割線,但是在使用中會發(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);
    }

}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末愕秫,一起剝皮案震驚了整個濱河市慨菱,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌戴甩,老刑警劉巖符喝,帶你破解...
    沈念sama閱讀 206,214評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異甜孤,居然都是意外死亡协饲,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評論 2 382
  • 文/潘曉璐 我一進(jìn)店門缴川,熙熙樓的掌柜王于貴愁眉苦臉地迎上來茉稠,“玉大人,你說我怎么就攤上這事把夸《撸” “怎么了?”我有些...
    開封第一講書人閱讀 152,543評論 0 341
  • 文/不壞的土叔 我叫張陵恋日,是天一觀的道長膀篮。 經(jīng)常有香客問我,道長岂膳,這世上最難降的妖魔是什么誓竿? 我笑而不...
    開封第一講書人閱讀 55,221評論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮谈截,結(jié)果婚禮上烤黍,老公的妹妹穿的比我還像新娘。我一直安慰自己傻盟,他們只是感情好速蕊,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著娘赴,像睡著了一般规哲。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上诽表,一...
    開封第一講書人閱讀 49,007評論 1 284
  • 那天唉锌,我揣著相機(jī)與錄音,去河邊找鬼竿奏。 笑死袄简,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的泛啸。 我是一名探鬼主播绿语,決...
    沈念sama閱讀 38,313評論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了吕粹?” 一聲冷哼從身側(cè)響起种柑,我...
    開封第一講書人閱讀 36,956評論 0 259
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎匹耕,沒想到半個月后聚请,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,441評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡稳其,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評論 2 323
  • 正文 我和宋清朗相戀三年驶赏,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片既鞠。...
    茶點(diǎn)故事閱讀 38,018評論 1 333
  • 序言:一個原本活蹦亂跳的男人離奇死亡母市,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出损趋,到底是詐尸還是另有隱情患久,我是刑警寧澤,帶...
    沈念sama閱讀 33,685評論 4 322
  • 正文 年R本政府宣布浑槽,位于F島的核電站蒋失,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏桐玻。R本人自食惡果不足惜篙挽,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望镊靴。 院中可真熱鬧冷冗,春花似錦言津、人聲如沸缎玫。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽踊谋。三九已至蝉仇,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間殖蚕,已是汗流浹背轿衔。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留睦疫,地道東北人害驹。 一個月前我還...
    沈念sama閱讀 45,467評論 2 352
  • 正文 我出身青樓,卻偏偏與公主長得像蛤育,于是被迫代替她去往敵國和親宛官。 傳聞我的和親對象是個殘疾皇子葫松,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評論 2 345

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