文学起点网
当前位置: 首页 文学百科

ui布局详细讲解(一个月带你入门Flutter)

时间:2023-05-31 作者: 小编 阅读量: 2 栏目名: 文学百科

Flutter系列的文章我会持续更新一个月左右,力求利用1个月带大家入门Flutter,抓住这波技术风口,欢迎大家关注。是否存在Widget之间的覆盖?运行起来的效果如下图所示:第3步:实现按钮行按钮行有3个列,每个列都是同样的布局组成:1个Icon、1个Text。

Flutter系列的文章我会持续更新一个月左右,力求利用1个月带大家入门Flutter,抓住这波技术风口,欢迎大家关注。同时如果觉得这里代码排版不是很舒服的读者可以关注我的微信公众号“IT工匠”,我会同步更新,另外微信公众号上还有很多互联网必备资源(涉及算法、数据结构、java、深度学习、计算机网络、python、Android等互联网技术资料),欢迎大家关注、交流。

摘要

本文通过一个简单的实例来逐步为大家介绍如何在Flutter中构建漂亮的布局,通过本文你将会了解到以下几点:

  • Flutter的布局机制是如何工作的
  • 如何在垂直方向和水平方向布局Widget
  • 如何在Flutter中进行Widget的布局

本文档主要介绍如何在Flutter中进行布局,你将最终会构建一个下图这样的页面:

本文将一步一步带你构建一个像上图那样的页面。

第0步:创建一个Flutter项目
  1. 创建一个Flutter项目
  2. 将该项目的app bar的标题和app的标题设置为下面这样:

Widget build(BuildContext context) {return MaterialApp( title: 'Flutter layout demo', home: Scaffold( appBar: AppBar( title: Text('Flutter layout demo'), ), body: Center( child: Text('Hello World'), ), )); }

设置完成后运行起来效果如下图所示:

第1步:分析页面的布局逻辑

第一步是将整个页面布局分解为基本元素,主要从以下几点入手

  • 辨识行(Row)和列(Column)
  • 布局中是否存在网格(grid)?
  • 是否存在Widget之间的覆盖?
  • 整个UI布局是否需要tab栏?
  • 关注需要对其(alignment)、填充(padding)、边框(border)的区域

那么我们先来看一下整个页面的主要组成部分:

可以看到,整个页面的主要组成部分就是由红色框标注出来的4部分,这4部分位于同一个列(Column),分别是:1个Image、2个Row、1个Text。

再深入地分析一下每一个行(Row):

  • 第1行,也就是标题栏(Title section),它有3列:1个由2行text组成的列(Column)、1个星星的Icon、1个数字:

可以看到,由于第1个列(Column)占据了整个行的大部分空间,所以其应该被Expanded Widget包裹。

  • 第二行,我们也就是按钮列(button section),同样也包含了3个子元素,每一个子元素都是1个行,行里面的内容是1个icon和1个text:
  • 第三行:第三行没有很复杂的构成,就是单纯的text块。

经过以上的分析,我们将一个复杂的页面分解为了多个简单的组成部分,这样可以简化整个页面的实现,为了避免布局代码的混乱,我们应该利用变量和函数来构建布局的子部分,这一点我会在下面的代码中进行演示。

第2步:实现第一行(title Row)

直接给出代码,具体的解释在代码的注释中,大家注意看注释:

Widget titleSection = Container( //为整个Widget(即这一行)的所有所有方向设置32px的填充 padding: const EdgeInsets.all(32), child: Row( children: [ Expanded( /** 将Column放置在Expanded中,由于Expanded会默认占据所有当前Row的空闲可用空间,所以这个Column也会自然被拉伸的占据完所有当前Row可用的空闲空间。 */ child: Column( /**将Column的crossAxisAlignment属性设置为CrossAxisAlignment.start以保证这个列中的元素(即children属性中的Widget)在水平方向上排列在当前Column的起始位置 */ crossAxisAlignment: CrossAxisAlignment.start, children: [ /** 将这个Text放在Container中的目的是通过Container来添加填充(padding) */ Container( padding: const EdgeInsets.only(bottom: 8), child: Text( 'Oeschinen Lake Campground', style: TextStyle( fontWeight: FontWeight.bold, ), ), ), Text( 'Kandersteg, Switzerland', style: TextStyle( color: Colors.grey[500], ), ), ], ), ), /** 最后的2个元素分别是1个Icon和1个Text,分别用来显示星星和数字 */ Icon( Icons.star, color: Colors.red[500], ), Text('41'), ], ),);

直接将上面的变量titleSection放在在app body中即可:

@override Widget build(BuildContext context) { return MaterialApp( title: 'Flutter layout demo', debugShowCheckedModeBanner: false, home: Scaffold( appBar: AppBar( title: Text('Flutter layout demo'), ), body: Column( children: <Widget>[titleSection], ))); }

注意,这里通过使用变量titleSection来达到简化布局代码的目的,这样就不会导致MaterialApp(..)中的代码太长、太混乱。

运行起来的效果如下图所示:

第3步:实现按钮行(button row)

按钮行(button section)有3个列,每个列都是同样的布局组成:1个Icon、1个Text。这一行中的列之间的间距都是均匀的,由于构建每一列的代码几乎相同,因此创建一个名为buildbuttoncolumn()的私有方法,该方法接收一个颜色(Color)参数、一个图标(Icon)参数和1个文本(Text)参数,并返回一个具有以给定颜色绘制的Widget的列(Column),代码如下:

class MyApp extends StatelessWidget { @override Widget build(BuildContext context) { // ··· } Column _buildButtonColumn(Color color, IconData icon, String label) { return Column( mainAxisSize: MainAxisSize.min, mainAxisAlignment: MainAxisAlignment.center, children: [ Icon(icon, color: color), Container( margin: const EdgeInsets.only(top: 8), child: Text( label, style: TextStyle( fontSize: 12, fontWeight: FontWeight.w400, color: color, ), ), ), ], ); }}

_buildButtonColumn()方法将Icon直接添加到Column中,而Text是先用Container进行包裹,然后再将Container添加到Column中,这样做的目的是借助Container为Text设置顶部填充(padding),这样就不至于Text和Icon距离过近。

完成了_buildButtonColumn()函数之后,我们只需要在需要构建列(Column)的时候调用该函数并传入对应的3个参数即可构建出我们需要的列:

Color color = Theme.of(context).primaryColor;Widget buttonSection = Container( child: Row( mainAxisAlignment: MainAxisAlignment.spaceEvenly, children: [ _buildButtonColumn(color, Icons.call, 'CALL'), _buildButtonColumn(color, Icons.near_me, 'ROUTE'), _buildButtonColumn(color, Icons.share, 'SHARE'), ], ),);

注意这里设置Row的mainAxisAlignment属性值为MainAxisAlignment.spaceEvenly,目的是让Row中的列均匀地占满整个行的可用空间。

然后将变量buttonSection放在app body中:

return MaterialApp( title: 'Flutter layout demo', debugShowCheckedModeBanner: false, home: Scaffold( appBar: AppBar( title: Text('Flutter layout demo'), ), body: Column( children: <Widget>[titleSection, buttonSection], )));

运行效果图如下图所示:

第4步:实现Text 部分

将Text部分定义为一个变量,然后将Text放置在一个Container中,并为Container设置padding属性:

Widget textSection = Container( padding: const EdgeInsets.all(32), child: Text( 'Lake Oeschinen lies at the foot of the Blüemlisalp in the Bernese ' 'Alps. Situated 1,578 meters above sea level, it is one of the ' 'larger Alpine Lakes. A gondola ride from Kandersteg, followed by a ' 'half-hour walk through pastures and pine forest, leads you to the ' 'lake, which warms to 20 degrees Celsius in the summer. Activities ' 'enjoyed here include rowing, and riding the summer toboggan run.', softWrap: true, ),);

注意这里设置softwrap属性值为true,这样可以保证文字可以在单词分界的地方换行而不是在单词中间换行。

然后将上面的textSection变量放在app body中:

return MaterialApp( title: 'Flutter layout demo', debugShowCheckedModeBanner: false, home: Scaffold( appBar: AppBar( title: Text('Flutter layout demo'), ), body: Column( children: <Widget>[titleSection, buttonSection, textSection], )));

运行效果图如下图所示:

第5步:实现Image部分

到现在为止我们已经完成了4行中的3行,只剩下图像那行还没实现,这一步我们来实现图像的显示:

  • 在项目的顶级目录下创建一个images文件夹
  • 将下图放在刚才创建的images文件夹下,命名为lake.jpg
  • 更新pubspec.yaml文件,添加assets标签,配置图片路径,这样就可以在代码中访问到你存放的图片:

flutter:uses-material-design: trueassets:- images/lake.jpg

然后我们就可以在代码中引用该图了:

return MaterialApp( title: 'Flutter layout demo', debugShowCheckedModeBanner: false, home: Scaffold( appBar: AppBar( title: Text('Flutter layout demo'), ), body: Column( children: <Widget>[ Image.asset( 'images/lake.jpg', width: 600, height: 240, fit: BoxFit.cover, ), titleSection, buttonSection, textSection ], )));

运行起来效果如下:

第6步:另一种尝试

我们来看前几步完成之后的最终代码:

return MaterialApp( title: 'Flutter layout demo', debugShowCheckedModeBanner: false, home: Scaffold( appBar: AppBar( title: Text('Flutter layout demo'), ), body: Column( children: <Widget>[ Image.asset( 'images/lake.jpg', width: 600, height: 240, fit: BoxFit.cover, ), titleSection, buttonSection, textSection ], )));

注意,我们将body的属性值设置为了Column,然后在Column中放置了我们实现的几个子Widget,现在我们换种方式,使用ListView来取代这个Column:

return MaterialApp( title: 'Flutter layout demo', debugShowCheckedModeBanner: false, home: Scaffold( appBar: AppBar( title: Text('Flutter layout demo'), ), body: ListView( children: [ Image.asset( 'images/lake.jpg', width: 600, height: 240, fit: BoxFit.cover, ), titleSection, buttonSection, textSection ], )));

效果如下:

可以看到,使用Column和使用ListView的静态视觉效果是一样的,那我们到底应该使用Column还是ListView呢?二者的区别在哪里呢?答案就在”动态”,二者在动态下的效果是不一样的,再具体点,使用Column构建的Widget是不支持滚动的,即不支持进行上下或者左右的滚动事件,而使用ListView会支持滚动事件,就像下面这样:

以上就是本文的所有内容,希望大家通过本文可以对Flutter的布局有更深入的了解和认知,我们下篇文章再见~
    推荐阅读
  • 卖火柴的小女孩主要讲了什么(卖火柴的小女孩讲述的是什么)

    卖火柴的小女孩主要讲了什么它讲述的是一个在圣诞夜卖火柴的小女孩的悲惨命运。因为没有卖掉一根火柴,小女孩一天没有吃东西。创作背景:1845年10月,安徒生再次访问意大利。在圣诞节即将到来的时候,格洛斯顿城的朋友们再三邀请安徒生去那里过圣诞节。安徒生兴冲冲地坐上了去格洛斯顿的马车。安徒生的马车在格洛斯顿的街道上缓缓前进,他要体会一下城市里的圣诞节气氛。安徒生回去以后就写了《卖火柴的小女孩》。

  • 芹菜炒木耳大白菜(钙含量是牛奶的3倍)

    韭菜炒豆芽的做法,就是这么简单,掌握好技巧,口感能提升不少,如果你认为韭菜炒豆芽的做法还不错,欢迎收藏起来学着做,或者是分享给更多的人学习!

  • 雅典为什么会繁华(同属希腊地区的雅典和斯巴达)

    古希腊城邦奴隶制得到如此广泛推行为日后城邦时代社会经济的发展以及各行各业的繁荣提供了重要条件。这也使得希腊城邦的奴隶制经济得到了空前规模的发展,尤其是在以雅典和斯巴达两大城邦的代表下,古希腊城邦时代迎来了其奴隶制社会经济发展的黄金时代。雅典型的奴隶制经济代表着古希腊城邦时代奴隶制社会发展的整体面貌,而斯巴达型的奴隶制经济只是局限于那些经济较为落后的城邦当中。

  • 电饭锅腊肠饭做法(电饭锅腊肠饭最简单的做法介绍)

    下面希望有你要的答案,我们一起来看看吧!电饭锅腊肠饭做法主料是腊肠、米,辅料是胡萝卜、玉米、葱、盐、鸡粉。先把材料准备好,大米洗净,腊肠切丁,玉米粒胡萝卜切丁,备用。电饭锅内放少许油,依照顺序放入胡萝卜,玉米腊肠,加入盐,鸡粉,酱油翻炒片刻,接着放入米,水量和平常一样,按下开关煮饭就可以了。开关跳了后,撒上些香葱,加入些酱油调味拌匀即可出锅。

  • 地笼放什么饵料(地笼的介绍)

    下面内容希望能帮助到你,我们来一起看看吧!地笼放什么饵料地笼里面放的饵料比较好的:大骨头稍微有一点点肉屑。还有红虫但数量要多一些。还可以鸡鸭骨头等等腥味越大越好。地笼是中原地区的业余或专业捕鳝或泥鳅,河虾小鱼等的工具。地笼的长度可依据养殖水面的长度、宽度而定,一般为15节左右,总长约7~8米。材质为塑料纤维。地笼有很多入口,但内部构造比较复杂,鱼类进去后就很难出来。

  • 卡路里最低的食物排行(卡路里最低的食物有哪些)

    卡路里最低的食物排行芝麻菜。芝麻菜在日常生活中是比较常见,不仅含有低热量低脂肪和低胆固醇,而且高纤维和大量的维生素A,K,C以及其它丰富的营养。西兰花低热量、丰富维生素、大量矿物质及抗氧化剂,可降低各种癌变的发生率。绿豆芽热量较低,水分和膳食纤维较高。西红柿果实营养丰富,具特殊风味。

  • 新买的内衣有很大的味道怎么回事 新买的内衣有很大的味道怎么回事呀

    不少女性朋友可能有这样的困扰,新买的内衣挂在内衣店里没有异味,可是拿回家洗一遍之后却散发出很大的味道。内衣长期挤压产生味道新买的内衣从生产到购买经过了较长的时间,而在此期间内衣被长时间闷在包装袋里面或者是长时间受到挤压,这也容易使内衣在包装袋中产生很大的味道。

  • 修车最容易被更换的零件(更换下来的旧件哪些可以卖掉)

    可以最大程度上降低我们的损失呢?第4个零部件就是发动机以及空调压缩机了,相信我们都应该知道,车辆当中最为贵重的零部件就是发电机,所以如果我们的车辆出现问题时,发电机也能够卖到一个非常好的价钱。但前提是一定要经过维修之后还可以再次使用的,才能够进行售卖,如果没有维修价值了的话,也只能够是当废铁卖掉。

  • 孕妇晚期腰痛可以按摩吗(孕妇腰酸可以按摩缓解吗)

    孕妇腰酸可以按摩缓解吗孕期随着腹中胎儿越长越大,身体会变得笨重,由于重心前移,增大的子宫压迫脊柱导致背部肌肉紧张,可能就会引起腰酸背痛,这主要是一个生理性的表现怀孕期间运动量太少,体重明显的增加,怀孕的时候身体里面的激素水。

  • qq隐藏图标的方法(操作步骤)

    qq隐藏图标的方法我们打开QQ软件,输入账号密码登录,进入主界面后,点击个人头像,鼠标移到头像上,会出现浮动框,能看到当前使用的应用图标。确定上述操作后,重新查看自己的个人资料,图标已经隐藏成功,操作完以上步骤后,你的QQ好友就看不到你的手机QQ图标了。我们还可以打开QQ,点击最左侧的选项之后选择设置选项,在系统设置中点击登录选项,在主面板中选择在任务栏通知区域显示QQ图标的对勾去掉即可。