Flutter. FractionallySizedBox

欢迎关注我的公众号OpenFlutter,感恩

一个能将子控件的大小调整为可用总空间的一个分数的控件。

对于宽度和高度,FractionallySizedBox 会对它的子控件施加一个紧凑的约束,这个约束是它从父控件在相应轴上接收到的最大约束的一个倍数(通常小于 1.0)。如果某个轴的因子为 null,那么父控件的约束将直接传递给子控件。

dart 复制代码
  return Scaffold(
    body: Center(
        child: FractionallySizedBox(
          widthFactor: 0.97,
          heightFactor: 0.97,
          child: Container(
            color: Colors.blue,
            width: 100,
            height: 100,
          ),
        )),
  );
}

Center 会占据所有屏幕空间,并对其子控件 FractionallySizedBox 施加宽松的 约束。FractionallySizedBox 在两个轴向上都取其 97% 的大小,并将此大小作为紧凑的 约束施加给蓝色的 Container

Container 本身的大小(100x100)被忽略了。在 FractionallySizedBox 内部指定大小通常没有意义。在上面的例子中,我们应该使用 ColoredBox 而不是 Container


对齐只有在父控件提供紧凑的约束时才有效。

dart 复制代码
return Scaffold(
  body: Container(
    color: Colors.red,
    width: 300,
    height: 800,
    child: FractionallySizedBox(
      widthFactor: 0.5,
      heightFactor: 0.5,
      alignment: Alignment.bottomRight,
      child: ColoredBox(
        color: Colors.blue,
      ),
    ),
  ),
);

Container 应用了 300x800 的紧凑约束。ColoredBox 变成了 150x400,并对齐到底部右侧。

恕我直言,上面的例子意义不大,因为在固定大小的 Container 中使用 FractionallySizedBox 没有任何优势。我们可以使用更直接的 SizedBox 来替代。

像我这样的新手会直觉地尝试用它来调整 Column/Row 子控件的大小。

顺便说一句,以下代码是 ChatGPT 建议的:

dart 复制代码
return Scaffold(
  body: Center(
      child: Row(
        children: [
          FractionallySizedBox(
            widthFactor: 0.3,
            child: Container(color: Colors.blue),
          ),
          FractionallySizedBox(
            widthFactor: 0.7,
            child: Container(color: Colors.green),
          ),
        ],
      )),
);

这段代码会触发一个异常:BoxConstraints forces an infinite width(BoxConstraints 强制了一个无限的宽度)。

原因是 Row 沿着主轴(x 轴)对其子控件施加了无限制的 约束。而 0.3 乘以无限仍然是无限。

我们可以使用 Expanded 来解决这个问题。

dart 复制代码
return Scaffold(
  body: Center(
      child: SizedBox(
        height: 100,
        child: Row(
          children: [
            Expanded(
              child: FractionallySizedBox(
                widthFactor: 0.3,
                child: Container(color: Colors.blue),
              ),
            ),
            Expanded(
              child: FractionallySizedBox(
                widthFactor: 0.7,
                child: Container(color: Colors.green),
              ),
            ),
          ],
        ),
      )),
);

但这个结果并不是我所期望的。

也许在某些情况下有用,但通过来说我们想要的是这样的:

而且这可以用更少的代码实现:

dart 复制代码
return Scaffold(
  body: Center(
      child: SizedBox(
        height: 100,
        child: Row(
          children: [
            Flexible( flex: 3, child: Container(color: Colors.blue),),
            Flexible(flex: 7, child: Container(color: Colors.green)),
          ],
        ),
      )),
);

在父控件应用紧凑 约束时,FractionallySizedBox 没有意义;而在无限制约束内部,它又无法正常工作。

那么,它在哪里可以很好地发挥作用呢?

它在那些应用宽松约束minWidth < maxWidth, maxWidth != double.infinity)的控件内部能很好地工作,例如 ScaffoldCenterConstrainedBox

FractionallySizedBox 非常适合创建响应式布局,当你想让控件占据其父控件特定百分比的大小时,它是一个很棒的选择。

最佳应用场景

  • 一个占据可用宽度 80% 的按钮。(EzGlassButton 默认占据 width = double.infinity
dart 复制代码
return Scaffold(
    body: Center(
      child: FractionallySizedBox(
        widthFactor: 0.8,
        child: EzGlassButton(
          child: EzText('CLICK ME!', fontSize: 24,),
        ),
      ),
    ));

这在大屏幕上看起来太大了:

我们可以使用 ConstrainedBox 来进一步改进它。

dart 复制代码
return Scaffold(
    body: Center(
      child: ConstrainedBox(
        constraints: BoxConstraints(
          maxWidth: 500,
        ),
        child: FractionallySizedBox(
          widthFactor: 0.8,
          child: EzGlassButton(
            child: EzText('CLICK ME!', fontSize: 24,),
          ),
        ),
      ),
    ));

现在它的大小与其父控件成比例,但宽度不能超过 500 像素。


ConstrainedBox 包裹的 FractionallySizedBox 是一个非常强大的组合,在响应式 Flutter 设计中经常使用。

dart 复制代码
return Scaffold(
    body: Center(
      child: Wrap(
        children: [
          ConstrainedBox(
            constraints: BoxConstraints(
              maxWidth: 400,
            ),
            child: FractionallySizedBox(
                widthFactor: 0.9,
                child: Container(color: Colors.red, height: 150, )
            ),
          ),
          ConstrainedBox(
            constraints: BoxConstraints(
              maxWidth: 400,
            ),
            child: FractionallySizedBox(
                widthFactor: 0.9,
                child:  Container(color: Colors.blue, height: 150, )
            ),
          ),
          ConstrainedBox(
            constraints: BoxConstraints(
              maxWidth: 400,
            ),
            child: FractionallySizedBox(
                widthFactor: 0.9,
                child:  Container(color: Colors.orange, height: 150, )
            ),
          ),
        ],
      ),
    ));

在任何手机设备上,Container 会占据屏幕宽度的 90% 并垂直显示。然而,在桌面设备上,它们的宽度会受到 ConstrainedBox 的限制,并水平显示。

就是这些了。

感谢您的阅读,祝您设计愉快!

相关推荐
梵得儿SHI13 小时前
Vue 开发环境搭建全指南:从工具准备到项目启动
前端·javascript·vue.js·node.js·pnpm·vue开发环境·nvm版本管理
八月ouc13 小时前
每日小知识点:10.14 webpack 有几种文件指纹
前端·webpack
苏琢玉13 小时前
从 Hexo 到 Astro:重构我的个人博客
前端·hexo
街尾杂货店&13 小时前
webpack - 单独打包指定JS文件(因为不确定打出的前端包所访问的后端IP,需要对项目中IP配置文件单独拿出来,方便运维部署的时候对IP做修改)
前端·javascript·webpack
月光技术杂谈13 小时前
用Deepseek 实现一个基于web的扣图应用
前端·javascript·html5·ccs·tensorflow.js·canvas api
金梦人生14 小时前
Css性能优化
前端·css
Holin_浩霖14 小时前
UI设计的底层逻辑:从组件到系统的跃迁
前端
Holin_浩霖14 小时前
前端开发者的 Web3 全图解实战 二
前端
写代码的皮筏艇14 小时前
CSS属性继承与特殊值
前端·css
kevlin_coder14 小时前
🚀 实现同一个滚动区域包含多个虚拟滚动列表
前端·javascript