Flutter学习及实战

目录

1.Dart语法

2.Flutter工程目录

3.Widget简介

4.常用Widget介绍

5.实战

先看一下最终实现的效果:

代码

简介

Flutter是Google发布的一个用于创建跨平台、高性能移动应用的框架。它没有使用原生控件,而是实现了一个自绘引擎,使用自身的布局、绘制系统。开发Flutter应用使用的是Dart语言

一.Dart语法

Dart语言跟Java或者Kotlin的语法使用上差不多,下面就快速介绍一下Flutter开发中常用到的不同之处:

变量

Dart声明基本类型的变量可以使用var来接收任意类型的变量,一旦赋值类型就不能改变,也可以使用具体的类型来声明。

var i = 10; //一旦赋值类型就确定了,以后的类型就不能改变了。
int j = 10;

使用dynamic或Object来声明任意类型的对象类型。

函数

1.函数简写:

void main() => runApp(MyApp());
等价于
void main(){
  runApp(MyApp());
}

如果函数体只有一个表达式,可以使用=>来代替{}

2.可选命名参数
使用{param1, param2, …},用于指定命名参数。

//定义
void setData(int id,{String name,int count}){
    ......
}
//使用
setData(1);
setData(1,name: "慕涵盛华");
setData(1,name: "慕涵盛华",count: 2)

可选命名参数在Flutter中使用非常多。

修饰符

Dart里没有private/protected/public等权限修饰符,但是要实现private,只需要将需要修饰的字段或者方法,加上:"_"

//前缀带_的变量或者方法表示是类私有的变量或者方法。
int _count = 0; 
void _setCount(int count){
   _count  = count;
}
//使用
_setData(1);

异步支持

Dart类库有非常多的返回Future或者Stream对象的函数。 这些函数被称为异步函数:它们只会在设置好一些耗时操作之后返回,比如像 IO操作。而不是等到这个操作完成。

使用关键字async和await来异步编程。

//异步方法
void _getData() async {
    //等待返回结果
    final response = await http.get(URL_HOME_FIND_DETAIL_LIST);
    debugPrint("获取的数据为:${response.body}");
};

以上介绍这些语法都是在Flutter开发中经常使用的,其他语法的用法跟Java或者Kotlin基本一样了。

二.Flutter工程目录

创建类型

从上图中我们可以看出使用AndroidStudio创建Flutter项目是有4中类型,每一种类型的用处下面都给出了说明。这里说一下Flutter ApplicationFlutter Module,这两种工程目录下都包含Android和Ios工程目录,但是Flutter Module;类型下的这两个工程目录是隐藏式的。

目录结构

具体看一下pubspec.yaml文件的内容:

name: flutter_eyepetizer
description: A new Flutter module.
#应用或包的版本号
version: 1.0.0+1

environment:
  sdk: ">=2.1.0 <3.0.0"

#Flutter应用依赖
dependencies:
  flutter:
    sdk: flutter
  cupertino_icons: ^0.1.2
  dio: ^2.1.0
  http: ^0.12.0
  video_player: 0.10.0+2

#开发环境依赖的工具包
dev_dependencies:
  flutter_test:
    sdk: flutter

#flutter相关的配置选项
flutter:
  uses-material-design: true
  #配置本地图片
  assets:
    - images/ic_action_search.png
    - images/ic_tab_strip_icon_category.png
    - images/ic_tab_strip_icon_category_selected.png
    - images/ic_tab_strip_icon_feed.png
    - images/ic_tab_strip_icon_feed_selected.png
    - images/ic_tab_strip_icon_follow.png
    - images/ic_tab_strip_icon_follow_selected.png
    - images/ic_tab_strip_icon_profile.png
    - images/ic_tab_strip_icon_profile_selected.png
    - images/ic_home_public.png

  module:
    androidPackage: com.gfd.flutter_eyepetizer
    iosBundleIdentifier: com.gfd.flutterEyepetizer

Flutter的依赖使用的是pub仓库

三.Widget简介

Flutter中几乎所有的对象都是一个Widget,与原生开发中“控件”不同的是,Flutter中的widget的概念更广泛,它不仅可以表示UI元素,也可以表示一些功能性的组件如:用于手势检测的 GestureDetector widget、用于应用主题数据传递的Theme等等。而原生开发中的控件通常只是指UI元素。

Widget的功能是“描述一个UI元素的配置数据,它就是说,Widget其实并不是表示最终绘制在设备屏幕上的显示元素,而只是显示元素的一个配置数据。实际上,Flutter中真正代表屏幕上显示元素的类是Element,也就是说Widget只是描述Element的一个配置。并且一个Widget可以对应多个Element,这是因为同一个Widget对象可以被添加到UI树的不同部分,而真正渲染时,UI树的每一个Widget节点都会对应一个Element对象。

四.常用Widget介绍

1.布局类

线性布局Row和Column

Row相当于Android中LinearLayout设置为android:orientation="horizontal"
Column相当于Android中LinearLayout设置为android:orientation="vertical"

Row(
    //对于Row来说:主轴为水平方向,纵轴的垂直方向
    mainAxisAlignment: MainAxisAlignment.center,//主轴的对其方式
    crossAxisAlignment: CrossAxisAlignment.center,//纵轴的对其方式
    textDirection: TextDirection.ltr,//水平方向上的布局顺序:从left to right
    verticalDirection: VerticalDirection.down,//垂直方向上的对齐方式 :从上到下
    children: <Widget>[
        Text("子控件1"),
        Text("子控件2")
    ],
);

弹性布局

弹性布局允许子widget按照一定比例来分配父容器空间,类似Android中的android:layout_weightFlutter中的弹性布局主要通过FlexExpanded来配合实现。

//Row和Column都继承自Flex,参数基本相同,所以能使用Flex的地方一定可以使用Row或Column。
Flex(
    direction: Axis.horizontal,
    children: <Widget>[
    Expanded(
        flex: 1, //比例 相当于LinearLayout中的android:layout_weight = 1
        child: Container(height: 32.0, color: Colors.blue)),
    Expanded(
        flex: 2,

        child: Container(height: 32.0, color: Colors.green))
    ],
)

Spacer是对Expanded的一个包装,功能是占用指定比例的空间。

Spacer(
    flex: 1,
),
//Spacer内部实现
new Expanded(
  flex: flex,
  child: const SizedBox(
    height: 0.0,
    width: 0.0,
  ),
);

流式布局

在使用Row或者Colum时,如果子Widget超出屏幕范围,则会报溢出错误:

上图表示的是右边溢出部分报错。这是因为Row默认只有一行,如果超出屏幕不会折行。而流式布局在超出屏幕显示范围会自动换行。Flutter中通过WrapFlow来支持流式布局。类似Android中的FlexboxLayout.
Wrap:

Wrap(
    //主轴方向
    direction: Axis.horizontal,
    //主轴方向间距
    spacing: 8.0,
    //纵轴方向间距
    runSpacing: 4.0,
    //沿主轴方向靠左对齐
    alignment: WrapAlignment.start,
    children: <Widget>[
    Chip(
        label: Text("慕涵盛华"),
        avatar: CircleAvatar(
            backgroundColor: Colors.blue, child: Text("A"))),
    Chip(
        label: Text("慕涵盛华"),
        avatar: CircleAvatar(
            backgroundColor: Colors.blue, child: Text("B"))),
    Chip(
        label: Text("慕涵盛华"),
        avatar: CircleAvatar(
            backgroundColor: Colors.blue, child: Text("C"))),
    Chip(
        label: Text("慕涵盛华"),
        avatar: CircleAvatar(
            backgroundColor: Colors.blue, child: Text("D"))),
    Chip(
        label: Text("慕涵盛华"),
        avatar: CircleAvatar(
            backgroundColor: Colors.blue, child: Text("E"))),
    ],
);
效果图

Flow
一般很少会使用Flow,因为其过于复杂,需要自己实现子Widget的位置转换,Flow主要用于一些需要自定义布局策略或性能要求较高的场景。

层叠布局

层叠布局Android中的帧布局:FrameLayout类似。Flutter中使用StackPositioned来实现层叠布局Stack允许子Widget堆叠,而Positioned可以给子Widget定位(根据Stack的四个角)。

////通过ConstrainedBox来确保Stack占满屏幕
ConstrainedBox(
  constraints: BoxConstraints.expand(),
  child: Stack(
    //决定没有定位的子Widget如何去适应Stack的大小。
    // StackFit.loose:使用子widget的大小,StackFit.expand:扩伸到Stack的大小。
    fit: StackFit.loose,
    alignment:Alignment.center,//指定未定位widget的对齐方式
    children: <Widget>[
      Container(
        width: 32.0,
        height: 32.0,
        color: Colors.blue,
      ),
      Positioned(
        top: 10, 
        left: 10,
        child: Text("慕涵盛华"))
    ],
  ),
));

2.容器类

容器类Widget布局类Widget都作用于其子Widget,不同的是:

  • 布局类Widget一般都需要接收一个Widget数组(children),而容器类Widget一般只需要接受一个子Widget(child)
  • 布局类Widget是按照一定的排列方式来对其子Widget进行排列,而容器类Widget一般只是包装其子Widget,对其添加一些修饰或限制。

Padding

Padding可以给其子Widget添加内间距,类似Android布局属性中的android:layout_paddingXXX

Padding(
    //only:指定方向上的间距
    padding: EdgeInsets.only(left: 10,top: 10),
    //padding: EdgeInsets.all(10), 四个方向的间距
    //padding: EdgeInsets.fromLTRB(10, 10, 5, 8), 一次为 l t r b
    //padding: EdgeInsets.symmetric(horizontal: 8), 左右的距离
    child: Text("慕涵盛华"),
)

布局限制类容器:ConstrainedBox和SizedBox

ConstrainedBox:对子widget添加额外的约束

ConstrainedBox(
    //constraints:约束
    constraints: BoxConstraints(
        minWidth: double.infinity, //宽度尽可能大
        maxHeight: 50.0 //最大高度为50像素
        ),
    child: Container(
      color: Colors.blue,
    )
);

SizedBox:用于给子Widget指定固定的宽高

SizedBox(
    width: 50,
    height: 50,
    //child属性可以不设置,可以用来设置间距
    child: Container(
      color: Colors.blue,
    ),
);

装饰容器:DecoratedBox

可以在其子Widget绘制前后绘制一个装饰如背景、边框、渐变等。

DecoratedBox(
    //decoration:装饰
    decoration: BoxDecoration(
        borderRadius: BorderRadius.circular(4.0), //圆角
        gradient: LinearGradient(//背景渐变
            colors: [Colors.red, Colors.orange[700]]), 
        boxShadow: [ //阴影
          BoxShadow(
              color: Colors.black54,
              offset: Offset(2.0, 2.0),
              blurRadius: 4.0)
        ]),
    child: Padding(
        padding: EdgeInsets.symmetric(vertical: 18, horizontal: 80),
        child: Text("Login",style: TextStyle(color: Colors.white))
);
效果图

变换:Transform

Transform可以在其子Widget绘制时对其应用一个矩阵变换,实现平移、缩放、旋转等效果。

Container(
    color: Colors.black,
    child: new Transform(
    alignment: Alignment.topRight, //相对于坐标系原点的对齐方式
     // Transform.translate(offset: Offset(-20.0, -5.0) : 平移 
     //Transform.rotate 旋转
    transform: new Matrix4.skewY(0.3), //矩阵变换:沿Y轴倾斜0.3弧度
    child: new Container(
      padding: const EdgeInsets.all(8.0),
      color: Colors.deepOrange,
      child: const Text('慕涵盛华'),
    ),
    ),
);

Container

Container通过组合多种Widget来实现复杂强大的功能。

Container({
  this.alignment,
  this.padding, //容器内补白,属于decoration的装饰范围
  Color color, // 背景色
  Decoration decoration, // 背景装饰
  Decoration foregroundDecoration, //前景装饰
  double width,//容器的宽度
  double height, //容器的高度
  BoxConstraints constraints, //容器大小的限制条件
  this.margin,//容器外补白,不属于decoration的装饰范围
  this.transform, //变换
  this.child,
})

3.Material Design风格Widget

在大多数应用首页中顶部包含一个标题栏,底部包含一个底部导航等。Flutter Material库提供了一个Scaffold Widget,它是一个路由页的骨架,可以非常容易的拼装出一个完整的页面。类似下面这种页面:

Scaffold:

Scaffold({
    Key key,
    this.appBar, : AppBar 类似Android中的ActionBar
    this.body,
    this.floatingActionButton,
    this.floatingActionButtonLocation,
    this.floatingActionButtonAnimator,
    this.persistentFooterButtons,
    this.drawer,  ://Drawer:抽屉菜单
    this.endDrawer,
    this.bottomNavigationBar, :页面底部导航栏
    this.bottomSheet,
    this.backgroundColor,
    this.resizeToAvoidBottomPadding,
    this.resizeToAvoidBottomInset,
    this.primary = true,
    this.drawerDragStartBehavior = DragStartBehavior.start,
    this.extendBody = false,
  })

使用示例:

//TabBar必须有一个TabController
DefaultTabController(
  length: 4,//tab数量
  child: Scaffold(
    backgroundColor: Colors.white,
    appBar: AppBar(
      title: Text("图片列表"),
      elevation: 0.0,
      //左侧图标
      leading: IconButton(
          icon: Icon(Icons.menu), onPressed: () => debugPrint("按钮点击")),
      //右侧图标(可以指定多个)
      actions: <Widget>[
        IconButton(
            icon: Icon(Icons.search),
            tooltip: "搜索按钮",
            onPressed: () => debugPrint("搜索按钮被点击")),
        IconButton(
            icon: Icon(Icons.more_horiz),
            tooltip: "搜索按钮",
            onPressed: () => debugPrint("更多按钮被点击")),
      ],
      //顶部导航Tab 具体显示的内容与TabBarView对应
      bottom: TabBar(
          indicatorColor: Colors.white,
          indicatorSize: TabBarIndicatorSize.tab,
          indicatorWeight: 2.0,
          tabs: <Widget>[
            Tab(icon: Icon(Icons.local_florist)),
            Tab(icon: Icon(Icons.change_history)),
            Tab(icon: Icon(Icons.directions_bike)),
            Tab(icon: Icon(Icons.call_missed))
          ]),
    ),
    body: TabBarView(children: <Widget>[
      NavigationDemo(),
      SliverDemo(),
      GridViewDemo(),
      PageViewDemo2()
    ]),
    drawer: DrawerDemo(),
    bottomNavigationBar: BottomNavigationBarDemo(),
  ),
)

4.可滑动Widget

SingleChildScrollView

类似于Android中的ScrollView

SingleChildScrollView({
  this.scrollDirection = Axis.vertical, //滚动方向,默认是垂直方向
  this.reverse = false,  //是否反向滑动
  this.padding, 
  bool primary,  //是够使用默认的ScrollController
  this.physics, 
  this.controller, //ScrollController
  this.child,
})

ListView和GirdView

类似Android中的列表控件。

CustomScrollView,Sliver

类似Android中的CoordinatorLayout + AppBarLayout + CollapsingToolbarLayout

五.实战

效果图:

代码

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

推荐阅读更多精彩内容