flutter布局详解及代码示例(上)

news2024/12/29 11:17:58

布局

基本布局

  • Row(水平布局):在水平(X轴)方向上排列子widget的列表。
  • Column(垂直布局):在垂直(Y轴)方向上排列子widget的列表。
  • Stack(可重叠布局):可以允许其子widget简单的堆叠(Z轴)在一起。
  • IndexedStack(单一显示可重叠布局):相比Stack后面编码的widget显示在Z轴上方,IndexedStack通过index可以控制显示哪个widget显示在Z轴的上方,注意,只显示一个。
  • ListView(滚动列表):最常用的滚动widget。

Row

  • 不带滚动属性,如果超出约束范围,在debug版本下会提示溢出。

代码

Row(
   crossAxisAlignment: CrossAxisAlignment.start,
   children: <Widget>[
       new Text('1'),
       new Text('2'),
       new Text('3'),
   	],
)

常用参数

mainAxisAlignment
  • X轴方向上的对齐方式
  • spaceBetween:沿X轴方向均分,子widget之间的空白区域相等,两侧贴边,比如只有两个子widget就在两边,三个就左中右这样。
  • spaceAround:沿X轴方向均分,左右两侧不贴边,有空白区域,但空白区域是子widget之间的空白区域的1/2。
  • spaceEvenly:沿X轴方向均分,左右两侧不贴边,有空白区域,空白区域和子widget之间的空白区域大小一致。
crossAxisAlignment
  • Y轴方向上的对齐方式
  • stretch:让子widget充满Y轴。
  • baseline:在Y轴上和基线适配,搭配textBaseline用
textBaseline
  • 文本内容的基线类型,就是文本的水平边界线,注意,只有水平
  • alphabetic:以字母字符底部的水平线对齐(不论中英文,都是用文字的最底部对齐)
  • ideographic:以表意字符的水平线对齐。(英文如pgj这些底部会比中文矮一些)
mainAxisSize
  • X轴方向应占用多少空间,默认最大化的占用主轴空间。
  • 若为MainAxisSize.min,则mainAxisAlignment只能为start。
verticalDirection
  • 垂直方向上的排列顺序
  • up:子widget从bottom开始往top排列;
  • down:子widget从top开始往bottom排列。
  • 搭配crossAxisAlignment的start和end属性有如下不同:
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
textDirection
  • 与verticalDirection类似,水平方向上的排列顺序
  • ltr:left to right,从左到右排列;
  • rtl:rigth to left,从右到左排列;
  • 同verticalDirection类似,搭配mainAxisAlignment使用,如下:
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

Column

  • 不带滚动属性,如果超出约束范围,在debug版本下会提示溢出。

代码

Column(
   crossAxisAlignment: CrossAxisAlignment.start,
   children: <Widget>[
       new Text('1'),
       new Text('2'),
       new Text('3'),
   	],
)

常用参数

mainAxisAlignment
  • Y轴方向上的对齐方式
  • spaceBetween:沿Y轴方向均分,子widget之间的空白区域相等,比如只有两个子widget就在上下,三个就上中下这样。
  • spaceAround:沿Y轴方向均分,上下两侧不贴边,有空白区域,但空白区域是子widget之间的空白区域的1/2。
    • spaceEvenly:沿Y轴方向均分,上下两侧不贴边,有空白区域,空白区域和子widget之间的空白区域大小一致。
crossAxisAlignment
  • X轴方向上的对齐方式
  • stretch:让子widget充满X轴。
  • baseline:相当于start,因为textBaseline只有水平,因为没有文字是需要人歪头90度看的。
mainAxisSize
  • Y轴方向应占用多少空间,默认最大化的占用主轴空间。
  • 若为MainAxisSize.min,则mainAxisAlignment只能为start。
verticalDirection
  • 垂直方向上的排列顺序
  • up:子widget从bottom开始往top排列;
  • down:子widget从top开始往bottom排列。
  • 搭配mainAxisAlignment的start和end属性有如下不同:
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
textDirection
  • 与verticalDirection类似,水平方向上的排列顺序
  • ltr:left to right,从左到右排列;
  • rtl:rigth to left,从右到左排列;
  • 同verticalDirection类似,搭配crossAxisAlignment使用,如下:
    外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

Stack

  • 可以子widget无视位置压在另一个子widget的布局
  • 内部使用Positioned定位子widget位置宽高
    • width,left和right三个属性不能同时使用,必须至少有一个属性是null;
    • height,top和bottom三个属性不能同时使用,必须至少有一个属性是null。

代码

Stack(
  alignment: Alignment(-1.0,0.0),//等价于Alignment.centerLeft
  children: <Widget>[
    Positioned(child: Container(width: 100.0,height: 100.0,color: Colors.red,),),
    Positioned(child: Container(width: 90.0,height: 90.0,color: Colors.green,),),
    Positioned(child: Container(width: 80.0,height: 80.0,color: Colors.blue,),),
    new Text('-1,0\ncenterLeft',textAlign: TextAlign.left,),
  ],
)

完整示例

Container(
    width: 200,
    height: 200,
    child: Stack(
        children: <Widget>[
        Positioned(
            // 若是不指定位置就是(0,0)
            // 若是不指定size就是子view的size
            // top: 50.0,
            // left: 50.0,
            child: Container(
                width: 200,
                height: 200,
            ),
        ),
        Positioned(
            top: 0.0,
            left: 190.0,
            child: Container(
            width: 10.0,
            height: 10.0,
            color: Colors.green,
            ),
        ),
        ],
    ),
);

常用参数

alignment
  • 可以传递AlignmentDirectional、Alignment、FractionalOffset三大类,主要用的是带方向的AlignmentDirectional。
textDirection
  • 搭配AlignmentDirectional的start或者end实现类似Row的mainAxisAlignment+textDirection的效果。
fit
  • loose:从Stack传递过来的约束被解除,即:
    • 当Stack大小被限制为300*400,子widget元素的大小将可设置为,宽:0到300,高:0到400。
  • expand:从Stack传递过来的约束被锁定到最大,即:
    • 当Stack大小被限制为300*400,子widget元素的大小将锁定为,宽:300,高:400。
  • passthrough:不改变子组件约束条件。和loose一致。
overflow
  • 是否裁剪溢出的子widget元素
  • clip:裁剪溢出部分
  • visible:显示溢出部分

IndexedStack

  • 相比Stack多了一个index参数,控制显示哪个子widget
  • 内部使用Positioned定位子widget位置宽高
    • width,left和right三个属性不能同时使用,必须至少有一个属性是null;
    • height,top和bottom三个属性不能同时使用,必须至少有一个属性是null。

代码

IndexedStack(
  sizing: StackFit.expand,
  index: 1,
  children: <Widget>[
    Positioned(child: Container(width: 100.0,height: 100.0,color: Colors.red,),),
    Positioned(child: Container(width: 130.0,height: 90.0,color: Colors.green,),),
    Positioned(child: Container(width: 80.0,height: 120.0,color: Colors.blue,),),
   ],
),

index就是children的length的区间之中的值。
比如上面的代码,就只显示index为1的子widget,也就是第二个。

常用参数
  • 因为只显示一个,所以Stack的常用参数基本用不到

ListView

    • listview需要拓展长度,比如是纵向的listview,就需要指定父view的height。
  • 可以理解为可以滚动的Column。
  • 有四种构造方式
    • List():默认构造;
      • 直接把子widget硬编码,如果子widget较多,有性能风险。
    • ListView.builder();
      • 不带分割线
      • 懒加载,只会渲染可见的条目。
    • ListView.separated;
      • 带分割线
      • separatorBuilder是必传参数,用于构造分割线view
      • 分割线返回的是一个widget(不必一定是Divider),所以可以自定义
    • ListView.custom。
      • 搭配SliverChildBuilderDelegate自定义实现不同类型子widget,比如子widget和分割线交替出现等复杂场景。

代码

/// 默认构造
ListView(
  children: [ 
    new Text('1'),
    Divider(),
    new Text('2'),
    Divider(),
    new Text('3'),
    Divider(),
  ],
),
/// ListView.builder()构造
final List<String> name = <String>['1','2','3',];
ListView.builder(
  itemCount: name.length,
  itemBuilder: (context, index) {
    return Text(name[index]);
  },
),
/// ListView.separated()构造
ListView.separated(
  itemBuilder: (context, index) {
    return Text(name[index]);
  },
  separatorBuilder: (context, index) {
    return Divider();//分割线
  },
  itemCount: name.length,
),
/// ListView.custom()
ListView.custom(
  childrenDelegate: SliverChildBuilderDelegate(
    (context, index) {
      return Text(name[index]);
    },
    childCount: name.length,
  ),
),

详细代码

var lv1 = ListView(
    //主轴滚动方向:垂直vertical,竖直horizontal ,默认为垂直vertical,
    scrollDirection: Axis.vertical,
    //是否反向 默认为false,正常顺序从起始点开始正序,true为从末尾开始排列
    reverse: false,
    //滚动控制器,默认为null
    controller:null,
    //是否强制滚动(顶部或底部时是否可以滚动),默认是false,如果为true,Controller必须为null
    primary: false,
    //视图如何响应用户的手势滑动,有内置实现强制可以滚动 const AlwaysScrollableScrollPhysics();和强制不可以滚动const ScrollPhysics(),可忽略primary属性
    physics: const AlwaysScrollableScrollPhysics(),
    //默认为false,滚动视图在[滚动方向]中的范围是否应由正在查看的内容决定。
    shrinkWrap: true,
    //item的padding值
    padding: const EdgeInsets.all(10.0),
    //item交叉轴方向的大小,默认自适应,vertical时为高度,horizontal时为高度
    itemExtent: 50.0,
    //是否自动保存滑出屏幕外的字widget的状态,保证widget不被回收,可复用,false的手动保存。默认为true
    addAutomaticKeepAlives: true,
    //是否放置到重绘列表中,复杂widget可提高性能,默认为true
    addRepaintBoundaries: true,
    //缓存区大小,默认为250
    cacheExtent: 250.0,
    //默认构造函数中特有,直接将子widget放置内,一次性渲染完成,适合少量数据
    children: [
        new Text('测试1'),
        Divider(),
        new Text('测试2'),
        Divider(),
        new Text('测试3'),
        Divider(),
        new Text('测试4'),
        Divider(),
        new Text('测试5'),
        Divider(),
    ],
);

每个item不同的listview

  • 用switch-case返回不同的item来实现不同的item的view
import 'package:flutter/material.dart';

class MyListPage extends StatelessWidget {
  final List<Item> itemList = [
    Item(type: ItemType.button, data: ['Button 1', 'Button 2']),
    Item(type: ItemType.text, data: 'Hello World'),
    Item(
        type: ItemType.image,
        data: 'https://example.com/images/image.jpg'),
  ];

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text('My List Page'),
      ),
      body: ListView.builder(
        itemCount: itemList.length,
        itemBuilder: (BuildContext context, int index) {
          Item item = itemList[index];

          Widget itemWidget;

          // 根据item类型构建不同的widget
          switch (item.type) {
            case ItemType.button:
              List<String> buttonTexts = item.data;
              itemWidget = Row(
                children: [
                  ElevatedButton(
                    onPressed: () {},
                    child: Text(buttonTexts[0]),
                  ),
                  ElevatedButton(
                    onPressed: () {},
                    child: Text(buttonTexts[1]),
                  ),
                ],
              );
              break;

            case ItemType.text:
              String textData = item.data;
              itemWidget = Text(textData);
              break;

            case ItemType.image:
              String imageUrl = item.data;
              itemWidget = Image.network(imageUrl);
              break;
          }

          return ListTile(
            title: itemWidget,
          );
        },
      ),
    );
  }
}

class Item {
  final ItemType type;
  final dynamic data;

  Item({
    required this.type,
    required this.data,
  });
}

enum ItemType {
  button,
  text,
  image,
}

垂直滑动里嵌套水平滑动

class _PageBizCertificateState extends eState<PageBizCertificate> {

  final List<String> items = List.generate(20, (index) => 'Item $index');

  @override
  Widget pageBody(BuildContext context) {
    return ListView.builder(
      itemCount: items.length,
      itemBuilder: (context, index) {
        return MyListItem(title: items[index]);
      },
    );
  }

  Widget MyListItem({required String title}) {
    return Container(
      padding: EdgeInsets.all(16.0),
      child: Column(
        crossAxisAlignment: CrossAxisAlignment.start,
        children: [
          Text('第一行文本'),
          SizedBox(height: 8.0),
          MyHorizontalScrollList(),
          SizedBox(height: 8.0),
          Text('第三行文本'),
        ],
      ),
    );
  }

  Widget MyHorizontalScrollList() {
    return Container(
      height: 100.0,
      child: ListView.builder(
        scrollDirection: Axis.horizontal,
        itemCount: 10,
        itemBuilder: (context, index) {
          return Container(
            width: 100.0,
            margin: EdgeInsets.all(4.0),
            color: Colors.blue,
            child: Center(
              child: Text('水平项 $index'),
            ),
          );
        },
      ),
    );
  }

}

参数列表

  • scrollDirection:主轴滚动方向;默认垂直滚动Axis.vertical,还可水平滚动Axis.horizontal
  • reverse:是否反向排列,默认false,即正序排列
  • controller:滚动控制器,默认为null
  • primary:是否强制滚动,默认false
  • physics:响应用户滑动。
    • NeverScrollableScrollPhysics(), // 禁止滑动
  • shrinkWrap:滚动视图在滚动方向中的范围是否应由正在查看的内容决定。默认为false
  • padding:距边框item边间距
  • cacheExtent:缓存大小,默认250.0
  • itemExtent:item在交叉轴方向上的长度;null时为自适应
  • addAutomaticKeepAlives:是否自动保存滑出屏幕外的字widget的状态,默认true,保存,可复用
  • addRepaintBoundaries:是否放置到重绘列表中;默认true,可提高性能
  • itemBuilder:builder和separated构造函数特有要传入参数,定义item的样式
  • separatorBuilder:separated构造函数特有要传入的参数,定义分割线的样式
  • itemCount:列表可滚动的item数量,默认构造没有
  • childrenDelegate:内部都是靠此类实现滚动和布局,但只有custom构造函数需要传入,其余都内部实现

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1260715.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【九章斩题录】Leetcode:面试题 01.03. URL化(C/C++)

精品题解 &#x1f525; 《九章斩题录》 &#x1f448; 猛戳订阅 面试题 01.03. URL化 &#x1f4da; 题目&#xff1a;URL化。编写一种方法&#xff0c;将字符串中的空格全部替换为%20。假定该字符串尾部有足够的空间存放新增字符&#xff0c;并且知道字符串的“真实”长度。…

本地训练,立等可取,30秒音频素材复刻霉霉讲中文音色基于Bert-VITS2V2.0.2

之前我们使用Bert-VITS2V2.0.2版本对现有的原神数据集进行了本地训练&#xff0c;但如果克隆对象脱离了原神角色&#xff0c;我们就需要自己构建数据集了&#xff0c;事实上&#xff0c;深度学习模型的性能和泛化能力都依托于所使用的数据集的质量和多样性&#xff0c;本次我们…

使用最小花费爬楼梯(力扣LeetCode)动态规划

使用最小花费爬楼梯 题目描述 给你一个整数数组 cost &#xff0c;其中 cost[i] 是从楼梯第 i 个台阶向上爬需要支付的费用。一旦你支付此费用&#xff0c;即可选择向上爬一个或者两个台阶。 你可以选择从下标为 0 或下标为 1 的台阶开始爬楼梯。 请你计算并返回达到楼梯顶…

Springboot日志-logback

logback-spring.xml的配置项 共有一个父标签、两种属性、三个节点: 一个父标签&#xff1a;configuration 两种属性&#xff1a;contextName和property 三个节点&#xff1a;appender、root、logger 日志级别 日志级别从低到高分为TRACE < DEBUG < INFO < WARN &…

telnet-MISC-bugku-解题步骤

——CTF解题专栏—— 题目信息&#xff1a; 题目&#xff1a;这是一张单纯的图片 作者&#xff1a;未知 提示&#xff1a;无 解题附件&#xff1a; 解题思路&#xff1a; (⊙﹏⊙)这是个什么文件pcap文件分析_pcap文件打开-CSDN博客查了一下&#xff0c;但没看懂&#xff0c…

人工智能关键技术决定机器人产业的前途

人工智能&#xff08;Artificial Intelligence&#xff0c;AI&#xff09;是指让计算机或机器具有类似于人类的智能和学习能力的技术。人工智能技术与机器人技术的结合将改变传统的机器人行业格局&#xff0c;就像智能手机对传统手机的颠覆一样。本文从人工智能技术的发展趋势、…

使用STM32和蓝牙模块进行无线数据传输的实践

无线数据传输在现代通信领域中具有重要的地位&#xff0c;而蓝牙技术是一种常用的无线数据传输技术。本文介绍了如何使用STM32微控制器和蓝牙模块实现无线数据传输的方案&#xff0c;包括硬件设计、蓝牙模块配置、数据发送和接收等步骤&#xff0c;并给出相应的代码示例。 一、…

从中序与后序遍历序列构造二叉树(C++实现)

从中序与后序遍历序列构造二叉树 题目思路代码代码讲解 题目 力扣&#xff1a;从中序与后序遍历序列构造二叉树 思路 代码 class Solution { public:TreeNode* _build(vector<int>& inorder, vector<int>& postorder,int & peri,int lefti,int right…

2023.11.27如何使用内网穿透工具实现Java远程连接操作本地Elasticsearch搜索引擎

文章目录 前言1. Windows 安装 Cpolar2. 创建Elasticsearch公网连接地址3. 远程连接Elasticsearch4. 设置固定二级子域名 前言 简单几步,结合Cpolar内网穿透工具实现Java远程连接操作本地Elasticsearch。 什么是elasticsearch&#xff1f;一个开源的分布式搜索引擎&#xff0…

普通表计读数开发思路

一、普通表计类型介绍&#x1f349; 常见的普通表计有SF6&#xff0c;压力表&#xff0c;油位表&#xff08;指针类&#xff09;等。 图1&#xff1a;( 压力表) 图2&#xff1a;&#xff08;油位表-指针类&#xff09; 图3&#xff1a;&#xff08;SF6表&#xff09; 图4:&a…

系列二十一、Spring中bean的创建顺序

一、概述 我们知道启动IOC容器时&#xff0c;Spring会为我们创建各种各样的bean&#xff0c;那么思考一个问题&#xff0c;bean的创建顺序是由什么决定的呢&#xff1f;答&#xff1a;bean的创建顺序是由BeanDefinition的注册信息决定的&#xff0c;这个其实很好理解&#xff0…

linux安装部署redis

1、下载redis包2、解压3、进入解压路径编译安装4、修改配置文件使redis后台运行5、启动 1、下载redis包 https://redis.io/download/ 2、解压 tar -zxvf redis-7.2.3.tar.gz3、进入解压路径编译安装 cd redis-7.2.3 make && make install默认安装路径&#xff1a; …

CocosCreator 之 Tween缓动系统的使用

版本&#xff1a; 3.4.0 语言&#xff1a; TypeScript 环境&#xff1a; Mac 简介 在CocosCreator 3.x版本后&#xff0c; Tween缓动系统代替了原有的Action动作。官方使用缓动系统的主要目的之一是用于解决离线动画无法满足需求时的动态动画问题。 简单的示例&#xff1a; …

Linux文件操作应用及open和fork

1.文件操作的应用: 1).打开一个文件并往里面写入hello: #include <stdio.h> #include <unistd.h> #include <stdlib.h> #include <fcntl.h> #include <assert.h> int main() { int fdopen("file.txt",O_WRONLY|O_CREAT,0600); …

oracle免费资源 终止实例 以及新建一台实例的折腾记录

事情的背景是这样的&#xff0c;我的一台oracle小鸡&#xff0c;不太好用的样子&#xff0c;有时候SSH连不上&#xff0c;有时候莫名其妙卡住。所以我就想把它重新安装一下系统&#xff0c;恢复成最初的样子。 然后在网上查资料&#xff0c;是有办法把系统重装一下的。但是略微…

pdf加密文件解密(pdf文件解密小工具)

工具放在文章末尾&#xff01; 1.pdf文件加密后会有很多使用权限的限制很不方便&#xff0c;只要是为了pdf的数据不被二次利用&#xff0c;未加密的pdf功能都是可以正常使用的 2.加密后的pdf使用权限会被限制部分 3.工具只能解决pdf编辑等加密情况&#xff0c;不能解决文件打…

深度学习及其基本原理

深度学习的 Ups and Downs概念区分神经网络的构成深度学习基本原理深度学习的普遍近似定理扩展&#xff1a;反卷积网络——可视化每一层提取的特征 深度学习的 Ups and Downs 1958&#xff1a;感知机&#xff08;线性模型&#xff09;1969&#xff1a;感知机有局限性1980s&…

stream流和方法引用

1.Stream流 1.1体验Stream流【理解】 案例需求 按照下面的要求完成集合的创建和遍历 创建一个集合&#xff0c;存储多个字符串元素把集合中所有以"张"开头的元素存储到一个新的集合把"张"开头的集合中的长度为3的元素存储到一个新的集合遍历上一步得到的集…

Go 基本语法

一、​​​​变量定义方法 var 定义变量 var 变量名 类型 表达式 var name string "Snail" var age int 21 var isOK bool bool 2.类型推导方式定义变量 a 在函数内部&#xff0c;可以使用更简略的: 方式声明并初始化变量**注意&#xff1a;**短变量只能用于声…

MySQL数据库如何实现跨服务器访问数据

点击上方蓝字关注我 在使用MySQL数据库时&#xff0c;很多同学经常会问&#xff0c;我能跨服务器访问另一库的数据么&#xff1f;得到的答案很多时候是让人失望的。那么如果真的需要访问&#xff0c;又不想使用拷贝表及数据的方式&#xff0c;可以实现么&#xff0c;又该如何实…