List/Grid View

人间烟火/张佳伟版 Lv2

网格可滑动表格

摘要 (GridView)

在网格中使用 widget

当列的内容超出渲染容器的时候,它会自动支持滚动。
GridView.count 允许你制定列的数量
GridView.extent 允许你制定单元格的最大宽度
使用 GridView.count 创建一个网格,它在竖屏模式下有两行,在横屏模式下有三行。
可以通过为每个 GridTile 设置 footer 属性来创建标题。
https://github.com/cfug/flutter.cn/tree/main/examples/layout/gallery/lib/grid_list_demo.dart

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
Widget _buildGrid() => GridView.extent(
maxCrossAxisExtent: 150,
padding: const EdgeInsets.all(4),
mainAxisSpacing: 4,
crossAxisSpacing: 4,
children: _buildGridTileList(30));

// The images are saved with names pic0.jpg, pic1.jpg...pic29.jpg.
// The List.generate() constructor allows an easy way to create
// a list when objects have a predictable naming pattern.
List<Container> _buildGridTileList(int count) => List.generate(
count, (i) => Container(child: Image.asset('images/pic$i.jpg')));
···

···
GridView.count(
// Create a grid with 2 columns. If you change the scrollDirection to
// horizontal, this produces 2 rows.
crossAxisCount: 2,
// Generate 100 widgets that display their index in the List.
children: List.generate(100, (index) {
return Center(
child: Text(
'Item $index',
style: Theme.of(context).textTheme.headlineSmall,
),
);
}),
),
ListView

ListView,一个和列很相似的 widget,当内容长于自己的渲染盒时,就会自动支持滚动。
标准的 ListView 构造函数适用于短列表,对于具有大量列表项的长列表,需要用 ListView.builder 构造函数来创建。
与标准的 ListView 构造函数需要一次性创建所有列表项不同的是, ListView.builder 构造函数只在列表项从屏幕外滑入屏幕时才去创建列表项。

摘要 (ListView)
一个用来组织盒子中列表的专用 Column
可以水平或者垂直布局
当监测到空间不足时,会提供滚动
比 Column 的配置少,使用更容易,并且支持滚动
通过指定 scrollDirection 的值为水平方向,来覆盖默认的竖直方向

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
Widget _buildList() {
return ListView(
children: [
_tile('CineArts at the Empire', '85 W Portal Ave', Icons.theaters),
_tile('The Castro Theater', '429 Castro St', Icons.theaters),
_tile('Alamo Drafthouse Cinema', '2550 Mission St', Icons.theaters),
_tile('Roxie Theater', '3117 16th St', Icons.theaters),
_tile('United Artists Stonestown Twin', '501 Buckingham Way',
Icons.theaters),
_tile('AMC Metreon 16', '135 4th St #3000', Icons.theaters),
const Divider(),
_tile('K\'s Kitchen', '757 Monterey Blvd', Icons.restaurant),
_tile('Emmy\'s Restaurant', '1923 Ocean Ave', Icons.restaurant),
_tile('Chaiya Thai Restaurant', '272 Claremont Blvd', Icons.restaurant),
_tile('La Ciccia', '291 30th St', Icons.restaurant),
],
);
}

ListTile _tile(String title, String subtitle, IconData icon) {
return ListTile(
title: Text(title,
style: const TextStyle(
fontWeight: FontWeight.w500,
fontSize: 20,
)),
subtitle: Text(subtitle),
leading: Icon(
icon,
color: Colors.blue[500],
),
);
}

下面举例创建一个带数据的列表

1. 创建数据源

首先,需要获取列表的数据源。例如,数据源可以是消息集、搜索结果集或者商店商品集。大部分情况下,这些数据来自于网络请求或者数据库获取。

在下面的例子,使用 List.generate 构造函数生成包含 10,000 个字符串的集合。

1
List<String>.generate(10000, (i) => 'Item $i'),
2. 将数据源渲染成组件

为了将字符串集合展示出来,需要通过 ListView.builder 把集合中的每个字符串都渲染成组件。

在下面的例子中,将会把每个字符串用单行列表项显示在列表中。

1
2
3
4
5
6
7
8
9
10
11
ListView.builder(
itemCount: items.length,
prototypeItem: ListTile(
title: Text(items.first),
),
itemBuilder: (context, index) {
return ListTile(
title: Text(items[index]),
);
},
)
完整代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
import 'package:flutter/material.dart';

void main() {
runApp(
MyApp(
items: List<String>.generate(10000, (i) => 'Item $i'),
),
);
}

class MyApp extends StatelessWidget {
final List<String> items;

const MyApp({super.key, required this.items});

@override
Widget build(BuildContext context) {
const title = 'Long List';

return MaterialApp(
title: title,
home: Scaffold(
appBar: AppBar(
title: const Text(title),
),
body: ListView.builder(
itemCount: items.length,
prototypeItem: ListTile(
title: Text(items.first),
),
itemBuilder: (context, index) {
return ListTile(
title: Text(items[index]),
);
},
),
),
);
}
}

创建展示不同类型内容的列表

比方说,我们可能在开发一个列表,它显示一个标题,后跟一些与标题相关的项目,然后是另一个标题,依此类推。

创建一个拥有不同类型项目的数据源
将数据源的数据转换成列表 widget
  1. 创建一个具有不同类型项目的数据源

为了表示 List 中不同类型的项,我们需要为每一个类型的项目定义一个类。

在这个例子中,我们将制作一个展示了标题,后面有五条消息的应用。因此,我们将创建三个类:ListItem、HeadingItem 和 MessageItem。

你可以通过以下步骤,用 Flutter 创建这样的结构:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
/// The base class for the different types of items the list can contain.
abstract class ListItem {
/// The title line to show in a list item.
Widget buildTitle(BuildContext context);

/// The subtitle line, if any, to show in a list item.
Widget buildSubtitle(BuildContext context);
}

/// A ListItem that contains data to display a heading.
class HeadingItem implements ListItem {
final String heading;

HeadingItem(this.heading);

@override
Widget buildTitle(BuildContext context) {
return Text(
heading,
style: Theme.of(context).textTheme.headlineSmall,
);
}

@override
Widget buildSubtitle(BuildContext context) => const SizedBox.shrink();
}

/// A ListItem that contains data to display a message.
class MessageItem implements ListItem {
final String sender;
final String body;

MessageItem(this.sender, this.body);

@override
Widget buildTitle(BuildContext context) => Text(sender);

@override
Widget buildSubtitle(BuildContext context) => Text(body);
}
1
2
3
4
5
6
final items = List<ListItem>.generate(
1000,
(i) => i % 6 == 0
? HeadingItem('Heading $i')
: MessageItem('Sender $i', 'Message body $i'),
);
  1. 将数据源的数据转换成列表 widget

为了把每一个项目转换成 widget,我们将采用 ListView.builder() 构造方法。
通常,我们需要提供一个 builder 函数来确定我们正在处理的项目类型,并返回该类型项目的相应 widget。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
ListView.builder(
// Let the ListView know how many items it needs to build.
itemCount: items.length,
// Provide a builder function. This is where the magic happens.
// Convert each item into a widget based on the type of item it is.
itemBuilder: (context, index) {
final item = items[index];

return ListTile(
title: item.buildTitle(context),
subtitle: item.buildSubtitle(context),
);
},
)
  • Title: List/Grid View
  • Author: 人间烟火/张佳伟版
  • Created at: 2024-04-21 21:56:03
  • Updated at: 2024-04-21 21:28:37
  • Link: https://945912035.github.io/2024/04/21/2024-4-21/
  • License: This work is licensed under CC BY-NC-SA 4.0.
 Comments
On this page
List/Grid View