Flutter学习记录——11.流式布局组件

柔光的暖阳◎ 2023-06-19 02:56 116阅读 0赞

文章目录

  • 1.Flow Widget
  • 2.Wrap Widget
  • 3.总结

1.Flow Widget

流式布局可以用在商品标签列表、不规则瀑布流列表、网格布局的使用上。简单地说流式布局就是可以自动换行的布局,如我们一行里的控件放不下了,则自动绘制到下一行。

Flutter 的 Flow 就可以自己自定义规则来控制子布局排列。Flow 继承自MultiChildRenderObjectWidget,Flow 性能比较好,绘制也比较灵活,可以定制布局效果。

Flow 的构造方法如下:

  1. Flow({
  2. Key key,
  3. // 子布局排列配置规则
  4. @required this.delegate,
  5. // 布局子控件
  6. List<Widget> children = const <Widget>[],
  7. })

Flow 的构造方法很简单,最重要的就是 delegate 这个规则,我们按照需要编写即可。使用的时候,我们需要创建一个 Delegate 继承自 FlowDelegate。

我们接下来看一个完整的例子:

  1. class FlowSamplesState extends State<FlowSamples> {
  2. @override
  3. void initState() {
  4. super.initState();
  5. }
  6. @override
  7. Widget build(BuildContext context) {
  8. return Scaffold(
  9. appBar: AppBar(title: Text('Flow Demo'), primary: true),
  10. body: Flow(
  11. // 子控件排列绘制规则
  12. delegate: FlowWidgetDelegate(margin: EdgeInsets.all(10.0)),
  13. children: <Widget>[
  14. Container(
  15. width: 80.0,
  16. height: 80.0,
  17. color: Colors.orange,
  18. ),
  19. Container(
  20. width: 160.0,
  21. height: 80.0,
  22. color: Colors.teal,
  23. ),
  24. Container(
  25. width: 80.0,
  26. height: 80.0,
  27. color: Colors.red,
  28. ),
  29. Container(
  30. width: 80.0,
  31. height: 80.0,
  32. color: Colors.yellow,
  33. ),
  34. Container(
  35. width: 80.0,
  36. height: 80.0,
  37. color: Colors.brown,
  38. ),
  39. Container(
  40. width: 80.0,
  41. height: 80.0,
  42. color: Colors.purple,
  43. ),
  44. ],
  45. ));
  46. }
  47. }
  48. class FlowWidgetDelegate extends FlowDelegate {
  49. EdgeInsets margin = EdgeInsets.all(10);
  50. // 构造方法,传入每个child的间隔
  51. FlowWidgetDelegate({ this.margin});
  52. // 必须要重写的方法:child的绘制控制代码,可以调整尺寸位置
  53. @override
  54. void paintChildren(FlowPaintingContext context) {
  55. var screenWidth = context.size.width;
  56. double offsetX = margin.left; //记录横向绘制的x坐标
  57. double offsetY = margin.top; //记录纵向绘制的y坐标
  58. // 遍历子控件进行绘制
  59. for (int i = 0; i < context.childCount; i++) {
  60. // 如果当前x左边加上子控件宽度小于屏幕宽度则继续绘制,否则换行
  61. var width = context.getChildSize(i).width + offsetX + margin.right;
  62. if (width < screenWidth) {
  63. // 绘制子控件
  64. context.paintChild(i,
  65. transform: Matrix4.translationValues(offsetX, offsetY, 0.0));
  66. offsetX = width + margin.left;
  67. } else {
  68. offsetX = margin.left;
  69. offsetY += context.getChildSize(i).height + margin.top + margin.bottom;
  70. //绘制子控件
  71. context.paintChild(i,
  72. transform: Matrix4.translationValues(offsetX, offsetY, 0.0));
  73. offsetX += context.getChildSize(i).width + margin.left + margin.right;
  74. }
  75. }
  76. }
  77. // 必须要重写的方法:是否需要重绘
  78. @override
  79. bool shouldRepaint(FlowDelegate oldDelegate) {
  80. return oldDelegate != this;
  81. }
  82. // 可选重写方法:是否需要重新布局
  83. @override
  84. bool shouldRelayout(FlowDelegate oldDelegate) {
  85. return super.shouldRelayout(oldDelegate);
  86. }
  87. // 可选重写方法:设置Flow布局的尺寸
  88. @override
  89. Size getSize(BoxConstraints constraints) {
  90. return super.getSize(constraints);
  91. }
  92. // 可选重写方法:设置每个child的布局约束条件,会覆盖已有的
  93. @override
  94. BoxConstraints getConstraintsForChild(int i, BoxConstraints constraints) {
  95. return super.getConstraintsForChild(i, constraints);
  96. }
  97. }

我们看下 Flow 布局的效果图:

Flow

我们使用 Flow 的重点就是继承重写一个 FlowDelegate,FlowDelegate 里面最重要的方法就是paintChildren 方法,来定制我们的排列绘制子控件规则。

当然我们也可以实现更复杂一点的效果,可以根据自己的需求进行排列子控件,来实现流式布局或瀑布流布局效果。

2.Wrap Widget

Wrap 从字面意思上也很好理解,就是组件的大小是根据自身的实际大小来包裹的。在 Flutter 中Wrap 组件是一个可以使子控件自动换行的布局组件,默认的内部子控件排列方向是水平的。也就是超过指定宽度大小,就自动换到下一行。例如我们平时看到的商品标签页,标签是自动按行排列的。

其实 Wrap 可以实现的效果,Flow 也可以实现。我们可以根据方便程度进行合理选择。

Wrap 也是继承自 MultiChildRenderObjectWidget,内部可以放置多个子控件。

我们看下 Wrap 的构造方法:

  1. Wrap({
  2. Key key,
  3. // 子控件在主轴上的排列方向
  4. this.direction = Axis.horizontal,
  5. // 子控件在主轴方向上的对齐方式
  6. this.alignment = WrapAlignment.start,
  7. // 主轴方向上子控件的间距
  8. this.spacing = 0.0,
  9. // 子控件在纵轴方向上的对齐方式
  10. this.runAlignment = WrapAlignment.start,
  11. // 子控件在纵轴方向上的间距
  12. this.runSpacing = 0.0,
  13. // 交叉轴上子控件的对齐方式
  14. this.crossAxisAlignment = WrapCrossAlignment.start,
  15. // 文本方向
  16. this.textDirection,
  17. // direction为Vertical时子控件排序顺序
  18. this.verticalDirection = VerticalDirection.down,
  19. // 子控件集合
  20. List<Widget> children = const <Widget>[],
  21. })

属性虽然多,但是很好理解,因为Wrap 很多属性和 Row、Column、Flex 这些组件的属性有所重复。

接下来通过一个代码实例,来看下 Wrap 的特点和用法:

  1. class WrapSamplesState extends State<WrapSamples> {
  2. @override
  3. void initState() {
  4. super.initState();
  5. }
  6. @override
  7. Widget build(BuildContext context) {
  8. return Scaffold(
  9. appBar: AppBar(title: Text('Wrap Demo'), primary: true),
  10. // Wrap包裹的子控件会自动换行排列
  11. body: Wrap(
  12. spacing: 8.0, // 子控件横向间距
  13. runSpacing: 4.0, // 子控件纵向(每行)的间距
  14. // 子控件元素集合
  15. children: <Widget>[
  16. Chip(
  17. avatar: CircleAvatar(
  18. backgroundColor: Colors.blue.shade900, child: Text('A')),
  19. label: Text('全部'),
  20. ),
  21. Chip(
  22. avatar: CircleAvatar(
  23. backgroundColor: Colors.blue.shade900, child: Text('B')),
  24. label: Text('好评 66207'),
  25. ),
  26. Chip(
  27. avatar: CircleAvatar(
  28. backgroundColor: Colors.blue.shade900, child: Text('C')),
  29. label: Text('差评 3913'),
  30. ),
  31. Chip(
  32. avatar: CircleAvatar(
  33. backgroundColor: Colors.blue.shade900, child: Text('D')),
  34. label: Text('点映1'),
  35. ),
  36. Chip(
  37. avatar: CircleAvatar(
  38. backgroundColor: Colors.blue.shade900, child: Text('E')),
  39. label: Text('购票 75985'),
  40. ),
  41. Chip(
  42. avatar: CircleAvatar(
  43. backgroundColor: Colors.blue.shade900, child: Text('F')),
  44. label: Text('认证作者 23'),
  45. ),
  46. Chip(
  47. avatar: CircleAvatar(
  48. backgroundColor: Colors.blue.shade900, child: Text('G')),
  49. label: Text('同城 1496'),
  50. ),
  51. ],
  52. ));
  53. }
  54. }

我们看下实例中 Wrap 布局实现的效果图:

avatar

怎么样,是不是很简单?当我们流式布局效果不是非常复杂的情况下,Wrap 用起来要简单些,如果Wrap实现不了,那么就可以用 Flow 来尝试实现流式布局或瀑布式布局的效果。

3.总结

本节课主要是给大家讲解了 Flutter 的几个常用的流式布局组件 Widget 的用法和特点。Wrap 用法简单些,Flow 用法复杂些,Flow 可以实现更复杂定制化流式或瀑布式布局效果。

发表评论

表情:
评论列表 (有 0 条评论,116人围观)

还没有评论,来说两句吧...

相关阅读