flutter一文通3

1. flutter编程范式

和vue很像,flutter也是采用声明式编程. 有状态管理的概念

2. StatefulWidget

  • StatefulWidget是有 状态变化的widget
  • 状态交给State类来管理
  • 更改完状态后必须用setState()提交

3. StatelessWidget

  • StatelessWidget是无状态变化的widget
  • StatelessWidget里的变量应该都用final修饰
  • 必须实现build方法

二. 布局

  • colum和row是会尽量占更大空间的
  • 如果想让colum和row内容居中,应该mainAxisAlignment: MainAxisAlignment.center,

三. 生命周期

image.png
  • 创建完成后会初始化
  • 当有setState时,会把状态管理器设置为脏,下一次循环时,会触发build重构
  • 当父组件进行重绘时,子组件会调用didUpdateConfig来把状态管理器设置为脏,然后重新build
import 'package:flutter/material.dart';

void main() => runApp(MyApp());

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Flutter Demo',
      theme: ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: Scaffold(
        appBar: AppBar(
          title: Text("data"),
        ),
        body: MyWidget(),
      ),
    );
  }
}

class MyWidget extends StatefulWidget {
  MyWidget() {
    print("MyWidget的构造函数被调用");
  }
  @override
  _MyWidgetState createState() {
    print("MyWidget的createState函数被调用");
    return _MyWidgetState();
  }
}

class _MyWidgetState extends State<MyWidget> {
  int counter = 0;
  _MyWidgetState() {
    print("_MyWidgetState的构造函数被调用");
  }
  void initState() {
    super.initState();
    print("_MyWidgetState的initState函数被调用");
  }

  @override
  void didChangeDependencies() {
    print("_MyWidgetState的didChangeDependencies函数被调用");
    super.didChangeDependencies();
  }

  @override
  void didUpdateWidget(MyWidget oldWidget) {
    print("_MyWidgetState的didUpdateWidget函数被调用");
    super.didUpdateWidget(oldWidget);
  }

  @override
  Widget build(BuildContext context) {
    print("_MyWidgetState的build函数被调用");
    return Center(
      child: Column(
        children: <Widget>[
          RaisedButton(
              child: Text("+"),
              onPressed: () {
                setState(() {
                  counter++;
                });
              }),
          Text("counter:$counter")
        ],
      ),
    );
  }
}

执行顺序如下:
程序开始运行时

点击+号时

四. 网络请求DIO

1. 安装DIO

打开pubspec.yaml, 在dependencies里面添加

dio: ^3.0.3 

保存的瞬间,vscode会自动运行命令:

flutter pub get
vscode中会自动执行这一步

2. 封装网络请求

建立文件 lib/network/request.dart

import 'package:dio/dio.dart';
import './http_config.dart';

class HttpRequest {
  static BaseOptions baseOptions = //基础选项
      BaseOptions(baseUrl: BASE_URL, connectTimeout: TIMEOUT);
  //1. 创建dio实例
  static final dio = Dio(baseOptions); //创建实例

  //类的request函数异步返回一个
  static Future request(String url,
      {String method = 'get', Map<String, dynamic> params}) async {
//用static 修饰,使之成为类方法,可以直接类调用

    //2. 请求
    Options options = Options(method: method); //本次请求的选项
    try {
      final result =
          await dio.request(url, queryParameters: params, options: options);
      return result;
    } catch (e) {
      throw e;
    }
  }
}

3. 如何调用?

调用网络请求一般是在statefulWidget 的 initSate中调用

五. 页面路由动画效果

如果想要各种酷炫的页面切换动画,需要自己重写一个PageRouteBuilder
整个过程如下:
新建一个文件myRouter.dart

import 'package:flutter/material.dart';

class MyRouter extends PageRouteBuilder {
  final Widget widget;

  MyRouter(this.widget)
      : super(
            transitionDuration: Duration(milliseconds: 500), //单位是可以换的
            pageBuilder: (BuildContext context, Animation<double> animation1,
                Animation<double> animation2) {
              return widget;
            },
            transitionsBuilder: (BuildContext context,
                Animation<double> animation1,
                Animation<double> animation2,
                Widget child) {
              //--------------左右移动效果--------------------
              return SlideTransition(
                position: Tween<Offset>(
                        begin: Offset(-1.0, 0.0), end: Offset(0.0, 0.0))
                    .animate(CurvedAnimation(
                        parent: animation1, curve: Curves.fastOutSlowIn)),
                child: child,
              );

              //--------------缩放效果--------------------
              // return ScaleTransition(
              //   scale: Tween(begin: 0.0, end: 1.0).animate(CurvedAnimation(
              //       parent: animation1, curve: Curves.fastOutSlowIn)),
              //   child: child,
              // );

              //--------------渐入渐出效果--------------------
              // return FadeTransition(
              //   opacity: Tween(begin: 0.0, end: 1.0).animate(CurvedAnimation(
              //       parent: animation1, curve: Curves.fastOutSlowIn)),
              //   child: child,
              // );

              //--------------旋转加缩放效果--------------------
              // return RotationTransition(
              //   turns: Tween(begin: 0.0, end: 1.0).animate(CurvedAnimation(
              //       parent: animation1, curve: Curves.fastOutSlowIn)),
              //   child: ScaleTransition(
              //     scale: Tween(begin: 0.0, end: 1.0).animate(CurvedAnimation(
              //         parent: animation1, curve: Curves.fastOutSlowIn)),
              //     child: child,
              //   ),
              // );
            });
}

引用时:

onPressed: () {
              Navigator.of(context).push(MyRouter(Page2nd()));
            }),

六. TabBar //tab切换条

tabbar用法还挺特殊

import 'package:flutter/material.dart';

void main() => runApp(MyApp());

class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
        title: 'Flutter Demo',
        theme: ThemeData(
          primarySwatch: Colors.blue,
        ),
        home: KeepAliveDemo());
  }
}

class KeepAliveDemo extends StatefulWidget {
  _KeepAliveDemoState createState() => _KeepAliveDemoState();
}

/*
with是dart的关键字,意思是混入的意思,
就是说可以将一个或者多个类的功能添加到自己的类无需继承这些类,
避免多重继承导致的问题。
SingleTickerProviderStateMixin 主要是我们初始化TabController时,
需要用到vsync ,垂直属性,然后传递this
*/
class _KeepAliveDemoState extends State<KeepAliveDemo>
    with SingleTickerProviderStateMixin {
  TabController _controller;
  List tabs = <Tab>[
    Tab(
      text: "tab1",
    ),
    Tab(
      text: "tab2",
    ),
    Tab(
      text: "tab3",
    )
  ];
  @override
  void initState() {
    //重写初始化阶段, 给_controller初始化
    _controller =
        TabController(length: tabs.length, vsync: this //动画效果的异步处理,默认格式,背下来即可
            );
    super.initState();
  }

  @override
  void dispose() {
    //重写销毁阶段, 给_controller销毁
    _controller.dispose();
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
        appBar: AppBar(
          title: Text('Keep Alive Demo'),
          bottom: TabBar(
            tabs: tabs,
            controller: _controller,
          ),
        ),
        body: TabBarView(controller: _controller, children: [
          Text("111"),
          Text("222"),
          Text("333"),
        ]));
  }
}

七. 保持状态

这里我们做这样一个效果:tabbar切换后, 被切换掉的页面还保持原先的状态
这个案例更综合一些,耐住性子结合上一节来看。关键处在于 with混合一个AutomaticKeepAliveClientMixin类 重写一个get方法

import 'package:flutter/material.dart';

class MyHomePage extends StatefulWidget {
  @override
  _MyHomePageState createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage>
    with AutomaticKeepAliveClientMixin {
  int _counter = 0;

  @override
  bool get wantKeepAlive => true;

  void _incrementCounter() {
    setState(() {
      _counter++;
    });
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.center,
          children: <Widget>[
            Text("点一下加1"),
            Text(
              '$_counter',
              style: Theme.of(context).textTheme.display1,
            )
          ],
        ),
      ),
      floatingActionButton: FloatingActionButton(
          child: Icon(Icons.add), tooltip: "+++", onPressed: _incrementCounter),
    );
  }
}

八. 折叠瓦片

import 'package:flutter/material.dart';

class ExpandDemo extends StatefulWidget {
  @override
  _ExpandDemoState createState() => _ExpandDemoState();
}

class _ExpandDemoState extends State<ExpandDemo> {
  bool _expandState;
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text("折叠瓦片")),
      body: ExpansionTile(
        leading: Icon(Icons.add_location),
        title: Text("标题$_expandState"),
        subtitle: Text("副标题"),
        trailing: Icon(Icons.arrow_downward), //可以不写,有默认的
        backgroundColor: Colors.white12,
        initiallyExpanded: true, //默认打开
        onExpansionChanged: (val) {
          setState(() {
            _expandState = val;
          });
        },
        children: <Widget>[
          Text("展开内容第一行"),
          Image.network("https://s2.ax1x.com/2019/04/15/AXx8w8.jpg"),
        ],
      ),
    );
  }
}

九. 单子组件滚动视图 SingleChildScrollView

十. 可展开列表

主要使用的组件:ExpansionPanelList, ExpansionPanel

import 'package:flutter/material.dart';

class ExpandDemo extends StatefulWidget {
  @override
  _ExpandDemoState createState() => _ExpandDemoState();
}

class _ExpandDemoState extends State<ExpandDemo> {
  List<int> mList;
  List<ExpandStateBean> expandStateList;

  // 初始化时清空一下列表,再生成几个栗子
  _ExpandDemoState() {
    mList = []; //为了生成栗子准备的列表
    expandStateList = []; //存放每个项的展开状态的数组
    for (var i = 0; i < 20; i++) {
      mList.add(i);
      expandStateList.add(ExpandStateBean(i, false));
    }
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(title: Text("可展开列表")),
      body: SingleChildScrollView(
        //ExpansionPanelList必须放在滚动组件里
        child: ExpansionPanelList(
            children: mList.map((value) {
              //子元素必须是ExpansionPanel类,直接用map生成
              return ExpansionPanel(
                  canTapOnHeader: true, //点击头部可以展开否
                  isExpanded: expandStateList[value].isOpen, //是否展开
                  headerBuilder: (context, bol) {
                    //头部的组件
                    return ListTile(
                      title: Text("this is No.$value"),
                    );
                  },
                  body: ListTile(
                    //身体的组件
                    title: Text("this is No.$value's content!"),
                  ));
            }).toList(),
            expansionCallback: (num, val) {
              //被点击时的回调
              setState(() {
                expandStateList[num].isOpen = !val;
              });
            }),
      ),
    );
  }
}

//创建一个类 里面存储索引和开关状态之间的关系.
class ExpandStateBean {
  var isOpen;
  var index;
  ExpandStateBean(this.index, this.isOpen);
}

十一. 贝塞尔曲线绘制

通过绘制贝塞尔曲线,我们可以做一些异形的效果.使得我们的APP更美. 关键问题在于绘制曲线路径.

import 'package:flutter/material.dart';

class MyCliper extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text("贝塞尔曲线"),
      ),
      body: Column(
        children: <Widget>[
          ClipPath(
            clipper: MyBottomCliper(),
            child: Container(
              color: Colors.amberAccent,
              height: 200.0,
            ),
          )
        ],
      ),
    );
  }
}

class MyBottomCliper extends CustomClipper<Path> {
  //MyBottomCliper是对CustomClipper的继承,
  @override
  Path getClip(Size size) {
    //必须重写方法getClip, 返回一个path类型的路径
    var path = Path();
    path.lineTo(0, 0); //就像画图一样,从第一个点开始绘制并填充
    path.lineTo(0, size.height - 50); // 第二个点

    var firstControlPoint =
        Offset(size.width / 2, size.height); //为了绘制贝塞尔曲线,先声明两个关键点,此为控制点
    var firstEndPoint = Offset(size.width, size.height - 50); //此为结束点

    path.quadraticBezierTo(
        firstControlPoint.dx,
        firstControlPoint.dy, //绘制贝塞尔曲线
        firstEndPoint.dx,
        firstEndPoint.dy);
    path.lineTo(size.width, 0); //回至顶部
    return path;
  }

  @override
  bool shouldReclip(CustomClipper<Path> oldClipper) {
    //必须重写shouldReclip, 返回false,一般不改动
    return false;
  }
}

关于控制点 和 结束点 的意义, 有一点PS基础的同学看这张图就明白了

用两个贝塞尔曲线绘制波浪线, 理解了控制点和结束点的意义,就可以自由发挥了.

    var firstControlPoint =
        Offset(size.width / 4, size.height); //为了绘制贝塞尔曲线,先声明第一组两个关键点
    var firstEndPoint = Offset(size.width / 2, size.height - 50);

    path.quadraticBezierTo(
        firstControlPoint.dx,
        firstControlPoint.dy, //绘制第一组贝塞尔曲线
        firstEndPoint.dx,
        firstEndPoint.dy);

    var secondControlPoint =
        Offset(size.width / 4 * 3, size.height - 100); //为了绘制贝塞尔曲线,再声明第二组两个关键点
    var secondEndPoint = Offset(size.width, size.height - 50);

    path.quadraticBezierTo(
        secondControlPoint.dx,
        secondControlPoint.dy, //绘制第二组贝塞尔曲线
        secondEndPoint.dx,
        secondEndPoint.dy);
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容

  • 转自 Q吹个大气球Q 本文主要介绍了Flutter布局相关的内容,对相关知识点进行了梳理,并从实际例子触发,进一步...
    chilim阅读 1,916评论 0 17
  • 本文主要介绍了Flutter布局相关的内容,对相关知识点进行了梳理,并从实际例子触发,进一步讲解该如何去进行布局。...
    Q吹个大气球Q阅读 9,714评论 6 51
  • 1. 简介 在介绍Flutter布局之前,我们得先了解Flutter中的一些布局相关的特性。 1.1 边界约束(b...
    24K纯城阅读 523评论 0 0
  • 已经从印度回来有几个星期了,之前差不多已经把这一路的经历都记好了,今晚无事,写个总结帖,把这些文章合在一处,方便发...
    bmy阅读 682评论 2 1
  • よりほうが 与其说比较更不如理解为 递进关系 このパソコンより、あのパソコンのほうが性能がいい比起这个电脑 那个电...
    萌囧囧阅读 1,636评论 0 0