Flutter系列之Flex佈局詳解
PS:想做一件事很容易,真正去做一件事很困難。
Flutter 是 Google 推出的跨平臺 UI 框架,可以快速地在 Android 和 IOS 上構建高質量的應用程式,其主要特點是 Flutter 具有快速開發的能力、富有表現力和靈活的 Ui 以及良好的原生效能,本篇文章主要介紹 Flutter 中的 Flex 佈局,如下:
- Flex基礎
- Flex常用設定
- Row和Column
- Expanded和Flexible
- Space
- 總結
Flex基礎
Flex 佈局方式已經廣泛使用在前端、小程式開發之中,如果之前已經學習過 Flex 佈局,那麼在 Flutter 中也是大同小異的,Flexible Box 示意圖如下:
關於這張圖的介紹請檢視前面這篇文章:
Flex Widget 可以設定主軸方向,如果知曉主軸方向,可以直接使用 Row 或者 Column,Flex Widget 不能滾動,如果涉及到滾動可以嘗試使用 ListView,Flex Widget 的內容超過其寬度和高度,則顯示黃黑相間的警告條紋,以水平方向為例出現的報錯資訊如下:
I/flutter (14749): ══╡ EXCEPTION CAUGHT BY RENDERING LIBRARY ╞═════════════════════════════════════════════════════════
I/flutter (14749): The following assertion was thrown during layout:
I/flutter (14749): A RenderFlex overflowed by 440 pixels on the right.
報錯顯示如下:
Flex常用設定
Flex 常用屬性如下:
- direction
- mainAxisAlignment
- mainAxisSize
- crossAxisAlignment
- verticalDirection
- textBaseline
1. direction
設定主軸方向,可設定的值為 Axis.horizontal 和 Axis.vertical,交叉軸與主軸方向垂直。
2. mainAxisAlignment:
設定子 Widget 沿著主軸方向的排列方式,預設 MainAxisAlignment.start,可設定的方式如下:
- MainAxisAlignment.start:左對齊,預設值;
- MainAxisAlignment.end:右對齊;
- MainAxisAlignment.center:居中對齊;
- MainAxisAlignment.spaceBetween:兩端對齊;
- MainAxisAlignment.spaceAround:每個 Widget 兩側的間隔相等,與螢幕邊緣的間隔是其他 Widget 之間間隔的一半;
- MainAxisAlignment.spaceEvently:平均分佈各個 Widget,與螢幕邊緣的間隔與其他 Widget 之間的間隔相等.
對比效果如下:
3. mainAxisSize
設定主軸的大小,預設 MainAxisSize.max,可設定的值如下:
- MainAxisSize.max:主軸的大小是父容器的大小;
- MainAxisSize.min:主軸的大小是其資 Widget 大小之和。
對比效果如下:
將 mainAxisAlignment 設定成 spaceBetween,如果 mainAxisSize 設定為 max,則是整個 Row 寬度基礎上按照 spaceBetween 的方式進行排列,如果 mainAxisSize 設定為 min,則是三個 Container 寬度之和範圍內按照 spaceBetween 的方式進行排列。
4. crossAxisAlignment
設定子 Widget 沿著交叉軸方向的排列方式,預設 CrossAxisAlignment.center,可設定的方式如下:
- CrossAxisAlignment.start:與交叉軸的起始位置對齊;
- CrossAxisAlignment.end:與交叉軸的結束位置對齊;
- CrossAxisAlignment.center:居中對齊;
- CrossAxisAlignment.stretch:填充整個交叉軸;
- CrossAxisAlignment.baseline:按照第一行文字基線對齊。
對比效果如下:
5. verticalDirection
設定垂直方向上的子 Widget 的排列順序,預設為 VerticalDirection.down,設定方式如下:
- VerticalDirection.down:start 在頂部,end 在底部;
- VerticalDirection.up:start 在底部,end 在頂部。
對比效果如下:
注意觀察交叉軸設定的 CrossAxisAlignment.end,在此基礎上垂直方向上的變化。
6. textBaseline
設定文字對齊的基線型別,可設定的值如下:
- TextBaseline.alphabetic:與字母基線對齊;
- TextBaseline.ideographic:與表意字元基線對齊;
使用時當 crossAxisAlignment 設定為 baseline 時,必須設定 textBaseline 屬性的值,使用方式如下:
// textBaseline
class FlexSamplePage extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text("Flex Sample"),
centerTitle: true,
),
body: Row(
children: <Widget>[
Expanded(
child: Row(
children: <Widget>[
Text("躬",style: TextStyle(fontSize: 40,),),
Text("x",style: TextStyle(fontSize: 60,),),
Text("ing",style: TextStyle(fontSize: 16,),),
Text("之",style: TextStyle(fontSize: 16,),),
],
)),
Expanded(
child: Row(
crossAxisAlignment: CrossAxisAlignment.baseline,
textBaseline: TextBaseline.alphabetic,
children: <Widget>[
Text("躬",style: TextStyle(fontSize: 40,),),
Text("x",style: TextStyle(fontSize: 60,),),
Text("ing",style: TextStyle(fontSize: 16,),),
Text("之",style: TextStyle(fontSize: 16, ),),
],
)),
Expanded(
child: Row(
crossAxisAlignment: CrossAxisAlignment.baseline,
textBaseline: TextBaseline.ideographic,
children: <Widget>[
Text("躬",style: TextStyle(fontSize: 40, ),),
Text("x",style: TextStyle(fontSize: 60,),),
Text("ing",style: TextStyle(fontSize: 16,),),
Text("之",style: TextStyle(fontSize: 16,),),
],
))
],
),
);
}
}
分別為不設定 textBaseline 屬性、設定 TextBaseline.alphabetic 和 TextBaseline.ideographic,對比效果如下:
兩者雖然在對應基線含義上有所不同,但是測試沒發現不同,後期繼續留意觀察,知道的朋友可以評論指出。
Row和Column
Row 和 Column 都繼承 Flex,Row 主軸的方向為水平方向,Column 主軸的方向為豎直方向,即在 Flex 基礎上設定了主軸方向 direction,如下:
// Row
direction: Axis.horizontal,
/// Column
direction: Axis.vertical,
如果確定主軸方向,可以直接使用 Row 或者 Column,使用方式和 Flex 一致。
Expanded和Flexible
Flexible 的 fix 屬性預設為 FlexFit.loose,而 Expanded 繼承 Flexible,其 fix 屬性指定為 FlexFit.tight,兩者因為其 fix 屬性不用而不同,若將 Flexible 的 fit 屬性設定為 FlexFit.tight,則 Flexible 與 Expanded 等效,可設定的 fit 屬性如下:
- tight:強制填充可利用的空間;
- loose:不強制填充可利用空間,Widget自身大小。
對比效果如下:
Expanded 可以使 Row、Column、Flex 裡面的元件填充沿著主軸可利用的空間,如果多個 Widget 都使用了 Expanded 元件,可以使用 Expanded 的 flex 屬性按照比例分配主軸空間,flex 屬性相當於 Android LinearLayout 的 weight 屬性,如下:
// Expanded
class ExpandedSamplePage extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text("Row Sample"),
centerTitle: true,
),
body: Row(
mainAxisAlignment: MainAxisAlignment.start,
mainAxisSize: MainAxisSize.max,
children: <Widget>[
Expanded(
flex: 1,
child: Container(
width: 50,
height: 50,
color: Colors.red,
child: Center(
child: Text(
"A",
style: TextStyle(fontSize: 20, color: Colors.white),
),
)),
),
Expanded(
flex: 2,
child: Container(
width: 50, // Row Expanded下width無效
height: 50, // Column Expanded下height無效
color: Colors.green,
child: Center(
child: Text(
"B",
style: TextStyle(fontSize: 20, color: Colors.white),
),
)),
),
Container(
width: 50,
height: 50,
color: Colors.yellow,
child: Center(
child: Text(
"C",
style: TextStyle(fontSize: 20, color: Colors.white),
),
)),
],
));
}
}
顯示效果如下:
Spacer
Spacer 用來調節 Widget 之間的間距,會佔據所有的剩餘空間,因此 MainAxisAlignment 的設定將無效,Spacer 的屬性 flex 用於設定剩餘空間的分配權重,預設值為 1,表示佔據所有剩餘空間,如果兩個以上 Spacer 則按照 flex 分配剩餘空間,程式碼參考如下:
class RowSamplePage1 extends StatelessWidget {
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text("Row Sample"),
centerTitle: true,
),
body: ConstrainedBox(
constraints: BoxConstraints(maxHeight: 150),
child: Row(
children: <Widget>[
Container(
width: 80,
height: 80,
color: Colors.red,
),
Spacer(flex: 1,),
Container(
width: 80,
height: 80,
color: Colors.green,
),
Spacer(flex: 2,),
Container(
width: 80,
height: 80,
color: Colors.yellow,
),
],
),
));
}
}
顯示效果如下:
以上主要學習了 Flutter 中 Flex 佈局相關內容,重點是理解 Flex 基本概念,在此基礎上對 Flex 佈局進行了學習和驗證,更多內容見微信公眾號躬行之。
相關文章
- Flutter佈局——Flex、FittedBox、Stack、Container佈局教程【超詳細】FlutterFlexAI
- 詳解CSS的Flex佈局CSSFlex
- Flutter 佈局詳解Flutter
- [Flutter必備]-Flex佈局完全解讀FlutterFlex
- CSS例項詳解:Flex佈局CSSFlex
- Flutter第4天--基礎控制元件(下)+Flex佈局詳解Flutter控制元件Flex
- Flutter 基礎佈局Widgets之Expanded詳解Flutter
- Flutter 基礎佈局Widgets之Align詳解Flutter
- Flutter 基礎佈局Widgets之FittedBox詳解Flutter
- Flutter 基礎佈局Widgets之Row詳解Flutter
- Flutter 基礎佈局Widgets之Stack詳解Flutter
- Flutter Container Widget 佈局詳解FlutterAI
- Flutter 佈局(一)- Container詳解FlutterAI
- 淺嘗flutter中的flex佈局FlutterFlex
- flutter系列之:在flutter中使用流式佈局Flutter
- Flutter佈局篇(1)–水平和垂直佈局詳解Flutter
- Flutter佈局篇(1)--水平和垂直佈局詳解Flutter
- css--flex彈性佈局詳解和使用CSSFlex
- Flutter 佈局(七)- Row、Column詳解Flutter
- Flutter 佈局控制元件篇-->Flex、ExpandedFlutter控制元件Flex
- CSS3之flex佈局CSSS3Flex
- css之彈性佈局(flex)CSSFlex
- flex 佈局Flex
- Flex佈局Flex
- Flutter 佈局(九)- Flow、Table、Wrap詳解Flutter
- Flutter 佈局(八)- Stack、IndexedStack、GridView詳解FlutterIndexView
- Flutter 佈局(三)- FittedBox、AspectRatio、ConstrainedBox詳解FlutterAI
- Flutter 佈局(二)- Padding、Align、Center詳解Flutterpadding
- Flutter 佈局(六)- SizedOverflowBox、Transform、CustomSingleChildLayout詳解FlutterZedORM
- Flutter 佈局(十)- ListBody、ListView、CustomMultiChildLayout詳解FlutterView
- Flutter佈局詳解,必知必會Flutter
- 佈局之: flex(CSS3新增)FlexCSSS3
- 浮動佈局 和 flex佈局Flex
- css flex佈局CSSFlex
- flex佈局原理Flex
- Flutter開發日記——Flutter佈局Widget詳解(下)Flutter
- Flutter開發日記——Flutter佈局Widget詳解(上)Flutter
- 圖解CSS3-flex佈局圖解CSSS3Flex