Flutter 开发从 0 到 1(三)布局与 ListView

上周日出去玩了,所以没时间写文章。我司加班到 11 点,次日能够晚上班一个小时,加班到 12 点,能够晚上班两个小时,以此类推,为何说这个,对的,加班次日我没有多睡觉,而是起来抓紧时间写文章,好了,废话很少说,进入今天的主题。markdown

布局

说到 Android 布局,不是很难,会在对应的 xml 里布局,Flutter 里没有 xml,都在代码里写,给人感受就很难,看下咱们要实现的布局:网络

打开 PhotoShop,看下背景色 #ededed,日期字体颜色 #a6a6a6,标题字体颜色 #1b1b1b,摘要字体颜色 #808080。app

公众号这是列表,我先将 item 搞定。看这布局都是线性布局,这要在以前,分分钟搞定,但 Flutter……好吧,Flutter 布局可没那么简单,我花了好几个小时才作好,期间遇到了很多困难。框架

Flutter 也有横向 Row 布局和竖向 Column 布局,我本想分 1 和 2 两个部分,最外层竖向 Column 包含 1 和 2,2 自己是 Column,包含一个 image 和两个 text,直接使用 Column 能够完成,当我须要设置 2 里面白色的背景色,发现 Column 根本没有背景色属性,因而把 2 最外层改形成 Container。ide

使用 Container 没有问题,布局也很快实现了,接下来是实现四角的圆角效果。oop

Container 有 decoration,能够实现圆角,我遇到了两个问题:布局

一、当 shape: BoxShape.circle 时不能设置 borderRadius ,会异常 异常信息:'shape != BoxShape.circle ||borderRadius == null': is not true.学习

二、使用 BoxDecoration
Container 不能使用 color,会报错: Cannot provide both a color and a decoration To provide both, use "decoration: BoxDecoration(color: color)".字体

觉得这样就实现了圆角,不,不会那么顺利的,发现图片根本没有圆角效果,这和以前 Java 实现方式不同,最外层都实现了圆角,对里面的布局(图片)竟然没有生效,最后只好把布局实现以下:ui

分红 一、二、3 部分,3 仍是 Container,2 和 3 圆角效果只对上下部分分别实现,完整代码以下:

blogItem() {
    var date = new Padding(
        padding: const EdgeInsets.only(
          top: 20.0,
          left: 10.0,
          right: 10.0,
        ),
        child: new Text(
          '2020年6月28日 22:49',
          textAlign: TextAlign.center,
          style: TextStyle(color: ColorCommon.dateColor, fontSize: 18),
        ));

    var cover = new Padding(
        padding: const EdgeInsets.only(
          top: 10.0,
          left: 10.0,
          right: 10.0,
        ),
        child: new ClipRRect(
            borderRadius: BorderRadius.only(
                topLeft: Radius.circular(10.0),
                topRight: Radius.circular(10.0)),
            child: new Image.network(
              'http://pic1.win4000.com/wallpaper/2020-04-21/5e9e676001e20.jpg',
            )));

    var title = new Text(
      'APP 开发从 0 到 1(一)需求与准备',
      style: TextStyle(color: ColorCommon.titleColor, fontSize: 22),
    );

    var summary = new Padding(
        padding: const EdgeInsets.only(
          top: 5.0,
        ),
        child: new Text('一我的作一个项目,你也能够。',
            textAlign: TextAlign.left,
            style: TextStyle(color: ColorCommon.summaryColor, fontSize: 18)));

    var titleSummary = new Container(
      padding: const EdgeInsets.all(10.0),
      alignment: Alignment.topLeft,
      decoration: new BoxDecoration(
        color: Colors.white,
        borderRadius: BorderRadius.only(
            bottomLeft: Radius.circular(10.0),
            bottomRight: Radius.circular(10.0)),
        shape: BoxShape.rectangle,
      ),
      margin: const EdgeInsets.only(left: 10, right: 10.0),
      child: Column(
        crossAxisAlignment: CrossAxisAlignment.start,
        mainAxisSize: MainAxisSize.min,
        children: <Widget>[title, summary],
      ),
    );

    var blogItem = new Column(
      children: <Widget>[date, cover, titleSummary, date, cover, titleSummary],
    );

    return blogItem;
  }
复制代码

实际效果以下:

好了,关于 Flutter 布局就讲到这里,我只是针对这个项目所须要的去实现,Flutter 布局的东西还有不少,其属性让人眼花撩乱,能够去官网一个个去仔细学习和实践。

ListView

blogItem 写好了,完成了大头,接下来就要填充真实的数据,在《APP 开发从 0 到 1(二)框架与网络》,有说到网络请求,其中有提到 setstate(){} 方法,调用这个方法会回调 build 方法,这样咱们能够在 build 方法加个判断,先加载 Progress,待网络数据请求完,再显示 ListView,代码以下:

@override
  Widget build(BuildContext context) {
    var content;
    if (blogList.isEmpty) {
      content = new Center(
        // 可选参数 child:
        child: new CircularProgressIndicator(),
      );
    } else {
      content = new ListView(children: blogItem());
    }

    return Scaffold(
      backgroundColor: ColorCommon.backgroundColor,
      appBar: AppBar(
        title: Text('AndBlog'),
      ),
      body: content,
      floatingActionButton: FloatingActionButton(
        tooltip: 'Increment',
        child: Icon(Icons.add),
      ), // This trailing comma makes auto-formatting nicer for build methods.
    );
  }

blogItem() {
    List<Widget> widgets = [];
    for (int i = 0; i < blogList.length; i++) {
      Blog blog = blogList[i];
      var date = new Padding(
          padding: const EdgeInsets.only(
            top: 20.0,
            left: 10.0,
            right: 10.0,
          ),
          child: new Text(
            // 填充真实数据
            blog.date,
            textAlign: TextAlign.center,
            style: TextStyle(color: ColorCommon.dateColor, fontSize: 18),
          ));

      //……

      var blogItem = new Column(
        children: <Widget>[
          date,
          cover,
          titleSummary,
        ],
      );
      widgets.add(blogItem);
    }
    return widgets;
  }
复制代码

这样 ListView 也完成了,接下来须要完成的是 ListView 加载更多。

相关文章
相关标签/搜索