Flutter 布局(十)- ListBody感凤、ListView陪竿、CustomMultiChildLayout詳解

本文主要介紹Flutter布局中的ListBody族跛、ListView礁哄、CustomMultiChildLayout控件溪北,詳細(xì)介紹了其布局行為以及使用場(chǎng)景之拨,并對(duì)源碼進(jìn)行了分析敦锌。

1. ListBody

A widget that arranges its children sequentially along a given axis.

1.1 簡(jiǎn)介

ListBody是一個(gè)不常直接使用的控件,一般都會(huì)配合ListView或者Column等控件使用颖变。ListBody的作用是按給定的軸方向腥刹,按照順序排列子節(jié)點(diǎn)衔峰。

1.2 布局行為

在主軸上垫卤,子節(jié)點(diǎn)按照順序進(jìn)行布局穴肘,在交叉軸上舔痕,子節(jié)點(diǎn)尺寸會(huì)被拉伸,以適應(yīng)交叉軸的區(qū)域邢笙。

在主軸上侍匙,給予子節(jié)點(diǎn)的空間必須是不受限制的(unlimited),使得子節(jié)點(diǎn)可以全部被容納筐骇,ListBody不會(huì)去裁剪或者縮放其子節(jié)點(diǎn)。

1.3 繼承關(guān)系

Object > Diagnosticable > DiagnosticableTree > Widget > RenderObjectWidget > MultiChildRenderObjectWidget > ListBody

1.4 示例代碼

Flex(
  direction: Axis.vertical,
  children: <Widget>[
    ListBody(
      mainAxis: Axis.vertical,
      reverse: false,
      children: <Widget>[
        Container(color: Colors.red, width: 50.0, height: 50.0,),
        Container(color: Colors.yellow, width: 50.0, height: 50.0,),
        Container(color: Colors.green, width: 50.0, height: 50.0,),
        Container(color: Colors.blue, width: 50.0, height: 50.0,),
        Container(color: Colors.black, width: 50.0, height: 50.0,),
      ],
  )],
)

1.5 源碼解析

構(gòu)造函數(shù)如下:

ListBody({
  Key key,
  this.mainAxis = Axis.vertical,
  this.reverse = false,
  List<Widget> children = const <Widget>[],
})

1.5.1 屬性解析

mainAxis:排列的主軸方向唬滑。

reverse:是否反向。

1.5.2 源碼

ListBody的布局代碼非常簡(jiǎn)單模她,根據(jù)主軸的方向侈净,對(duì)子節(jié)點(diǎn)依次排布畜侦。

當(dāng)向右的時(shí)候旋膳,布局代碼如下,向下的代碼類(lèi)似:

double mainAxisExtent = 0.0;
RenderBox child = firstChild;
switch (axisDirection) {
case AxisDirection.right:
  final BoxConstraints innerConstraints = new BoxConstraints.tightFor(height: constraints.maxHeight);
  while (child != null) {
    child.layout(innerConstraints, parentUsesSize: true);
    final ListBodyParentData childParentData = child.parentData;
    childParentData.offset = new Offset(mainAxisExtent, 0.0);
    mainAxisExtent += child.size.width;
    assert(child.parentData == childParentData);
    child = childParentData.nextSibling;
  }
  size = constraints.constrain(new Size(mainAxisExtent, constraints.maxHeight));
  break;
}

當(dāng)向左的時(shí)候,布局代碼如下减俏,向上的代碼類(lèi)似:

double mainAxisExtent = 0.0;
RenderBox child = firstChild;
case AxisDirection.left:
  final BoxConstraints innerConstraints = new BoxConstraints.tightFor(height: constraints.maxHeight);
  while (child != null) {
    child.layout(innerConstraints, parentUsesSize: true);
    final ListBodyParentData childParentData = child.parentData;
    mainAxisExtent += child.size.width;
    assert(child.parentData == childParentData);
    child = childParentData.nextSibling;
  }
  double position = 0.0;
  child = firstChild;
  while (child != null) {
    final ListBodyParentData childParentData = child.parentData;
    position += child.size.width;
    childParentData.offset = new Offset(mainAxisExtent - position, 0.0);
    assert(child.parentData == childParentData);
    child = childParentData.nextSibling;
  }
  size = constraints.constrain(new Size(mainAxisExtent, constraints.maxHeight));
  break;

向右或者向下的時(shí)候垄懂,布局代碼很簡(jiǎn)單草慧,依次去排列仔雷。當(dāng)向左或者向上的時(shí)候舔示,首先會(huì)去計(jì)算主軸所占的空間惕稻,然后再去計(jì)算每個(gè)節(jié)點(diǎn)的位置俺祠。

1.6 使用場(chǎng)景

筆者自己從未使用過(guò)這個(gè)控件蜘渣,也想象不出場(chǎng)景,大家了解下有這么一個(gè)布局控件即可腿准。

2. ListView

A scrollable, linear list of widgets.

2.1 簡(jiǎn)介

ListView是一個(gè)非常常用的控件吐葱,涉及到數(shù)據(jù)列表展示的倦沧,一般情況下都會(huì)選用該控件窖认。ListView跟GridView相似扑浸,基本上是一個(gè)slivers里面只包含一個(gè)SliverList的CustomScrollView喝噪。

2.2 布局行為

ListView在主軸方向可以滾動(dòng)酝惧,在交叉軸方向,則是填滿ListView巫财。

2.3 繼承關(guān)系

Object > Diagnosticable > DiagnosticableTree > Widget > StatelessWidget > ScrollView > BoxScrollView > ListView

看繼承關(guān)系可知,這是一個(gè)組合控件悍及。ListView跟GridView類(lèi)似闽瓢,都是繼承自BoxScrollView。

2.4 示例代碼

ListView(
  shrinkWrap: true,
  padding: EdgeInsets.all(20.0),
  children: <Widget>[
    Text('I\'m dedicating every day to you'),
    Text('Domestic life was never quite my style'),
    Text('When you smile, you knock me out, I fall apart'),
    Text('And I thought I was so smart'),
  ],
)

ListView.builder(
  itemCount: 1000,
  itemBuilder: (context, index) {
    return ListTile(
      title: Text("$index"),
    );
  },
)

兩個(gè)示例都是官方文檔上的例子心赶,第一個(gè)展示四行文字扣讼,第二個(gè)展示1000個(gè)item。

2.5 源碼解析

構(gòu)造函數(shù)如下:

ListView({
  Key key,
  Axis scrollDirection = Axis.vertical,
  bool reverse = false,
  ScrollController controller,
  bool primary,
  ScrollPhysics physics,
  bool shrinkWrap = false,
  EdgeInsetsGeometry padding,
  this.itemExtent,
  bool addAutomaticKeepAlives = true,
  bool addRepaintBoundaries = true,
  double cacheExtent,
  List<Widget> children = const <Widget>[],
})

同時(shí)也提供了如下額外的三種構(gòu)造方法园担,方便開(kāi)發(fā)者使用届谈。

ListView.builder
ListView.separated
ListView.custom

2.5.1 屬性解析

ListView大部分屬性同GridView,想了解的讀者可以看一下之前所寫(xiě)的GridView相關(guān)的文章弯汰。這里只介紹一個(gè)屬性

itemExtent:ListView在滾動(dòng)方向上每個(gè)item所占的高度值。

2.5.2 源碼

@override
Widget buildChildLayout(BuildContext context) {
  if (itemExtent != null) {
    return new SliverFixedExtentList(
      delegate: childrenDelegate,
      itemExtent: itemExtent,
    );
  }
  return new SliverList(delegate: childrenDelegate);
}

ListView標(biāo)準(zhǔn)構(gòu)造布局代碼如上所示湖雹,底層是用到的SliverList去實(shí)現(xiàn)的咏闪。ListView是一個(gè)slivers里面只包含一個(gè)SliverList的CustomScrollView据某。源碼這塊兒可以參考GridView,在此不做更多的說(shuō)明。

2.6 使用場(chǎng)景

ListView使用場(chǎng)景太多了,一般涉及到列表展示的严拒,一般都會(huì)選擇ListView预鬓。

但是需要注意一點(diǎn),ListView的標(biāo)準(zhǔn)構(gòu)造函數(shù)適用于數(shù)目比較少的場(chǎng)景长窄,如果數(shù)目比較多的話,最好使用ListView.builder

ListView的標(biāo)準(zhǔn)構(gòu)造函數(shù)會(huì)將所有item一次性創(chuàng)建,而ListView.builder會(huì)創(chuàng)建滾動(dòng)到屏幕上顯示的item垮媒。

3. CustomMultiChildLayout

A widget that uses a delegate to size and position multiple children.

3.1 簡(jiǎn)介

之前單節(jié)點(diǎn)布局控件中介紹過(guò)一個(gè)類(lèi)似的控件--CustomSingleChildLayout入桂,都是通過(guò)delegate去實(shí)現(xiàn)自定義布局呵晚,只不過(guò)這次是多節(jié)點(diǎn)的自定義布局的控件沮脖,通過(guò)提供的delegate免姿,可以實(shí)現(xiàn)控制節(jié)點(diǎn)的位置以及尺寸紊婉。

3.2 布局行為

CustomMultiChildLayout提供的delegate可以控制子節(jié)點(diǎn)的布局,具體在如下幾點(diǎn):

  • 可以決定每個(gè)子節(jié)點(diǎn)的布局約束條件乔妈;
  • 可以決定每個(gè)子節(jié)點(diǎn)的位置;
  • 可以決定自身的尺寸,但是自身的自身必須不能依賴(lài)子節(jié)點(diǎn)的尺寸。

可以看到,跟CustomSingleChildLayout的delegate提供的作用類(lèi)似,只不過(guò)CustomMultiChildLayout的稍微會(huì)復(fù)雜點(diǎn)庶橱。

3.3 繼承關(guān)系

Object > Diagnosticable > DiagnosticableTree > Widget > RenderObjectWidget > MultiChildRenderObjectWidget > CustomMultiChildLayout

3.4 示例代碼

class TestLayoutDelegate extends MultiChildLayoutDelegate {
  TestLayoutDelegate();

  static const String title = 'title';
  static const String description = 'description';

  @override
  void performLayout(Size size) {
    final BoxConstraints constraints =
        new BoxConstraints(maxWidth: size.width);

    final Size titleSize = layoutChild(title, constraints);
    positionChild(title, new Offset(0.0, 0.0));

    final double descriptionY = titleSize.height;
    layoutChild(description, constraints);
    positionChild(description, new Offset(0.0, descriptionY));
  }

  @override
  bool shouldRelayout(TestLayoutDelegate oldDelegate) => false;
}

Container(
  width: 200.0,
  height: 100.0,
  color: Colors.yellow,
  child: CustomMultiChildLayout(
    delegate: TestLayoutDelegate(),
    children: <Widget>[
      LayoutId(
        id: TestLayoutDelegate.title,
        child: new Text("This is title",
            style: TextStyle(fontSize: 20.0, color: Colors.black)),
      ),
      LayoutId(
        id: TestLayoutDelegate.description,
        child: new Text("This is description",
            style: TextStyle(fontSize: 14.0, color: Colors.red)),
      ),
    ],
  ),
)

上面的TestLayoutDelegate作用很簡(jiǎn)單预伺,對(duì)子節(jié)點(diǎn)進(jìn)行尺寸以及位置調(diào)整瞒御。可以看到烤惊,每一個(gè)子節(jié)點(diǎn)必須用一個(gè)LayoutId控件包裹起來(lái)渡贾,在delegate中可以對(duì)不同id的控件進(jìn)行調(diào)整擂仍。

3.5 源碼解析

構(gòu)造函數(shù)如下:

CustomMultiChildLayout({
  Key key,
  @required this.delegate,
  List<Widget> children = const <Widget>[],
})

3.5.1 屬性解析

delegate:對(duì)子節(jié)點(diǎn)進(jìn)行尺寸以及位置調(diào)整的delegate肋坚。

3.5.2 源碼

@override
void performLayout() {
  size = _getSize(constraints);
  delegate._callPerformLayout(size, firstChild);
}

CustomMultiChildLayout的布局代碼很簡(jiǎn)單诚卸,調(diào)用delegate中的布局函數(shù)進(jìn)行相關(guān)的操作辫愉,本身做的處理很少,在這里不做過(guò)多的解釋。

3.6 使用場(chǎng)景

一些比較復(fù)雜的布局場(chǎng)景可以使用砰苍,但是有很多可替代的控件潦匈,使用起來(lái)也沒(méi)有這么麻煩,大家還是按照自己熟練程度選擇使用赚导。

4. 后話

筆者建了一個(gè)Flutter學(xué)習(xí)相關(guān)的項(xiàng)目茬缩,Github地址,里面包含了筆者寫(xiě)的關(guān)于Flutter學(xué)習(xí)相關(guān)的一些文章吼旧,會(huì)定期更新凰锡,也會(huì)上傳一些學(xué)習(xí)Demo,歡迎大家關(guān)注圈暗。

5. 參考

  1. ListBody class
  2. ListView class
  3. CustomMultiChildLayout class
  4. Working with long lists
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末掂为,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子厂置,更是在濱河造成了極大的恐慌菩掏,老刑警劉巖,帶你破解...
    沈念sama閱讀 206,214評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件昵济,死亡現(xiàn)場(chǎng)離奇詭異智绸,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)访忿,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門(mén)瞧栗,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人海铆,你說(shuō)我怎么就攤上這事迹恐。” “怎么了卧斟?”我有些...
    開(kāi)封第一講書(shū)人閱讀 152,543評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵殴边,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我珍语,道長(zhǎng)锤岸,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,221評(píng)論 1 279
  • 正文 為了忘掉前任板乙,我火速辦了婚禮是偷,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己蛋铆,他們只是感情好馋评,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評(píng)論 5 371
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著刺啦,像睡著了一般留特。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上洪燥,一...
    開(kāi)封第一講書(shū)人閱讀 49,007評(píng)論 1 284
  • 那天磕秤,我揣著相機(jī)與錄音,去河邊找鬼捧韵。 笑死市咆,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的再来。 我是一名探鬼主播蒙兰,決...
    沈念sama閱讀 38,313評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼芒篷!你這毒婦竟也來(lái)了搜变?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 36,956評(píng)論 0 259
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤针炉,失蹤者是張志新(化名)和其女友劉穎挠他,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體篡帕,經(jīng)...
    沈念sama閱讀 43,441評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡殖侵,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評(píng)論 2 323
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了镰烧。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片拢军。...
    茶點(diǎn)故事閱讀 38,018評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖怔鳖,靈堂內(nèi)的尸體忽然破棺而出茉唉,到底是詐尸還是另有隱情,我是刑警寧澤结执,帶...
    沈念sama閱讀 33,685評(píng)論 4 322
  • 正文 年R本政府宣布度陆,位于F島的核電站,受9級(jí)特大地震影響献幔,放射性物質(zhì)發(fā)生泄漏坚芜。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評(píng)論 3 307
  • 文/蒙蒙 一斜姥、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦铸敏、人聲如沸缚忧。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,240評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)闪水。三九已至,卻和暖如春蒙具,著一層夾襖步出監(jiān)牢的瞬間球榆,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,464評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工禁筏, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留持钉,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 45,467評(píng)論 2 352
  • 正文 我出身青樓篱昔,卻偏偏與公主長(zhǎng)得像每强,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子州刽,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評(píng)論 2 345

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

  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 171,501評(píng)論 25 707
  • 更多信息請(qǐng)查看flutter layout Layouts Sigle-child layout widgets ...
    one_cup閱讀 28,903評(píng)論 1 17
  • 1空执、通過(guò)CocoaPods安裝項(xiàng)目名稱(chēng)項(xiàng)目信息 AFNetworking網(wǎng)絡(luò)請(qǐng)求組件 FMDB本地?cái)?shù)據(jù)庫(kù)組件 SD...
    陽(yáng)明先生_x閱讀 15,968評(píng)論 3 119
  • 之所以寫(xiě)下這么大氣的題目,是源于我的自身經(jīng)歷穗椅。 第一段經(jīng)歷是在我因病退學(xué)在家做老師那段時(shí)間辨绊,那時(shí)候的我,做了老師在...
    紅豆面包淡咖啡閱讀 172評(píng)論 5 0
  • (陳湘) 像一支國(guó)家精英部隊(duì) 排著整齊的隊(duì)伍 邁著莊重的步伐 前進(jìn) 前進(jìn) ...
    柳絮輕飛閱讀 501評(píng)論 1 2