五、Flutter的基础Widget

一、文本Widget

Android中使用TextView,在iOS中使用UILabel来显示文本,在Flutter中使用Text组件来控制文本的显示

1.1 普通文本的显示

将文本的控制显示分成两类:

  • 控制文本布局的参数: textAlign(文本对齐方式)、textDirection(文本排版方向 )、maxLines(文本显示最大行数 )、 overflow (文本截断规则)等,都是构造函数中的参数;
  • 控制文本样式的参数: fontFamily(字体名称)、 fontSize(字体大小)、 color(文本颜色)、 shadows(文本阴影) 等,这些参数被封装到构造函数中的 style 参数中;
class Text extends StatelessWidget {

  const Text(
    this.data, {//文本内容 必传
    Key key,
    this.style,//用于设置文本内容的样式
    this.strutStyle,
    this.textAlign,//文本对齐方式
    this.textDirection,
    this.locale,
    this.softWrap,
    this.overflow,//文本溢出处理方式
    this.textScaleFactor,//逻辑像素的字体像素数 简单理解就是放大比例
    this.maxLines,//显示最大行数
    this.semanticsLabel,
    this.textWidthBasis,
    this.textHeightBehavior,
  }) : 
       textSpan = null,
       _applyTextScaleFactorToWidgetSpan = true,
       super(key: key);

上面仅对部分常用参数做了介绍,其余参数平常很少用

const TextStyle({
    this.inherit = true,
    this.color,//文本颜色 如果指定了foreground,则此值必须为null
    this.backgroundColor,//文本背景色
    this.fontSize,//字体大小
    this.fontWeight,
    this.fontStyle,//字体变体  斜体 直体
    this.letterSpacing,
    this.wordSpacing,//单词之间添加的空间间隔
    this.textBaseline,// 对齐文本的水平线
    this.height,//文本行与行的高度
    this.locale,
    this.foreground,//文本的前景色,不能与color共同设置
    this.background,//文本背景色
    this.shadows,//Flutter Decoration背景设定(边框、圆角、阴影、形状、渐变、背景图像等)
    this.fontFeatures,
    this.decoration,//绘制文本装饰  下划线 上划线 删除线
    this.decorationColor,//绘制文本装饰的颜色
    this.decorationStyle,//绘制文本装饰的样式 虚线 两条线 正弦线等
    this.decorationThickness,
    this.debugLabel,
    String fontFamily,//使用的字体名称
    List<String> fontFamilyFallback,
    String package,
  }) : fontFamily = package == null ? fontFamily : 'packages/$package/$fontFamily',
       _fontFamilyFallback = fontFamilyFallback,
       _package = package;

示例代码:

class TextDemo extends StatelessWidget {
  const TextDemo({
    Key key,
  }) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return Text(
      "《定风波》苏轼 莫听穿林打叶声,何妨吟啸且徐行。竹杖芒鞋轻胜马,谁怕?一蓑烟雨任平生。",
      textAlign: TextAlign.center,//文本对齐方式
      maxLines: 2,//显示最大行数
      overflow: TextOverflow.ellipsis,//文本溢出处理方式
      textScaleFactor: 1.3,//逻辑像素的字体像素数 简单理解就是放大比例
      style: TextStyle(
        decoration: TextDecoration.lineThrough,
        decorationColor: Colors.yellow,
        decorationStyle: TextDecorationStyle.wavy,
        fontSize: 15,
        fontWeight: FontWeight.bold,
        textBaseline: TextBaseline.alphabetic,
        wordSpacing: 10,
      ),
    );
  }
}

效果:


代码演示

1.2 富文本展示

Android中,使用SpannableString,在iOS中,使用NSAttributedString达到效果,在Flutter中使用Text组件来控制文本的显示

const Text.rich(//Text组件的另一个构造函数
    this.textSpan, {//必传参数 
    Key key,
    this.style,
    this.strutStyle,
    this.textAlign,
    this.textDirection,
    this.locale,
    this.softWrap,
    this.overflow,
    this.textScaleFactor,
    this.maxLines,
    this.semanticsLabel,
    this.textWidthBasis,
    this.textHeightBehavior,
    bool applyTextScaleFactorToWidgetSpan = false,
  }) : 
       data = null,//也可以传data 另外可以通过分片拼接
       _applyTextScaleFactorToWidgetSpan = applyTextScaleFactorToWidgetSpan,
       super(key: key);
  const TextSpan({
    this.text,
    this.children,//重要 拼接
    TextStyle style,
    this.recognizer,
    this.semanticsLabel,
  }) : super(style: style,);

示例代码:

class RichTextDemo extends StatelessWidget {
  const RichTextDemo({
    Key key,
  }) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return Text.rich(
      TextSpan(
        text: 'helloworld', style: TextStyle( color: Colors.red, fontSize: 25, ),
        children: [
          TextSpan( text: 'helloworld', style: TextStyle( color: Colors.green, fontSize: 22, ), ),
          WidgetSpan(child: Icon(Icons.favorite, color: Colors.red,)),
          TextSpan( text: 'helloworld', style: TextStyle( color: Colors.blue, fontSize: 20, ) ),
        ]
      ),
    );
  }
}

代码效果:


WechatIMG6.jpeg

进阶:通过源码可以得知,不管是普通文本还是富文本,只是两种Text的两种构造方法而已,在build函数中均是返回一个RichText对象

image.png

后面再学习过程中,我们都要学会看源码,查看他们底层是如何在使用

二、按钮Widget

2.1 按钮的基础

Material widget库中提供了多种按钮WidgetFloatingActionButton()、RaisedButton(浮动按钮带阴影)、FlatButton(扁平化)、OutlineButton(带边框按钮)等,都是继承于MaterialButton

话不多说,直接上代码:

const MaterialButton({
    Key key,
    @required this.onPressed,//方法要求实现 激活按钮调用方法
    this.onLongPress,//长按方法
    this.onHighlightChanged,//
    this.mouseCursor,
    this.textTheme,//定义按钮的基色,以及按钮的最小尺寸,内部填充和形状的默认值
    this.textColor,//按钮文本的颜色
    this.disabledTextColor,//disabled状态下的文本颜色
    this.color,//填充颜色
    this.disabledColor,//disabled状态下的按钮颜色
    this.focusColor,
    this.hoverColor,
    this.highlightColor,
    this.splashColor,//按钮被按下的水波纹颜色
    this.colorBrightness,//按钮的主题亮度,当设置了textColor、color颜色,此值无效!
    this.elevation,//设置阴影大小
    this.focusElevation,
    this.hoverElevation,
    this.highlightElevation,
    this.disabledElevation,
    this.padding,//内部边距调整
    this.visualDensity,
    this.shape,//按钮形状, 子类才起效果
    this.clipBehavior = Clip.none,
    this.focusNode,
    this.autofocus = false,
    this.materialTapTargetSize,//按钮默认上下有间距 通过此属性修改
    this.animationDuration,
    this.minWidth,
    this.height,
    this.enableFeedback = true,
    this.child,
  }) : super(key: key);

示例代码:

class ButtonDemo extends StatelessWidget {
  const ButtonDemo({
    Key key,
  }) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return Column(
      children: [
        RaisedButton(
          onPressed: (){

          },
          textTheme: ButtonTextTheme.primary,// 按钮文本基于ThemeData.primaryColor
          highlightColor: Colors.red,
          child: Column(
            children: [
              Text("高亮红色 竖", style: TextStyle(color: Colors.white),),
              Icon(Icons.favorite,)
            ],
          ),
        ),
        RaisedButton(
          onPressed: (){

          },
          highlightColor: Colors.red,
          elevation: 10,//阴影
          child: Row(
            children: [
              Text("高亮红色 横", style: TextStyle(color: Colors.white),),
              Icon(Icons.favorite,)
            ],
          ),
        ),
        RaisedButton(
          onPressed: (){

          },
          color: Colors.orange,
          textColor: Colors.white,
          child: Text("datafdfdsfdsfdsfsdfdsfdsfdsfds"
              "sf"),
        ),
        FlatButton(//扁平化按钮
          onPressed: (){
            
          },
          child: Text("data"),
          color: Colors.orange,
          textColor: Colors.white,
        ),
        OutlineButton(onPressed: (){ //带边框按钮

        },
          child: Text("data"),
          color: Colors.purple,
          textColor: Colors.orange,
        ),
        FloatingActionButton(
          child: Icon(Icons.add),
          onPressed: (){

          },
        ),
        FlatButton(
          onPressed: (){

          },
          child: Row(
            mainAxisSize: MainAxisSize.min,
            children: [
              Icon(Icons.favorite, color: Colors.red,),
              Text("喜欢")
            ],
          ),
          color: Colors.orange,
          shape: RoundedRectangleBorder(
            borderRadius: BorderRadius.circular(10),
          ),
        ),
      ],
    );
  }
}

代码效果:


Simulator Screen Shot - iPhone 11 - 2020-11-07 at 17.26.03.png
问题1:button上下有默认间距 如何调整?(materialTapTargetSize属性)
问题2:button如何自由调整大小?(有ButtonTheme包裹,修改)
问题3:button内文字的内边距如何调整?(padding属性)

三、图片Widget

Flutter中使用Image组件

Image组件有很多的构造函数,我们这里主要学习前3个:

  • Image.assets:加载本地资源图片;
  • Image.network:加载网络中的图片;
  • Image()
  • Image.file

1.1 本地图片 网络图片

我们先看Image()

const Image({
    Key key,
    @required this.image,//必传参数
    this.frameBuilder,
    this.loadingBuilder,
    this.errorBuilder,
    this.semanticLabel,
    this.excludeFromSemantics = false,
    this.width,//控件宽度
    this.height,//控件高度
    this.color,//图片的混合色值 不设置colorBlendMode会直接覆盖图片
    this.colorBlendMode,////混合模式
    this.fit,//填充模式
    this.alignment = Alignment.center,////对齐方式
    this.repeat = ImageRepeat.noRepeat,//重复方式
    this.centerSlice,//指定中心区域进行九个补丁图像 类似.9图
    this.matchTextDirection = false,
    this.gaplessPlayback = false,
    this.isAntiAlias = false,
    this.filterQuality = FilterQuality.low,
  }) : super(key: key);

示例代码:

class ImageDemo extends StatelessWidget {
  const ImageDemo({
    Key key,
  }) : super(key: key);

  @override
  Widget build(BuildContext context) {
    return ListView(
      children: [
        Column(
          children: [
            Image(image: NetworkImage('https://timgsa'
                '.baidu.com/timg?image&quality=80&size=b9999_10000&sec=1604055800173&'
                'di=3b254a30548dd54b2479bd895ee39f00&imgtype=0&src=http%3A%2F%2Fb-ssl.'
                'duitang.com%2Fuploads%2Fitem%2F201804%2F19%2F20180419202214_nrvqc.jpeg'),
              width: 200,
              height: 200,
              fit: BoxFit.contain,
              repeat: ImageRepeat.repeatY,
              // alignment: Alignment.bottomLeft,
              // -1  1范围
              alignment: Alignment(0, 0),
              color: Colors.red,//并不是背景色 而是混入颜色
              colorBlendMode: BlendMode.colorBurn,//混入模式
            ),
            FadeInImage(
              placeholder: AssetImage('assets/images/golden_box.png'),
              image: NetworkImage('https://timgsa.baidu.com/timg?image&quality=80&siz'
                'e=b9999_10000&sec=1604058608291&di=0c17e499e9d33439f61a1d1769b440fa&i'
                'mgtype=0&src=http%3A%2F%2F01.minipic.eastday.com%2F20170407%2F20170407'
                '003654_e05a7fc92c762cbe49f8c0adf0d0b41d_1.jpeg'),
              fadeInDuration: Duration(seconds: 5),
              fadeOutDuration: Duration(seconds: 5),
            ),
            Image.network('https://timgsa.baidu.com/timg?image&quality=80&siz'
                'e=b9999_10000&sec=1604058608291&di=0c17e499e9d33439f61a1d1769b440fa&i'
                'mgtype=0&src=http%3A%2F%2F01.minipic.eastday.com%2F20170407%2F20170407'
                '003654_e05a7fc92c762cbe49f8c0adf0d0b41d_1.jpeg'),
            Image(image: AssetImage('assets/images/global_logo.png')),
            Image.asset('assets/images/golden_box.png'),
            CircleAvatar(
              backgroundImage: NetworkImage('https://timgsa.baidu.com/timg?image&quality=80&siz'
                  'e=b9999_10000&sec=1604058608291&di=0c17e499e9d33439f61a1d1769b440fa&i'
                  'mgtype=0&src=http%3A%2F%2F01.minipic.eastday.com%2F20170407%2F20170407'
                  '003654_e05a7fc92c762cbe49f8c0adf0d0b41d_1.jpeg'),
              child: Container(
                alignment: Alignment(0, .5),
                width: 200,
                height: 200,
                child: Text("我是谁", style: TextStyle(fontSize: 8, color: Colors.red),)
              ),
            )
          ],
        ),
      ],
    );
  }
}

代码效果:


Simulator Screen Shot - iPhone 11 - 2020-11-07 at 17.47.16.png

本地图片的配置

image.png

1.2 网络占位图

通过FadeInImage实现网络请求占位图

const FadeInImage({
    Key key,
    @required this.placeholder,//必传
    this.placeholderErrorBuilder,
    @required this.image,//必传
    this.imageErrorBuilder,
    this.excludeFromSemantics = false,
    this.imageSemanticLabel,
    this.fadeOutDuration = const Duration(milliseconds: 300),//效果
    this.fadeOutCurve = Curves.easeOut,
    this.fadeInDuration = const Duration(milliseconds: 700),//效果
    this.fadeInCurve = Curves.easeIn,
    this.width,
    this.height,
    this.fit,
    this.alignment = Alignment.center,
    this.repeat = ImageRepeat.noRepeat,
    this.matchTextDirection = false,
  }) : super(key: key);

1.3 圆角头像

Flutter中实现圆角效果也是使用一些Widget来实现的。

1、CircleAvatar

const CircleAvatar({
    Key key,
    this.child,
    this.backgroundColor,
    this.backgroundImage,
    this.onBackgroundImageError,
    this.foregroundColor,
    this.radius,
    this.minRadius,
    this.maxRadius,
  }) :  super(key: key);

2、ClipOval

class _ZQHomeContentState extends State<ZQHomeContent> {
  @override
  Widget build(BuildContext context) {
    return ClipOval(
        child: Image.network(
          'https://tva1.sinaimg.cn/large/006y8mN6gy1g7aa03bmfpj3069069mx8.jpg',
          width: 200,
          height: 200,
        )
    );
  }
}

3、Container+BoxDecoration

后续讲到Container时再着重讲解

上面三种方法的使用

class _ZQHomeContentState extends State<ZQHomeContent> {
  @override
  Widget build(BuildContext context) {
    return Column(
      children: [
        Container(
          width: 200,
            height: 200,
            decoration: BoxDecoration(
              gradient: LinearGradient(
                colors: [Colors.red, Colors.cyan],
              ),
              borderRadius:BorderRadius.all(Radius.circular(200)),
              image: DecorationImage(
                image: NetworkImage(
                  'https://tva1.sinaimg.cn/large/006y8mN6gy1g7aa03bmfpj3069069mx8.jpg',
                ),
              ),
            ),
        ),
        CircleAvatar(
          radius: 100,
          backgroundImage: NetworkImage('https://timgsa.baidu.com/timg?image&quality=80&siz'
              'e=b9999_10000&sec=1604058608291&di=0c17e499e9d33439f61a1d1769b440fa&i'
              'mgtype=0&src=http%3A%2F%2F01.minipic.eastday.com%2F20170407%2F20170407'
              '003654_e05a7fc92c762cbe49f8c0adf0d0b41d_1.jpeg'),
        ),
        ClipOval(
            child: Image.network(
              "https://tva1.sinaimg.cn/large/006y8mN6gy1g7aa03bmfpj3069069mx8.jpg",
              width: 200,
              height: 200,
            )
        )
      ],
    );
  }
}

1.4 Icon

Icon字体图标和图片图标
字体图片矢量图 放大的时候不会失真

class IconDemo extends StatelessWidget {
  const IconDemo({
    Key key,
  }) : super(key: key);

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