九、Flutter的布局Widget

一. 单子布局组件

单子布局组件的含义是其只有一个子组件,可以通过设置一些属性设置该子组件所在的位置信息等。比较常用的单子布局组件有:Align、Center、Padding、Container。

1.1. Align组件

1.1.1.  Align介绍

看到Align这个词,我们就知道它有我们的对齐方式有关。

在其他端的开发中(iOS、Android、前端)Align通常只是一个属性而已,但是Flutter中Align也是一个组件。

我们可以通过源码来看一下Align有哪些属性:

这里我们特别解释一下widthFactorheightFactor作用:

因为子组件在父组件中的对齐方式必须有一个前提,就是父组件得知道自己的范围(宽度和高度);

如果widthFactorheightFactor不设置,那么默认Align会尽可能的大(尽可能占据自己所在的父组件);

我们也可以对他们进行设置,比如widthFactor设置为3,那么相对于Align的宽度是子组件跨度的3倍;

1.1.2. Align演练

我们简单演练一下Align

1.2. Center组件

1.2.1. Center介绍

事实上Center组件继承自Align,只是将alignment设置为Alignment.center。

源码分析

1.2.2. Center演练

我们将上面的代码Align换成Center

1.3. Padding组件

1.3.1. Padding介绍

Padding组件在其他端也是一个属性而已,但是在Flutter中是一个Widget,但是Flutter中没有Margin这样一个Widget,这是因为外边距也可以通过Padding来完成。

Padding通常用于设置子Widget到父Widget的边距(你可以称之为是父组件的内边距或子Widget的外边距)。

源码分析

1.3.2. Padding演练

1.4. Container组件

Container组件类似于其他Android中的View,iOS中的UIView。

如果你需要一个视图,有一个背景颜色、图像、有固定的尺寸、需要一个边框、圆角等效果,那么就可以使用Container组件。

1.4.1. Container介绍

Container在开发中被使用的频率是非常高的,特别是我们经常会将其作为容器组件。

下面我们来看一下Container有哪些属性:

大多数属性在介绍其它容器时都已经介绍过了,不再赘述,但有两点需要说明:

容器的大小可以通过widthheight属性来指定,也可以通过constraints来指定,如果同时存在时,widthheight优先。实际上Container内部会根据widthheight来生成一个constraints

colordecoration是互斥的,实际上,当指定color时,Container内会自动创建一个decoration;

1.4.2. Container演练

1.4.3. BoxDecoration

Container有一个非常重要的属性 decoration

他对应的类型是Decoration类型,但是它是一个抽象类。

在开发中,我们经常使用它的实现类BoxDecoration来进行实例化。

BoxDecoration常见属性:

部分效果演示:

1.4.4. 实现圆角图像

二. 多子布局组件

在开发中,我们经常需要将多个Widget放在一起进行布局,比如水平方向、垂直方向排列,甚至有时候需要他们进行层叠,比如图片上面放一段文字等;

这个时候我们需要使用多子布局组件(Multi-child layout widgets)。

比较常用的多子布局组件是Row、Column、Stack。

2.0 Flex组件

事实上,我们即将学习的Row组件和Column组件都继承自Flex组件。

Flex组件和Row、Column属性主要的区别就是多一个direction。

当direction的值为Axis.horizontal的时候,则是Row。

当direction的值为Axis.vertical的时候,则是Column。

在学习Row和Column之前,我们先学习主轴和交叉轴的概念。

因为Row是一行排布,Column是一列排布,那么它们都存在两个方向,并且两个Widget排列的方向应该是对立的。

它们之中都有主轴(MainAxis)和交叉轴(CrossAxis)的概念。

2.1. Row组件

2.1.1. Row介绍

Row组件用于将所有的子Widget排成一行,实际上这种布局应该是借鉴于Web的Flex布局。

如果熟悉Flex布局,会发现非常简单。

mainAxisSize

表示Row在主轴(水平)方向占用的空间,默认是MainAxisSize.max,表示尽可能多的占用水平方向的空间,此时无论子widgets实际占用多少水平空间,Row的宽度始终等于水平方向的最大宽度

而MainAxisSize.min表示尽可能少的占用水平空间,当子widgets没有占满水平剩余空间,则Row的实际宽度等于所有子widgets占用的的水平空间;

mainAxisAlignment:表示子Widgets在Row所占用的水平空间内对齐方式

如果mainAxisSize值为MainAxisSize.min,则此属性无意义,因为子widgets的宽度等于Row的宽度

只有当mainAxisSize的值为MainAxisSize.max时,此属性才有意义

MainAxisAlignment.start表示沿textDirection的初始方向对齐,

如textDirection取值为TextDirection.ltr时,则MainAxisAlignment.start表示左对齐,textDirection取值为TextDirection.rtl时表示从右对齐。

而MainAxisAlignment.end和MainAxisAlignment.start正好相反;

MainAxisAlignment.center表示居中对齐。

crossAxisAlignment:表示子Widgets在纵轴方向的对齐方式

Row的高度等于子Widgets中最高的子元素高度

它的取值和MainAxisAlignment一样(包含start、end、 center三个值)

不同的是crossAxisAlignment的参考系是verticalDirection,即verticalDirection值为VerticalDirection.down时crossAxisAlignment.start指顶部对齐,verticalDirection值为VerticalDirection.up时,crossAxisAlignment.start指底部对齐;而crossAxisAlignment.end和crossAxisAlignment.start正好相反;

2.1.2. Row演练

2.1.3. mainAxisSize

默认情况下,Row会尽可能占据多的宽度,让子Widget在其中进行排布,这是因为mainAxisSize属性默认值是MainAxisSize.max。

2.1.4. TextBaseline

2.1.5. Expanded

如果我们希望红色和黄色的Container Widget不要设置固定的宽度,而是占据剩余的部分,这个时候应该如何处理呢?这个时候我们可以使用 Expanded 来包裹 Container Widget,并且将它的宽度不设置值;flex属性,弹性系数,Row会根据两个Expanded的弹性系数来决定它们占据剩下空间的比例

2.2. Column组件

Column组件用于将所有的子Widget排成一列,学会了前面的Row后,Column只是和row的方向不同而已。

2.2.1. Column介绍

我们直接看它的源码:我们发现和Row属性是一致的,不再解释

2.2.2. Column演练

我们直接将Row的代码中Row改为Column,查看代码运行效果:

2.3.1. Stack介绍

我们还是通过源码来看一下Stack有哪些属性:

参数解析:

alignment:此参数决定如何去对齐没有定位(没有使用Positioned)或部分定位的子widget。所谓部分定位,在这里**特指没有在某一个轴上定位:**left、right为横轴,top、bottom为纵轴,只要包含某个轴上的一个定位属性就算在该轴上有定位。

textDirection:和Row、Wrap的textDirection功能一样,都用于决定alignment对齐的参考系即:textDirection的值为TextDirection.ltr,则alignment的start代表左,end代表右;textDirection的值为TextDirection.rtl,则alignment的start代表右,end代表左。

fit:此参数用于决定没有定位的子widget如何去适应Stack的大小。StackFit.loose表示使用子widget的大小,StackFit.expand表示扩伸到Stack的大小。

overflow:此属性决定如何显示超出Stack显示空间的子widget,值为Overflow.clip时,超出部分会被剪裁(隐藏),值为Overflow.visible 时则不会。

2.3.2. Stack演练

Stack会经常和Positioned一起来使用,Positioned可以决定组件在Stack中的位置,用于实现类似于Web中的绝对定位效果。

我们来看一个简单的演练:

注意:Positioned组件只能在Stack中使用。

你可能感兴趣的:(九、Flutter的布局Widget)