【轉(zhuǎn)載】為什么要使用ItemDecoration

為什么要使用ItemDecoration
泡在網(wǎng)上的日子 / 文 發(fā)表于2017-06-22 17:20 第277次閱讀 ItemDecoration,RecyclerView
0
編輯推薦:稀土掘金,這是一個(gè)針對(duì)技術(shù)開(kāi)發(fā)者的一個(gè)應(yīng)用,你可以在掘金上獲取最新最優(yōu)質(zhì)的技術(shù)干貨磷蜀,不僅僅是Android知識(shí)搔确、前端、后端以至于產(chǎn)品和設(shè)計(jì)都有涉獵荚孵,想成為全棧工程師的朋友不要錯(cuò)過(guò)!
原文:ItemDecoration in Android Part 1 : Avoid adding dividers to the view layout
Part 1:不要用view做分割線

1498123313800821.jpeg

首先,什么是ItemDecoration戴而?來(lái)看看官網(wǎng)是如何解釋的。
ItemDecoration允許從adapter的數(shù)據(jù)集合中為特定的item視圖添加特性的繪制以及布局間隔翩蘸。它可以用來(lái)實(shí)現(xiàn)item之間的分割線所意,高亮,分組邊界等催首。
我們不能簡(jiǎn)單的把ItemDecoration看成一個(gè)名字響亮的分割線扶踊。它比divider要多很多內(nèi)容。一個(gè)divider只能繪制在item之間郎任,但是ItemDecoration可以繪制在item的四邊秧耗。ItemDecoration為decoration的測(cè)量和繪制提供了全方位的控制。一個(gè)decoration可以是一條分割線舶治,也可以?xún)H僅是一個(gè)間隔(inset)分井。
但不幸的是车猬,絕大多數(shù)android開(kāi)發(fā)者都沒(méi)有使用item decoration。在這個(gè)分為三部分的系列文章中尺锚,我們將了解ItemDecoration的強(qiáng)大之處珠闰。
第一部分: 不要添加view來(lái)做分割線—?使用 ItemDecoration
第二部分: 不要使用padding來(lái)做間隔?—使用 ItemDecoration
第三部分: 在GridLayoutManager中高效的繪制decorations
本文是第一部分。
不要用view做分割線 —會(huì)影響性能
我曾看到一些開(kāi)發(fā)者在為RecyclerView添加divider的時(shí)候采用了一些捷徑瘫辩。原因很簡(jiǎn)單伏嗜,ListView原生支持divider,可以直接在xml中設(shè)置divider伐厌。
<ListView
android:id="@+id/activity_home_list_view"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:divider="@android:color/black"
android:dividerHeight="8dp"/>

但是到了RecyclerView承绸,就再也不能直接添加divider了。需要添加一個(gè)繪制divider的ItemDecoration弧械。但是開(kāi)發(fā)者發(fā)現(xiàn)它很麻煩八酒,于是直接把divider添加到(item的)view上,而不是使用ItemDecoration刃唐。
<LinearLayout android:orientation="vertical">
<LinearLayout android:orientation="horizontal">
<ImageView />
<TextView />
</LinearLayout>
<View
android:width="match_parent"
android:height="1dp"
android:background="#333" />
</LinearLayout>

每當(dāng)我們走捷徑的時(shí)候羞迷,都有可能會(huì)產(chǎn)生副作用。而這里的副作用是可能影響性能画饥。
當(dāng)在布局中添加了一個(gè)divider的時(shí)候衔瓮,我們?cè)黾恿藇iew的個(gè)數(shù)。我們都知道view的數(shù)目越少會(huì)得到越好的性能抖甘。有時(shí)候增加一個(gè)view來(lái)實(shí)現(xiàn)divider還會(huì)增加布局的層級(jí)热鞍。比如上面的例子中,我們不僅僅增加了一個(gè)view衔彻,還增加了一個(gè)包含它們的 linear layout薇宠。為了一個(gè)divider而創(chuàng)建了額外的布局。
不要用view做分割線?—會(huì)帶來(lái)副作用
因?yàn)閐ivider是view的一部分艰额,所以在item 動(dòng)畫(huà)期間澄港,divider也會(huì)一起跟著動(dòng)畫(huà)。如下圖:


1498117406107936.gif

顯然divider不應(yīng)該隨著item一起做動(dòng)畫(huà)柄沮。而是和item分開(kāi)回梧,像這樣才是對(duì)的:


1498120170236717.gif

不要用view做分割線—?缺乏靈活性
如果divider是(item的)view的一部分,那么你就無(wú)法控制它祖搓。你唯一能控制的就是根據(jù)item的position改變divider的可見(jiàn)狀態(tài)狱意。 而item decoration就靈活多了。
1498120202925785.png

In the above image for the last item in the group divider fills the entire width. Other dividers have a margin of 56dp to their left side. Here is the ItemDecorator’s onDraw code.
在上圖中拯欧,group最后一個(gè)item的divider充滿了整個(gè)寬度详囤。其它的divider都有一個(gè)56dp的左邊距。這是這個(gè)ItemDecorator的onDraw代碼:

@Override
public void onDraw(Canvas canvas, RecyclerView parent, RecyclerView.State state) {
canvas.save();
final int leftWithMargin = convertDpToPixel(56);
final int right = parent.getWidth();

final int childCount = parent.getChildCount();
for (int i = 0; i < childCount; i++) {
final View child = parent.getChildAt(i);
int adapterPosition = parent.getChildAdapterPosition(child);
left = (adapterPosition == lastPosition) ? 0 : leftWithMargin;
parent.getDecoratedBoundsWithMargins(child, mBounds);
final int bottom = mBounds.bottom + Math.round(ViewCompat.getTranslationY(child));
final int top = bottom - mDivider.getIntrinsicHeight();
mDivider.setBounds(left, top, right, bottom);
mDivider.draw(canvas);
}
canvas.restore();
}

不要用view做分割線—使用 ItemDecoration
寫(xiě)一個(gè)自己的ItemDecoration其實(shí)非常簡(jiǎn)單镐作。你只需要?jiǎng)?chuàng)建一個(gè)繼承了ItemDecoration的類(lèi)就可以了纬纪。重寫(xiě) getItemOffsets() 和 onDraw() 方法蚓再。具體實(shí)現(xiàn)可以參考 這個(gè) 示例。
而 25.0.0版本的支持庫(kù)中包各,我們有一個(gè)新的類(lèi) “DividerItemDecoration”摘仅。這個(gè)類(lèi)直接實(shí)現(xiàn)了divider。
DividerItemDecoration decoration = new DividerItemDecoration(getApplicationContext(), VERTICAL);
recyclerView.addItemDecoration(decoration);

提示
一個(gè)RecyclerView可以添加多個(gè)ItemDecoration问畅。發(fā)揮頭腦風(fēng)暴的時(shí)候到了娃属。

所有decoration都在item繪制之前繪制。如果你想讓decoration在view之后繪制护姆,重寫(xiě)onDrawOver() 而不是onDraw() 矾端。

所以下次想為RecyclerView添加分割線的時(shí)候,別使用在item布局添加view這種方式了卵皂,使用ItemDecoration秩铆。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市灯变,隨后出現(xiàn)的幾起案子殴玛,更是在濱河造成了極大的恐慌,老刑警劉巖添祸,帶你破解...
    沈念sama閱讀 206,839評(píng)論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件滚粟,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡刃泌,警方通過(guò)查閱死者的電腦和手機(jī)凡壤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,543評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)耙替,“玉大人亚侠,你說(shuō)我怎么就攤上這事∷咨龋” “怎么了硝烂?”我有些...
    開(kāi)封第一講書(shū)人閱讀 153,116評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)狐援。 經(jīng)常有香客問(wèn)我钢坦,道長(zhǎng)究孕,這世上最難降的妖魔是什么啥酱? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,371評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮厨诸,結(jié)果婚禮上镶殷,老公的妹妹穿的比我還像新娘。我一直安慰自己微酬,他們只是感情好绘趋,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,384評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布颤陶。 她就那樣靜靜地躺著,像睡著了一般陷遮。 火紅的嫁衣襯著肌膚如雪滓走。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 49,111評(píng)論 1 285
  • 那天帽馋,我揣著相機(jī)與錄音搅方,去河邊找鬼。 笑死绽族,一個(gè)胖子當(dāng)著我的面吹牛姨涡,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播吧慢,決...
    沈念sama閱讀 38,416評(píng)論 3 400
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼涛漂,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了检诗?” 一聲冷哼從身側(cè)響起匈仗,我...
    開(kāi)封第一講書(shū)人閱讀 37,053評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎岁诉,沒(méi)想到半個(gè)月后锚沸,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,558評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡涕癣,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,007評(píng)論 2 325
  • 正文 我和宋清朗相戀三年哗蜈,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片坠韩。...
    茶點(diǎn)故事閱讀 38,117評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡距潘,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出只搁,到底是詐尸還是另有隱情音比,我是刑警寧澤,帶...
    沈念sama閱讀 33,756評(píng)論 4 324
  • 正文 年R本政府宣布氢惋,位于F島的核電站洞翩,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏焰望。R本人自食惡果不足惜骚亿,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,324評(píng)論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望熊赖。 院中可真熱鬧来屠,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,315評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至迎膜,卻和暖如春泥技,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背磕仅。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,539評(píng)論 1 262
  • 我被黑心中介騙來(lái)泰國(guó)打工零抬, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人宽涌。 一個(gè)月前我還...
    沈念sama閱讀 45,578評(píng)論 2 355
  • 正文 我出身青樓平夜,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親卸亮。 傳聞我的和親對(duì)象是個(gè)殘疾皇子忽妒,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,877評(píng)論 2 345

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