Flutter利用GridView创建网格布局实现优美布局

请添加图片描述

文章目录

  • 简介
  • 使用详解
    • 导入依赖项
    • 创建一个基本的 GridView
    • 一些参数说明
    • 使用GridView.count来构造
  • 其他控制
  • 总结

简介

GridView 是 Flutter 中用于创建网格布局的强大小部件。它允许你在行和列中排列子小部件,非常适合显示大量项目,例如图像、文本、卡片等。

使用详解

以下是关于如何使用 GridView 控件的详细讲解:

导入依赖项

在你的 Flutter 项目中,首先确保已经导入了 flutter/material.dart 包,因为 GridView 是 material 包的一部分,然后你可以使用以下代码创建一个简单的 GridView:

import 'package:flutter/material.dart';

创建一个基本的 GridView

下面是一个基本的 GridView 示例,它创建了一个包含一组简单文本小部件的网格:

GridView(
  gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
    crossAxisCount: 2, // 列数
    mainAxisSpacing: 10.0, // 垂直间距
    crossAxisSpacing: 10.0, // 水平间距
  ),
  children: <Widget>[
    Container(
      color: Colors.red,
      child: Center(child: Text('Item 1')),
    ),
    Container(
      color: Colors.green,
      child: Center(child: Text('Item 2')),
    ),
    Container(
      color: Colors.blue,
      child: Center(child: Text('Item 3')),
    ),
    // 可以继续添加更多的子小部件
  ],
)

在这个示例中,我们创建了一个包含三个列的网格(crossAxisCount: 2)。每个网格项都由一个带有不同背景颜色的 Container 包装,并且包含一个居中对齐的文本小部件。

一些参数说明

SliverGridDelegateWithFixedCrossAxisCount
gridDelegate 属性是 GridView 的一个关键部分,它用于定义网格的布局方式。在上面的示例中,我们使用了 SliverGridDelegateWithFixedCrossAxisCount,它采用以下参数:

crossAxisCount:指定了列数。
mainAxisSpacing:指定了主轴(通常是垂直轴)上的间距。
crossAxisSpacing:指定了交叉轴(通常是水平轴)上的间距。
使用 builder 构建动态网格
如果你有一个大数据源,并且不想一次性加载所有数据,你可以使用 GridView.builder,它会根据需要生成网格项。以下是一个示例:

GridView.builder(
  gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
    crossAxisCount: 2,
    mainAxisSpacing: 10.0,
    crossAxisSpacing: 10.0,
  ),
  itemCount: yourData.length, // 数据源的长度
  itemBuilder: (BuildContext context, int index) {
    return Container(
      color: Colors.blue,
      child: Center(child: Text('Item $index')),
    );
  },
)

在这个示例中,itemBuilder 函数会根据索引值生成每个网格项。你可以根据实际需求从 yourData 数据源中获取数据。

使用GridView.count来构造

GridView.count(
              crossAxisCount: 4,
              children: List.generate(snapshot.data?.length as int, (index) {
                return Center(
                  child: Padding(
                    padding: const EdgeInsets.all(6.0),
                    child: Column(
                      mainAxisAlignment: MainAxisAlignment.center,
                      children: [
                        GestureDetector(
                          onTap: () {
                            print('Button $index is pressed,${snapshot.data?[index].sptime},${snapshot.data?[index].cycle},${snapshot.data?[index].retime}');
                            // 按钮点击事件
                            AppRouter.router.navigateTo(
                              context,
                              '/DoSportsPage/${snapshot.data?[index].sptime}/${snapshot.data?[index].cycle}/${snapshot.data?[index].retime}',
                              transition: TransitionType.fadeIn,
                            );
                            // Navigator.of(context).pop();
                            // Navigator.of(context).push(MaterialPageRoute(builder: (context) => const SportsShow()));
                          },
                          //long time press button to delete data
                          onLongPress: ( ) {
                            print('Button $index is longpressed id= ${snapshot.data?[index].id}');
                            DatabaseHelper.dbhelper.delete(snapshot.data?[index].id);
                            setState(() {
                              //show again
                            });
                          },
                          child: CircleAvatar(
                            child: Text(snapshot.data?[index].title.substring(0, 1)),
                            // radius: 30,
                            // backgroundImage: AssetImage('assets/touxiang.jpg'),
                            // backgroundColor: Colors.white,
                          ),
                        ),
                        const SizedBox(height: 8),
                        Text(
                          snapshot.data![index].title,
                          style: const TextStyle(fontSize: 12),
                        ),
                      ],
                    ),
                  ),
                );
              }),
            );
          } else {
            return const CircularProgressIndicator();
          }

这段代码是使用 GridView.count 构建一个网格布局,其中 crossAxisCount 属性定义了网格的列数,而 children 属性用于指定网格中的子小部件列表。

GridView.count: 这是一个 GridView 的构造函数,用于创建一个具有固定列数的网格布局。

crossAxisCount: 这是 GridView.count 构造函数的一个参数,它指定了网格的列数。在你的示例中,crossAxisCount 设置为 4,表示该网格将包含四列。

children: 这是一个包含要在网格中显示的子小部件的列表。在你的示例中,它使用 List.generate 函数生成了一个包含 snapshot.data?.length 个元素的列表。snapshot.data?.length 是一个可能为空的数据长度,因此在此处使用 ?. 运算符以避免出现空指针异常。

List.generate 函数接受两个参数:

第一个参数是要生成的列表的长度,这里是 snapshot.data?.length。

第二个参数是一个回调函数,用于生成列表中每个元素的内容。在你的示例中,回调函数 (index) 生成了网格中每个元素的内容。

所以,这段代码的作用是创建一个包含固定列数(4列)的网格,其中每个格子的内容由 List.generate 函数生成,数量等于 snapshot.data?.length。通常,snapshot.data 包含从数据源检索到的数据,而这段代码将这些数据以网格的形式进行展示。

其他控制

在 Flutter 中,GridView 提供了各种属性和参数,以便你可以配置滚动方向、内容填充以及滚动控制。以下是一些常用的属性和参数:

滚动方向(scrollDirection):
使用 scrollDirection 属性可以设置 GridView 的滚动方向。默认情况下,它是垂直方向的,但你可以将其设置为水平方向。

GridView(
  scrollDirection: Axis.horizontal, // 设置为水平滚动
  // 其他属性...
)

内容填充(padding):
使用 padding 属性可以设置 GridView 内容的内边距。你可以在内边距中定义空白区域,以便在网格周围留出一些空间。

GridView(
  padding: EdgeInsets.all(16.0), // 设置内边距为16.0
  // 其他属性...
)

滚动控制(controller):
你可以使用 controller 属性来自定义滚动控制器,以实现更高级的滚动控制功能。例如,你可以创建一个 ScrollController 并将其传递给 GridView,以便监听滚动位置、滚动到特定位置等。

ScrollController _controller = ScrollController();

GridView(
  controller: _controller,
  // 其他属性...
)

这样,你可以使用 _controller 对象来执行滚动操作,例如:

滚动到特定位置:_controller.jumpTo(offset) 或 _controller.animateTo(offset, duration: Duration(milliseconds: 500), curve: Curves.ease)

监听滚动位置:可以添加监听器 _controller.addListener(() { /* 处理滚动位置的变化 */ })

这些属性和参数使你能够根据需求对 GridView 进行自定义和控制,以创建不同类型的滚动网格布局。

总结

GridView 持许多其他配置选项,如滚动方向、内容填充、滚动控制等。你可以根据你的应用需求进行进一步的定制。总的来说,GridView 是一个非常灵活且强大的小部件,可用于创建各种网格布局。


结束语
Flutter是一个由Google开发的开源UI工具包,它可以让您在不同平台上创建高质量、美观的应用程序,而无需编写大量平台特定的代码。我将学习和深入研究Flutter的方方面面。从基础知识到高级技巧,从UI设计到性能优化,欢饮关注一起讨论学习,共同进入Flutter的精彩世界!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

js树形数组遍历练习,扁平化、格式化、获取节点父级

1.树形数组扁平化 数组扁平化的方式很多&#xff0c;这里主要是用递归处理&#xff0c;除此之外还有正则、扩展运算符等等 const list [{name:1,id:1,children:[{name:11,id:11,children:[{name:111,id:111}]},{name:12},]},{name:2,id:2,children:[{name:21,id:21,children:…

数据结构-栈和队列(一)

目录 1.栈 1.1 栈的概念及结构 1.2 栈的实现 初始化栈&#xff1a; 入栈&#xff1a; 判空函数&#xff1a; 出栈&#xff1a; 获取栈顶元素&#xff1a; 获取栈中有效元素的个数&#xff1a; 销毁栈&#xff1a; 完整代码&#xff1a; 测试&#xff1a; 2. 队列 …

在 Python 中使用 Selenium 按文本查找元素

我们将通过示例介绍在Python中使用selenium通过文本查找元素的方法。 在 Python 中使用 Selenium 按文本查找元素 软件测试是检查应用程序是否满足用户需求的技术。 该技术有助于使应用程序成为无错误的应用程序。 软件测试可以手动完成&#xff0c;也可以通过某些软件完成。…

解决Docker启动之npm版本不兼容问题

报错内容&#xff1a; npm WARN read-shrinkwrap This version of npm is compatible with lockfileVersion1, but package-lock.json was generated for lockfileVersion2. Ill try to do my best with it! npm WARN tar ENOENT: no such file or directory, open /home/wvp-…

ConcurrentHashMap 底层具体实现知道吗?实现原理是什么?

&#xff08;本文引自mic老师文档&#xff09; 之前分享过一期 HashMap 的面试题&#xff0c;然后有个小伙伴私信我说&#xff0c;他遇到了一个 ConcurrentHashMap 的问题不知道怎么回答。 于是&#xff0c;就有了这一期的内容&#xff01;&#xff01; …

Skywalking介绍

一个优秀的项目&#xff0c;除了具有高拓展的架构、高性能的方案、高质量的代码之外&#xff0c;还应该在上线后具备多角度的监控功能。现在企业中的监控服务也有很多&#xff0c;Skywalking除了提供多维度、多粒度的监控之外&#xff0c;也提供了良好的图形化界面以及性能剖析…

Python--- lstrip()--删除字符串两边的空白字符、rstrip()--删除字符串左边的空白字符、strip()--删除字符串右边的空白字符

strip() 方法主要作用&#xff1a;删除字符串两边的空白字符&#xff08;如空格&#xff09; lstrip() 方法 left strip&#xff0c;作用&#xff1a;只删除字符串左边的空白字符 rstrip() 方法&#xff0c;作用&#xff1a;只删除字符串右边的空白字符 strip 英 /strɪp…

Ansible playbook自动化运维工具详解

Ansible playbook自动化运维工具详解 一、playbook的相关知识1.1、playbook 的简介1.2、playbook的 各部分组成 二、基础的playbook剧本编写实例三、 playbook的定义、引用变量3.1、基础变量的定义与引用3.2、引用fact信息中的变量 四、playbook中的when条件判断和变量循环使用…

P02项目诊断报警组件(学习操作日志记录、单元测试开发)

★ P02项目诊断报警组件 诊断报警组件的主要功能有&#xff1a; 接收、记录硬件设备上报的报警信息。从预先设定的错误码对照表中找到对应的声光报警和蜂鸣器报警策略&#xff0c;结合当前的报警情况对设备下发报警指示。将报警消息发送到消息队列&#xff0c;由其它组件发送…

Collection集合 迭代器遍历Iterator 和集合增强For

迭代器遍历Iterator 标准写法: 增强For for(类型 名称 : 集合 ) 举例: 不仅可以集合也可以数组 底层仍然是iterator

设计模式—结构型模式之装饰器模式

设计模式—结构型模式之装饰器模式 适配器是连接两个类&#xff0c;可以增强一个类&#xff0c;装饰器是增强一个类。 向一个现有的对象添加新的功能&#xff0c;同时又不改变其结构。属于对象结构型模式。 创建了一个装饰类&#xff0c;用来包装原有的类&#xff0c;并在保…

ZZ308 物联网应用与服务赛题第E套

2023年全国职业院校技能大赛 中职组 物联网应用与服务 任 务 书 &#xff08;E卷&#xff09; 赛位号&#xff1a;______________ 竞赛须知 一、注意事项 1.检查硬件设备、电脑设备是否正常。检查竞赛所需的各项设备、软件和竞赛材料等&#xff1b; 2.竞赛任务中所使用的…

解决ERR: cURL error 77: Unable to initialize NSS: -8023

研发反映一个问题,上传文件时失败,日志内错误信息如下: ERR: cURL error 77: Unable to initialize NSS: -8023 (SEC_ERROR_PKCS11_DEVICE_ERROR) ... 这个功能使用了腾讯云的点播服务。因此立即联系了腾讯云客服。 搞了很久问题依旧。 反复测试,发现上传视频文件,错…

精解括号匹配问题与极致栈设计:揭开最大栈和最小栈的奥秘

目录 括号匹配问题最小栈最大栈 最大栈和最小栈是极致栈的两个重要变种。最大栈用于存储当前匹配的最大值&#xff0c;而最小栈用于存储当前匹配的最小值。 括号匹配问题 这个问题我们来看力扣20题的描述&#xff1a; 给定一个只包括 ‘(’&#xff0c;‘)’&#xff0c;‘{’…

大数据毕业设计选题推荐-营业厅营业效能监控平台-Hadoop-Spark-Hive

✨作者主页&#xff1a;IT毕设梦工厂✨ 个人简介&#xff1a;曾从事计算机专业培训教学&#xff0c;擅长Java、Python、微信小程序、Golang、安卓Android等项目实战。接项目定制开发、代码讲解、答辩教学、文档编写、降重等。 ☑文末获取源码☑ 精彩专栏推荐⬇⬇⬇ Java项目 Py…

RTC实时时钟——DS1302

DS1302目录 一、DS1302简介引脚定义与推荐电路 二、芯片手册1.操作寄存器的定义2.时序定义dc1302.cds1302.h 三、蓝桥杯实践 一、DS1302简介 RTC(Real Time Clock):实时时钟&#xff0c;是一种集成电路&#xff0c;通常称为时钟芯片。现在流行的串行时钟电路很多&#xff0c;如…

把wpf的窗体保存为png图片

昨晚在stack overflow刷问题时看到有这个问题&#xff0c;今天早上刚好来尝试学习一下 stack overflow的链接如下&#xff1a; c# - How to render a WPF UserControl to a bitmap without creating a window - Stack Overflow 测试步骤如下&#xff1a; 1 新建.net frame…

企业微信vs个人微信:对比对照一览表

继微信后&#xff0c;腾讯推出了企业微信。企业微信可以添个人微信为好友&#xff0c;有群聊和朋友圈&#xff0c;粗看起来与个人微信十分相似&#xff0c;那么它们有什么区别呢&#xff1f; 企业微信和个人微信的区别是什么&#xff0c;咱今天两张图来对比看看~

输电线路AR可视化巡检降低作业风险

随着现代工业的快速发展&#xff0c;各行业的一线技术工人要处理的问题越来越复杂&#xff0c;一些工作中棘手的问题迫切需要远端专家的协同处理。但远端专家赶来现场往往面临着专家差旅成本高、设备停机损失大、专业支持滞后、突发故障无法立即解决等痛点。传统的远程协助似乎…

数据结构与算法-(11)---有序表(OrderedList)

&#x1f308;个人主页: Aileen_0v0 &#x1f525;系列专栏:PYTHON学习系列专栏 &#x1f4ab;"没有罗马,那就自己创造罗马~" 目录 知识回顾及总结 有序表的引入 ​编辑 实现有序表 1.有序表-类的构造方法 2.有序表-search方法的实现 3.有序表-add方法的实现…