Flutter常用Widget详解(一)

发表于 3年以前  | 总阅读数:1636 次

前言


前几篇文章大家已经对Flutter环境搭建、所用开发语言和一些绘图原理有了一个初步了解,本篇是一个实战篇,进行应用开发时首先要进行UI界面的开发,Flutter所展示的UI界面都是由一些Widget组合而成,Widget可以理解为我们原生开发中的UI控件和UI布局控件。例如iOS中的UILabel、UIButton、UITableView,安卓中的Button、TextView、ListView等。下面带大家一起来看一下常用的Widget使用方法。


常用Widget介绍

文本控件

文本控件是日常开发中最常用的控件,Flutter提供了两个文本控件供我们使用,下面针对常用的几个属性进行介绍。

Text
Text(
  'Flutter allows you to build beautiful native apps on iOS and Android from a single codebase.',
  textAlign: TextAlign.center, // 文本对齐方式
),
Text(
  'Flutter allows you to build beautiful native apps on iOS and Android from a single codebase.',
  softWrap: false, // true时会自动换行处理;false时会判定为有无限的水平空间,不会换行
),
Text(
  'Flutter allows you to build beautiful native apps on iOS and Android from a single codebase.',
  maxLines: 1, //最大行数
  style: TextStyle(
    color: Colors.blue,
  ),
),
Text(
  'Flutter allows you to build beautiful native apps on iOS and Android from a single codebase.',
  overflow: TextOverflow.ellipsis, //溢出处理,这里ellipsis将多余的内容设置为...
),
Text(
  'Flutter allows you to build beautiful native apps on iOS and Android from a single codebase.',
  style: TextStyle( // 文本样式
    color: Colors.red, // 文本颜色
    fontSize: 14, // 字体大小
    fontWeight: FontWeight.w600, // 字体粗细程度
    fontStyle: FontStyle.normal, // 字体样式
    letterSpacing: 2, // 字母或字间距
    wordSpacing: 5, // 单词间距
    height: 2, // 行高,值为字体大小的倍数
    shadows: [Shadow(color: Colors.red, offset: Offset(1, 1), blurRadius: 5)], // 阴影
  ),
),
Text(
  'Flutter allows you to build beautiful native apps on iOS and Android from a single codebase.',
  style: TextStyle(
    decoration: TextDecoration.underline, // 文本装饰,此处设置下划线
    decorationColor: Colors.blue, // 文本装饰颜色
    decorationStyle: TextDecorationStyle.dotted, // 文本装饰样式
  ),
),

显示效果如下图:

RichText

富文本控件,可以对一段连续的文本设置不用的样式,实战中比较常见。

RichText(
  text: TextSpan(
    text: 'Flutter',
    style: TextStyle(color: Colors.black),
    children: <TextSpan>[
      TextSpan(
        text: ' allows you',
        style: TextStyle(
          color: Colors.green,
          decoration: TextDecoration.underline,
          decorationStyle: TextDecorationStyle.solid,
        ),
      ),
      TextSpan(
        text: ' to build beautiful native apps',
        style: TextStyle(
          fontSize: 18,
        )
      ),
      TextSpan(
        text: ' on iOS and Android',
        style: TextStyle(
          fontWeight: FontWeight.bold,
        )
      ),
      TextSpan(
          text: ' from a single codebase.',
          style: TextStyle(
            shadows: [Shadow(color: Colors.black38, offset: Offset(3, 3))],
          )
      ),
    ],
  ),
)

显示效果如下图:

图片控件

Image
Image.asset(
  'images/flutter_logo.png', //图片资源路径
),
Image.asset(
  'images/flutter_logo.png',
  width: 100, //图片宽度
  height: 100, //图片高度
  fit: BoxFit.fill, //适配显示方式,fill表示宽高填充满
),
Image.asset(
  'images/flutter_logo.png',
  color: Colors.red, //混合的颜色,和colorBlendMode一起使用
  colorBlendMode: BlendMode.overlay, //颜色和图片混合模式,功能较强大,其它模式参见官方文档或源码
),
Image.asset(
  'images/flutter_logo.png',
  width: 200,
  height: 200,
  repeat: ImageRepeat.repeat, //在宽高内重复平铺图片,直到铺满
)

显示效果如下图:

除以上使用的Image.asset()构造函数外,Image还有Image.file()、Image.network()和Image.memory()等命名构造函数。

  • Image.file

可通过路径加载SD卡中存储的图片,安卓使用此方法时需要申请android.permission.READ_EXTERNAL_STORAGE权限。

  • Image.network 可通过url加载网络图片。
  • Image.memory 可通过Uint8List对象加载内存中的图片。
Icon
Icon(
  Icons.adb,
),
Icon(
  Icons.adb,
  size: 50, //icon大小
),
Icon(
  Icons.adb,
  color: Colors.red, //icon颜色
)

显示效果如下图:

按钮控件

按钮控件在Flutter中有两种风格的button,安卓Material Design风格和iOS Cupertino风格。

RaisedButton
RaisedButton(
  onPressed: null, // onPressed为null视为不可点击
  disabledTextColor: Colors.grey, // 不可点击的文本颜色
  disabledColor: Colors.blue, // 不可点击的按钮颜色
  disabledElevation: 5, // 不可点击时图层高度
  child: Text('Disabled Button'),
),
RaisedButton(
  onPressed: () { // onPressed不为null视为可点击
    print('You click the button');
  },
  textColor: Colors.white, // 文本颜色
  color: Colors.blueAccent, // 按钮颜色
  highlightColor: Colors.lightBlue, //点击按钮后高亮的颜色
  elevation: 5, // 按钮图层高度
  highlightElevation: 8, // 点击按钮高亮后的图层高度
  animationDuration: Duration(milliseconds: 300), // 点击按钮后过渡动画时间
  child: Text('ClickButton'),
)

显示效果如下图:

CupertinoButton
CupertinoButton(
  child: Text('Click'),
  disabledColor: Colors.blueGrey, //不可点击时按钮颜色,color属性不设置该值无效
  onPressed: null, // onPressed为null视为不可点击
),
CupertinoButton(
  child: Text('Click'),
  color: Colors.lightBlue,
  disabledColor: Colors.blueGrey, //不可点击时按钮颜色,color属性不设置该值无效
  onPressed: null, // onPressed为null视为不可点击
),
CupertinoButton(
  child: Text('Click'),
  color: Colors.lightBlue, // 按钮颜色
  borderRadius: BorderRadius.all(Radius.circular(15.0)), // 按钮圆角设置
  onPressed: () { // onPressed不为null视为可点击
    print('You click the button');
  },
)

显示效果如下图:

输入控件

输入控件同样有两种风格,分别是Material Design风格的TextField和Cupertino风格的CupertinoTextField。

TextField
TextField(
  controller: TextEditingController(text: 'Hello Flutter'), // 默认输入内容
),
TextField(
  decoration: InputDecoration( //输入框装饰
    hintText: 'please input something', // 输入提示
    contentPadding: EdgeInsets.all(10), // 输入框内容padding值
  ),
),
TextField(
  decoration: InputDecoration(
    labelText: 'Nickname', // 输入框文本标签
    labelStyle: TextStyle(color: Colors.black, fontWeight: FontWeight.bold), // 标签文本风格
    hintText: 'please input nickname', // 输入提示
    helperText: 'nickname可由大小写字母、数字和下划线组合,不能包含特殊字符', // 帮助提示文本
  ),
),
TextField(
  decoration: InputDecoration(
    labelText: 'Password', // 输入框文本标签
    labelStyle: TextStyle(color: Colors.black, fontWeight: FontWeight.bold), // 标签文本风格
    hintText: 'please input password', // 输入提示
    errorText: 'password输入错误', // 错误提示文本
    prefixIcon: Icon(Icons.security), // 输入框前缀图标
  ),
)

显示效果如下图:

CupertinoTextField
CupertinoTextField(
  controller: TextEditingController(text: 'Hello Flutter'), // 默认输入内容
),
CupertinoTextField(
  placeholder: 'please input something', // 输入提示
  padding: EdgeInsets.only(left: 10, right: 10), // 输入框内容padding值
),
CupertinoTextField(
  placeholder: 'please input something', // 输入提示
  decoration: BoxDecoration( // 文本框装饰
    color: Colors.lightBlue, // 文本框颜色
    border: Border.all(color: Colors.red, width: 1), // 输入框边框
    borderRadius: BorderRadius.all(Radius.circular(10)), // 输入框圆角设置
    boxShadow: [BoxShadow(color: Colors.redAccent, offset: Offset(0, 5))], //装饰阴影
  ),
),
CupertinoTextField(
  decoration: BoxDecoration( // 文本框装饰
      image: DecorationImage( //文本框装饰背景图片
image: AssetImage('images/flutter_logo.png'),
repeat: ImageRepeat.repeat,
      )
  ),
),
CupertinoTextField(
  placeholder: 'please input something', // 输入提示
  prefix: Text('用户名:'), // 输入框前缀图文
),
CupertinoTextField(
  placeholder: 'please input something', // 输入提示
  prefix: Icon(Icons.security), // 输入框前缀图文
  enabled: false, // 是否可编辑
)

显示效果如下图:

从TextField和CupertinoTextField的属性设置来看,还是有很多实现不一样的地方,所以如果大家要针对iOS和安卓手机原有的风格开发UI时,要根据平台不同使用不同的Widget来实现,不同风格的Widget属性要熟练掌握。具体其它的一些属性请参考官方文档或源码。

选择控件

选择控件包括Material Design风格的Checkbox、Radio、Switch、Slider和Cupertino风格的CupertinoSwitch、CupertinoSlider、CupertinoSegmentedControl等。

Checkbox
Checkbox(
  value: true,
  onChanged: null,
  tristate: true,
),
Checkbox(
  value: null, // 为null时tristate值必须为true,表示有三种状态
  onChanged: (checked) {},
  activeColor: Colors.redAccent, // checkbox颜色
  tristate: true, // 是否有三种状态
),
Checkbox(
  value: false, // 未选中状态
  onChanged: (checked) {},
  activeColor: Colors.redAccent, // checkbox颜色
  tristate: false, // 是否有三种状态
),
Checkbox(
  value: true, // 选中状态
  onChanged: (checked) {},
  activeColor: Colors.redAccent, // checkbox颜色
  tristate: false, // 是否有三种状态
),
Checkbox(
  value: isChecked, // 控件状态值
  onChanged: (checked) {
    print("checked = $checked");
    setState(() { // 状态改变后需要通过setState刷新Widget改变状态
      this.isChecked = checked;
    });
  },
  tristate: true, //是否有三种状态
  activeColor: Colors.blueAccent, // checkbox颜色
)

显示效果如下图:

Radio
String _character = 'A';

Radio<String>(
  value: 'A', // 代表的值
  groupValue: _character, // 当前radio group选中的值,当该值与value值匹配时则被选中
  onChanged: (String newValue) {
    setState(() { // 点击当前控件时更新状态
      _character = newValue;
    });
  },
),
Radio<String>(
  value: 'B',
  groupValue: _character,
  onChanged: (String newValue) {
    setState(() {
      _character = newValue;
    });
  },
),
Radio<String>(
  value: 'C',
  groupValue: _character,
  onChanged: (String newValue) {
    setState(() {
      _character = newValue;
    });
  },
)

显示效果如下图:

Switch
bool _switchChecked = true;

Switch(
  value: true,
  activeColor: Colors.blueAccent, // 激活状态开关颜色
  activeTrackColor: Colors.lightBlue, //激活状态开关轨道颜色
  onChanged: null, // 为null时则开关不可操作
),
Switch(
  value: true,
  activeColor: Colors.blueAccent, // 激活状态开关颜色
  activeTrackColor: Colors.lightBlue, //激活状态开关轨道颜色
  onChanged: (flag) {}, // 为null时则开关不可操作
),
Switch(
  value: false,
  inactiveThumbColor: Colors.white, // 未激活状态开关颜色
  inactiveTrackColor: Colors.grey, // 未激活状态开关轨道颜色
  onChanged: (flag) {},
),
Switch(
  value: _switchChecked,
  onChanged: (flag) {
    setState(() { // 状态改变是通过setState改变状态值
      _switchChecked = flag;
    });
  },
)

显示效果如下图:

Slider
double _sliderValue = 0.3;

Slider(
  value: _sliderValue, // 当前滑块定位到的值
  onChanged: (val) { // 滑动监听
    setState(() { // 通过setState设置当前值
      _sliderValue = val;
    });
  },
  onChangeStart: (val) { // 开始滑动时的监听
    print('changeStart: val = $val');
  },
  onChangeEnd: (val) { // 滑动结束时的监听
    print('changeEnd: val = $val');
  },
  min: 0, // 最小值
  max: 1, // 最大值
  activeColor: Colors.blueAccent, //滑动过的颜色
  inactiveColor: Colors.lightBlueAccent, //未达到的颜色
)

显示效果如下图:

CupertinoSwitch
bool _switchChecked = true;

CupertinoSwitch(
  value: true, //开关值
),
CupertinoSwitch(
  value: false,
),
CupertinoSwitch(
  value: _switchChecked,
  onChanged: (flag) {
    setState(() { // 状态改变是通过setState改变状态值
      _switchChecked = flag;
    });
  },
)

显示效果如下图:

CupertinoSlider
CupertinoSlider(
  value: _sliderValue, // 当前滑块定位到的值
  onChanged: (val) { // 滑动监听
    setState(() { // 通过setState设置当前值
      _sliderValue = val;
    });
  },
  onChangeStart: (val) { // 开始滑动时的监听
    print('changeStart: val = $val');
  },
  onChangeEnd: (val) { // 滑动结束时的监听
    print('changeEnd: val = $val');
  },
  min: 0, // 最小值
  max: 1, // 最大值
  activeColor: Colors.red, //滑动过的颜色
)

显示效果如下图:

CupertinoSegmentedControl

Map<String, Text> map = {'apple': Text('Apple'), 'orange': Text('Orange'), 'banana': Text('Banana')};
String _fruit = 'apple';
  
CupertinoSegmentedControl(
  children: map, // 数据
  groupValue: _fruit, // 选中的数据
  onValueChanged: (fruit) {
    setState(() { // 数据改变时通过setState改变选中状态
      _fruit = fruit;
    });
  },
  unselectedColor: CupertinoColors.white, // 未选中颜色
  selectedColor: CupertinoColors.activeBlue, // 选中颜色
  borderColor: CupertinoColors.activeBlue, // 边框颜色
  pressedColor: const Color(0x33007AFF), // 点击时候的颜色
)

显示效果如下图:

总结

以上为部分常用Widget的一些常用属性详细介绍,其他常用Widget后续继续分享。

 相关推荐

刘强东夫妇:“移民美国”传言被驳斥

京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。

发布于:7月以前  |  808次阅读  |  详细内容 »

博主曝三大运营商,将集体采购百万台华为Mate60系列

日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。

发布于:7月以前  |  770次阅读  |  详细内容 »

ASML CEO警告:出口管制不是可行做法,不要“逼迫中国大陆创新”

据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。

发布于:7月以前  |  756次阅读  |  详细内容 »

抖音中长视频App青桃更名抖音精选,字节再发力对抗B站

今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。

发布于:7月以前  |  648次阅读  |  详细内容 »

威马CDO:中国每百户家庭仅17户有车

日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。

发布于:7月以前  |  589次阅读  |  详细内容 »

研究发现维生素 C 等抗氧化剂会刺激癌症生长和转移

近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。

发布于:7月以前  |  449次阅读  |  详细内容 »

苹果据称正引入3D打印技术,用以生产智能手表的钢质底盘

据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。

发布于:7月以前  |  446次阅读  |  详细内容 »

千万级抖音网红秀才账号被封禁

9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...

发布于:7月以前  |  445次阅读  |  详细内容 »

亚马逊股东起诉公司和贝索斯,称其在购买卫星发射服务时忽视了 SpaceX

9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。

发布于:7月以前  |  444次阅读  |  详细内容 »

苹果上线AppsbyApple网站,以推广自家应用程序

据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。

发布于:7月以前  |  442次阅读  |  详细内容 »

特斯拉美国降价引发投资者不满:“这是短期麻醉剂”

特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。

发布于:7月以前  |  441次阅读  |  详细内容 »

光刻机巨头阿斯麦:拿到许可,继续对华出口

据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。

发布于:7月以前  |  437次阅读  |  详细内容 »

马斯克与库克首次隔空合作:为苹果提供卫星服务

近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。

发布于:7月以前  |  430次阅读  |  详细内容 »

𝕏(推特)调整隐私政策,可拿用户发布的信息训练 AI 模型

据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。

发布于:7月以前  |  428次阅读  |  详细内容 »

荣耀CEO谈华为手机回归:替老同事们高兴,对行业也是好事

9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。

发布于:7月以前  |  423次阅读  |  详细内容 »

AI操控无人机能力超越人类冠军

《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。

发布于:7月以前  |  423次阅读  |  详细内容 »

AI生成的蘑菇科普书存在可致命错误

近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。

发布于:7月以前  |  420次阅读  |  详细内容 »

社交媒体平台𝕏计划收集用户生物识别数据与工作教育经历

社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”

发布于:7月以前  |  411次阅读  |  详细内容 »

国产扫地机器人热销欧洲,国产割草机器人抢占欧洲草坪

2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。

发布于:7月以前  |  406次阅读  |  详细内容 »

罗永浩吐槽iPhone15和14不会有区别,除了序列号变了

罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。

发布于:7月以前  |  398次阅读  |  详细内容 »
 相关文章
如何有效定位Flutter内存问题? 3年以前  |  14815次阅读
Flutter的手势GestureDetector分析详解 4年以前  |  10915次阅读
Flutter插件详解及其发布插件 4年以前  |  10696次阅读
在Flutter中添加资源和图片 5年以前  |  7943次阅读
 目录