前面两篇文章介绍了
Flutter win
环境的安装,以及利用listview
实现了简单布局。(这篇文章篇幅有点长,读完大概须要8.88
分钟)html
Flutter 入门实现 ListView 列表页面以及收藏页面git
Flutter 环境搭建以及填坑指南(Win10 系统且已有 Android 开发环境 github
这篇文章主要介绍如下内容:bash
2. widget 如何添加到 ListView 中?app
3. ListView 点击事件,单个 widget 点击事件框架
4. ( 重点 ) widget 如何垂直、水平摆放?less
5. 图片、Icon、Text widget 的简单使用dom
在这篇文章Flutter 入门实现 ListView 列表页面以及收藏页面中虽然实现了一个列表,可是怎么实现的还没仔细研究,如今就先从研究 ListView
的实现开始吧。ide
先看一下以前实现的效果:
RandomWordsState
类中的下面代码:
return new ListTile(
// 单词布局
title: new Text(
pair.asPascalCase,
style: _biggerFont,
),
// 喜欢当心心布局
trailing: new Icon(
alreadySaved ? Icons.favorite : Icons.favorite_border,
color: alreadySaved ? Colors.red : null,
),
// ListView item 的点击事件
onTap: () {
// 通知框架状态已经改变
setState(() {
if (alreadySaved) {
_saved.remove(pair);
} else {
_saved.add(pair);
}
});
},
);
复制代码
从上面的代码中能够看出实现这个列表主要就是操做了 ListTile
这个 Widget
,那么咱们看看这个ListTile
的构造方法都提供了上面功能吧!注释都写在代码中了~
/// Requires one of its ancestors to be a [Material] widget.
const ListTile({
Key key,
// 在列表的左边添加的 widget(如文末图中的左边图片)
this.leading,
// 标题
this.title,
// 副标题
this.subtitle,
// 在列表的右边添加的 widget(如文末图中的右边心形 icon)
this.trailing,
// 若是 isThreeLine 为 true, subtitle 则不能为 null,默认为 false
// 若是 副标题为空则列表平铺一行显示,若是 副标题 不为空,则副标题所占的布局是两行
// 若是 isThreeLine = true 则副标题能够显示三行。
this.isThreeLine = false,
// bool 类型,默认为false ,若是为 true 则 ListTile 在垂直方向是密集型摆放
//(具体效果待后面去实现,这里只是看了注释后的理解)
this.dense,
// 内容的边距
this.contentPadding,
// item 是否能点击
this.enabled = true,
// item 的点击事件
this.onTap,
// item 的长按事件
this.onLongPress,
// item 是否选中标记
this.selected = false,
})
复制代码
flutter
的开发语言是dart
,这个语言以前没学过,这里就先不深刻学习了,网上有比较好的这个博主 恋猫月亮 写的系列的文章写的很好。我这里就只写实现上面图片中展现的效果。
ListView 不只能够实现列表布局,还能够实现 Android 中的 ScrollView 的功能。关于 滚动的 widget 还有其余的实现,可参考**可滚动Widget简介**
实现列表功能,若是是符合ListTile
的样式,直接使用ListTile
实现很方便。(目前还没学到如何实现多布局,后面学习再写一篇)
实现ScrollView
功能,直接在build
方法的body
中添加咱们的 widget
。伪代码以下:
定义了一个
DetailScreen
详情页面,继承了StatelessWidget
,在build
方法中返回了页面的内容:appBar
是标题栏,body
是显示的内容
class DetailScreen extends StatelessWidget {
// Declare a field that holds the pair
final WordPair pair;
// In the constructor, require a pair
DetailScreen({Key key, @required this.pair}) : super(key: key);
@override
Widget build(BuildContext context) {
return new Scaffold(
appBar: new AppBar(
// 接收传递过来的单词作标题名字
title: new Text("${pair.asPascalCase}"),
),
// 使用 ListView 作滚动列表
body: new ListView(
children: [
// 显示网络图片
new Image.asset(
'images/wali.jpg',
width: 600.0,
height: 240.0,
fit: BoxFit.cover,
),
// 标题行(文末有实现效果)
titleSection,
// 按钮行(文末有实现效果)
buttonSection,
// 描述文本(文末有实现效果)
textSection,
],
),
);
}
}
复制代码
上面的伪代码中在构建一个有状态的widget
的时候会重写 build
方法,在该方法中的 body
返回咱们实现的 widget
便可。
若是ListView
使用ListTile
实现列表的话,直接使用ListTile
中的 onTap
实现列表点击效果。若是不是用ListTile
实现,那就使用单个widget
的点击事件吧。
widget
的若是有onTap
方法能够直接调用该方法便可实现,若是没有该方法,则须要使用GestureDetector
来实现点击效果,例如上图中的心形喜欢点击事件,伪代码实现以下:
关于
GestureDetector
可参考该文章 手势识别GestureDetector
trailing: new GestureDetector(
// 心形喜欢 icon
child: new Icon(
alreadySaved ? Icons.favorite : Icons.favorite_border,
color: alreadySaved ? Colors.red : null,
),
// 点击事件
onTap: (){
// 通知框架状态已经改变
setState(() {
if (alreadySaved) {
_saved.remove(pair);
} else {
_saved.add(pair);
}
});
},
),
复制代码
最多见的布局模式之一是垂直或水平排列widget。在
Flutter
可使用行Row
水平排列widget
,并使用列Column
垂直排列widget
。 同时,每一个孩子自己能够是一个Row
或一个Column
,依此类推。如下示例显示如何在行或列内嵌套行或列。
经过下面两张图片能够学到在实现布局时如何拆分: 参考连接 widget 布局
在控制行或列对齐其子项时使用mainAxisAlignment
( 主轴 ) 和crossAxisAlignment
( 横轴 ) 属性来。 对于行(Row)
来讲,**主轴是水平方向,横轴垂直方向。**对于列(Column)
来讲,主轴垂直方向,横轴水平方向。
center → const MainAxisAlignment :子 widget
在主轴方向上居中显示
end → const MainAxisAlignment :子 widget
在主轴方向上居右边显示,若是是水平方向,那么由 TextDirection 来决定end
是在左边(ltr)仍是在右边(rtl)。若是是竖直方向,那么由 VerticalDirection 来决定end
是在上边(up)仍是在下边(down)
start → const MainAxisAlignment:子 widget
在主轴方向上居左边显示,同理 end
。
spaceAround → const MainAxisAlignment:子widget
中的第一个和最后一个widget
距离边的距离是它与中间的距离的一半。(这里比较抽象,后期实现效果了会顺带讲解)
spaceBetween → const MainAxisAlignment:子widget
中的第一个和最后一个贴边,剩余的子widget
将中间的空间平分。
spaceEvenly → const MainAxisAlignment:子widget
将布局空间彻底平分
values → const List<
CrossAxisAlignment 属性相同的基本和上面介绍的同样,就不赘述了。
widget
填充横轴(CrossAxisAlignment),若是横轴上子widget
不少,会使横轴变得很挤。Expanded widget
Expanded widget,能够将widget的大小设置为适和行或列
Expanded
有个属性flex(弹性系数)
,默认状况下,每一个widget的弹性系数为1,也就是会铺满布局。
例如实现:三张图片水平铺满屏幕,便可使用Expanded
包裹Image
而后设置flex:1
,不设置也行,由于默认flex
就是1
伪代码以下:
body: new Center(
child: new Row(
crossAxisAlignment: CrossAxisAlignment.center,
children: [
new Expanded(
child: new Image.asset('images/pic1.jpg'),
),
new Expanded(
child: new Image.asset('images/pic2.jpg'),
),
new Expanded(
child: new Image.asset('images/pic3.jpg'),
),
复制代码
汇集 widgets
默认状况下,行或列沿着其主轴会尽量占用尽量多的空间,但若是要将孩子紧密汇集在一块儿,能够将
mainAxisSize
设置为MainAxisSize.min
。
例如实现:五个星形图标图标紧凑在一块儿。
class _MyHomePageState extends State<MyHomePage> {
@override
Widget build(BuildContext context) {
var packedRow = new Row(
mainAxisSize: MainAxisSize.min,
children: [
new Icon(Icons.star, color: Colors.green[500]),
new Icon(Icons.star, color: Colors.green[500]),
new Icon(Icons.star, color: Colors.green[500]),
new Icon(Icons.star, color: Colors.black),
new Icon(Icons.star, color: Colors.black),
],
);
// ...
}
复制代码
这些都是基础控件使用起来很简单的,可查看这个文档介绍:基础 Widget
在学习完上面的内容咱们能够实现出目前的样式了:
实现列表功能,以及一个可滚动的详情页面,顶部的标题是 listview 传递过去的单词。
文章篇幅过长了,这里就不介绍上图的实现过程了,源码地址以下 以为还凑合的给个star
,好让我继续坚持写下去~ 源码地址
看源码或者上文中不理解的能够联系我,我会知无不言的。
本文完。