Flutter开发实战七:深入布局原理

2019-04-0121:49:35APP与小程序开发Comments3,094 views字数 3588阅读模式

知道了 WidgetElementRenderObject 三者之间的关系,其中我们最为熟知的 Widget ,作为“配置文件”的存在,在 Flutter 中它的功能都是比较单一的,属于 “颗粒度比较细的存在” ,写代码时就像拼乐高“积木”,那这“积木”究竟怎么拼的?下面就 深入 去挖挖有意思的东西吧。( ̄▽ ̄)文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

一、单子元素布局

在 Flutter 单个子元素的布局 Widget 中,Container 无疑是被用的最广泛的,因为它在“功能”上并不会如 Padding 等 Widget 那样功能单一,这是为什么呢?文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

究其原因,从下图源码可以看出,Container 其实也只是把其他“单一”的 Widget 做了二次封装,然后通过配置来达到“多功能的效果”而已。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

Flutter开发实战七:深入布局原理

接着我们先看 ConstrainedBox 源码,从下图源码可以看出,它是继承了 SingleChildRenderObjectWidget,关键是 override 了 createRenderObject 方法,返回了 RenderConstrainedBox文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

这里体现了第六篇中的 Widget 与 RenderObject 的关系文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

是的,RenderConstrainedBox 就是继承自 RenderBox,从而实现RenderObject 的布局,这里我们得到了它们的关系如下 :文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

WidgetRenderObject
RenderConstrainedBoxRenderConstrainedBox
Flutter开发实战七:深入布局原理

然后我们继续对其他每个 Widget 进行观察,可以看到它们也都是继承SingleChildRenderObjectWidget ,而“简单来说”它们不同的地方就是 RenderObject 的实现了:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

WidgetRenderBox (RenderObject)
AlignRenderPositionedBox
PaddingRenderPadding
TransformRenderTransform
OffstageRenderOffstage

所以我们可以总结:真正的布局和大小计算等行为,都是在 RenderBox 上去实现的。 不同的 Widget 通过各自的 RenderBox 实现了“差异化”的布局效果。所以找每个 Widget 的实现,找它的 RenderBox 实现就可以了。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

这里我们通过 Offstage 这个Widget 小结下,Offstage 这个 Widget 是通过 offstage 标志控制 child 是否显示的效果,同样的它也有一个 RenderOffstage ,如下图,通过 RenderOffstage 的源码我们可以“真实”看到 offstage 标志位的作用:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

Flutter开发实战七:深入布局原理

所以大部分时候,我们的 Widget 都是通过实现 RenderBox 实现布局的 ,那我们可不可抛起 Widget 直接用 RenderBox呢?答案明显是可以的,如果你闲的?疼的话!文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

Flutter 官方为了治疗我们“?疼”,提供了一个叫 CustomSingleChildLayout 的类,它抽象了一个叫 SingleChildLayoutDelegate 的对象,让你可以更方便的操作 RenderBox 来达到自定义的效果。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

Flutter开发实战七:深入布局原理

如下图三张源码所示,SingleChildLayoutDelegate 的对象提供以下接口,并且接口 前三个 是按照顺序被调用的,通过实现这个接口,你就可以轻松的控制RenderBox 的 布局位置、大小 等。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

Flutter开发实战七:深入布局原理
Flutter开发实战七:深入布局原理
Flutter开发实战七:深入布局原理

二、多子元素布局

事实上“多子元素布局”和单子元素类似,通过“举一反三”我们就可以知道它们的关系了,比如:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

  • RowColum 都继承了 Flex,而 Flex 继承了MultiChildRenderObjectWidget 并通过 RenderFlex 创建了 RenderBox
  • Stack 同样继承 MultiChildRenderObjectWidget 并通过 RenderStack 创建了 RenderBox
WidgetRenderBox (RenderObject)
Row/Colum/FlexRenderFlex
StackRenderStack
FlowRenderFlow
WrapRenderWrap

同样“多子元素布局”也提供了 CustomMultiChildLayoutMultiChildLayoutDelegate 满足你的“?疼”需求。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

三、多子元素滑动布局

滑动布局作为 “多子元素布局” 的另一个分支,如 ListViewGridViewPageview ,它们在实现上要复杂的多,从下图一个的流程上我们大致可以知道它们的关系:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

Flutter开发实战七:深入布局原理

由上图我们可以知道,流程最终回产生两个 RenderObject文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

  • RenderSliverBase class for the render objects that implement scroll effects in viewports.
  • RenderViewportA render object that is bigger on the inside.
/// [RenderViewport] cannot contain [RenderBox] children directly. Instead, use
/// a [RenderSliverList], [RenderSliverFixedExtentList], [RenderSliverGrid], or
/// a [RenderSliverToBoxAdapter], for example.
复制代码

并且从 RenderViewport的说明我们知道,RenderViewport内部是不能直接放置 RenderBox,需要通过 RenderSliver 大家族来完成布局。而从源码可知:RenderViewport 对应的 Widget Viewport 就是一个 MultiChildRenderObjectWidget (你看,又回到 MultiChildRenderObjectWidget 了吧。)文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

再稍微说下上图的流程:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

  • ListViewPageviewGridView 等都是通过 ScrollableViewPortSliver大家族实现的效果。这里简单不规范描述就是:一个“可滑动”的控件,嵌套了一个“视觉窗口”,然后内部通过“碎片”展示 children
  • 不同的是 PageView 没有继承 SrollView,而是直接通过 NotificationListenerScrollNotification 嵌套实现。

注意 TabBarView 内部就是:NotificationListener + PageView文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

是不是觉得少了什么?哈哈哈,有的有的,官方同样提供了解决“?疼”的自定义滑动 CustomScrollView ,它继承了 ScrollView,可通过 slivers 参数实现布局,这些 slivers 最终回通过 ScrollablebuildViewport 添加到 ViewPort 中,如下代码所示:文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

CustomScrollView(
  slivers: <Widget>[
    const SliverAppBar(
      pinned: true,
      expandedHeight: 250.0,
      flexibleSpace: FlexibleSpaceBar(
        title: Text('Demo'),
      ),
    ),
    SliverGrid(
      gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent(
        maxCrossAxisExtent: 200.0,
        mainAxisSpacing: 10.0,
        crossAxisSpacing: 10.0,
        childAspectRatio: 4.0,
      ),
      delegate: SliverChildBuilderDelegate(
        (BuildContext context, int index) {
          return Container(
            alignment: Alignment.center,
            color: Colors.teal[100 * (index % 9)],
            child: Text('grid item $index'),
          );
        },
        childCount: 20,
      ),
    ),
    SliverFixedExtentList(
      itemExtent: 50.0,
      delegate: SliverChildBuilderDelegate(
        (BuildContext context, int index) {
          return Container(
            alignment: Alignment.center,
            color: Colors.lightBlue[100 * (index % 9)],
            child: Text('list item $index'),
          );
        },
      ),
    ),
  ],
)

作者:恋猫de小郭
链接:https://juejin.im/post/5c8c6ef7e51d450ba7233f51
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html

文章源自菜鸟学院-https://www.cainiaoxueyuan.com/xcx/11110.html
  • 本站内容整理自互联网,仅提供信息存储空间服务,以方便学习之用。如对文章、图片、字体等版权有疑问,请在下方留言,管理员看到后,将第一时间进行处理。
  • 转载请务必保留本文链接:https://www.cainiaoxueyuan.com/xcx/11110.html

Comment

匿名网友 填写信息

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定