初略讲解Flutter的Image和Icon(基础组件)

562 阅读6分钟

在Flutter中,可以通过ImageWidget来加载并显示图片,Image的数据源可以是asset、文件和内存以及网络。

ImageProvider

ImageProvider是一个抽象类,主要定义了图片数据获取的接口load(),从不同的数据源获取图片需要实现不同的ImageProvider,如AssetImage是实现了从asset中加载图片的ImageProvider,而NetworkImage实现了从网络中加载图片的ImageProvider

Image

Image Widget有一个必选的image参数,它对应一个ImageProvider(AssetImageNetworkImage)。下面我们分别演示一个如何从asset和网络中加载图片。

从asset中加载图片

  1. 在工程跟目录下创建一个images目录,并将图片avatar.jpg拷贝到该目录。
  2. pubspec.yaml中的flutter部分添加如下内容:
      assets:
        - images/avatar.jpg
    
    注意:由于yaml文件对缩进严格,所以必须严格按照每一层两个空格的方式进行缩进,此处assets前面应有两个空格。
  3. 加载该图片
    Image(
      image: AssetImage("images/avatar.jpg"),
      width: 100.0
    );
    
    Image也提供了一个快捷的构造函数Image.asset用于从asset中加载并显示图片:
    Image.asset("images/avatar.jpg",
      width: 100.0,
    )
    
    运行上面两个示例,图片加载成功后显示如下:

从网络中加载图片

Image(
    image: NetworkImage("https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=3513223148,3798354759&fm=11&gp=0.jpg"),
    width:100.0,
)

Image也提供了一个快捷的构造函数Image.network用于从网络中加载并显示图片:

Image.network(
    "https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=3513223148,3798354759&fm=11&gp=0.jpg", 
    width:100.0,
)

运行上面两个示例,图片加载成功后显示如下:

属性

Image在显示图片时定义了一系列属性,通过这些属性我们可以控制图片的显示外观、大小和混合效果等,我们看一下Image的主要属性:

const Image({
  ...
  this.width, //图片的宽
  this.height, //图片高度
  this.color, //图片的混合色值
  this.colorBlendMode, //混合模式
  this.fit,//缩放模式
  this.alignment = Alignment.center, //对齐方式
  this.repeat = ImageRepeat.noRepeat, //重复方式
  ...
})
  • widthheight:用于设置图片的宽、高,当不指定宽高时,图片会根据当前父容器的限制,尽可能的显示其原始大小,如果只设置widthheight的其中一个,那么另一个属性默认会按比例缩放,但也可以通过下面介绍的fit属性来指定适应规则。
  • fit:该属性用于在图片的显示空间和图片本身大小不同时指定图片的适应模式。适应模式是在BoxFit中定义,它是一个枚举类型,有如下值:
    • fill:会拉伸填充满显示空间,图片本身长宽比会发生变化,图片会变形。
    • cover:会按图片的长宽比放大后居中填满显示空间,图片不会变形,超出显示空间部分会被裁剪。
    • contain:这是图片的默认适应规则,图片会在保证图片本身长宽比不变的情况下缩放以适应当前显示空间,图片不会变形,
    • fitWidth:图片的宽度会缩放到显示空间的宽度,高度会按比例缩放,然后居中显示,图片不会变形,超出显示空间部分会被裁剪。
    • fitHeight:图片的高度会缩放到显示空间的高度,宽度会按比例缩放,然后居中显示,图片不会变形,超出显示空间部分会被裁剪。
    • none:图片没有适应策略,会在显示空间内显示图片,如果图片比显示空间大,则显示空间只会显示图片中间部分。

  • colorcolorBlendMode:在图片绘制时可以对每一个像素进行颜色混合处理,color指定混合色,而colorBlendMode指定混合模式,下面是一个简单的示例:

    Image(
      image: AssetImage("images/avatar.jpg"),
      width: 100.0,
      color: Colors.blue,
      colorBlendMode: BlendMode.difference,
    );
    

    运行效果如下(彩色):

  • repeat:当图片本身大小小于显示空间时,指定图片的重复规则,简单示例如下:

    Image(
      image: AssetImage("images/avatar.jpg"),
      width: 100.0,
      height: 200.0,
      repeat: ImageRepeat.repeatY ,
    )
    

    运行效果如下:

完整示例代码如下:

import 'package:flutter/material.dart';

class ImageAndIconRoute extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    var img=AssetImage("images/avatar.jpg");
    return SingleChildScrollView(
      child: Column(
        children: <Image>[
          Image(
            image: img,
            height: 50.0,
            width: 100.0,
            fit: BoxFit.fill,
          ),
          Image(
            image: img,
            height: 50,
            width: 50.0,
            fit: BoxFit.contain,
          ),
          Image(
            image: img,
            width: 100.0,
            height: 50.0,
            fit: BoxFit.cover,
          ),
          Image(
            image: img,
            width: 100.0,
            height: 50.0,
            fit: BoxFit.fitWidth,
          ),
          Image(
            image: img,
            width: 100.0,
            height: 50.0,
            fit: BoxFit.fitHeight,
          ),
          Image(
            image: img,
            width: 100.0,
            height: 50.0,
            fit: BoxFit.scaleDown,
          ),
          Image(
            image: img,
            height: 50.0,
            width: 100.0,
            fit: BoxFit.none,
          ),
          Image(
            image: img,
            width: 100.0,
            color: Colors.blue,
            colorBlendMode: BlendMode.difference,
            fit: BoxFit.fill,
          ),
          Image(
            image: img,
            width: 100.0,
            height: 200.0,
            repeat: ImageRepeat.repeatY ,
          )
        ].map((e){
          return Row(
            children: <Widget>[
              Padding(
                padding: EdgeInsets.all(16.0),
                child: SizedBox(
                  width: 100,
                  child: e,
                ),
              ),
              Text(e.fit.toString())
            ],
          );
        }).toList()
      ),
    );
  }
}

Image缓存

Flutter框架对加载过的图片是有缓存的(内存),默认最大缓存数量是1000,最大缓存空间为100M。关于Image的详细内容及原理将会在后续进行深入介绍。

Icon

在Flutter中,可以像Web开发一样使用iconfont(即“字体图标”),它是将图标做成字体文件,然后通过指定不同的字符而显示不同的图片。

在字体文件中,每一个字符都对应一个位码,而每一个位码都对应一个显示字符,不同的字体就是指字形不同,即字符对应的字形是不同的。而在iconfont中,只是将位码对应的字形做成了图标,所以不同的字符最终就会渲染成不同的图标。

在Flutter开发中,iconfont和图片相比有如下优势:

  1. 体积小:可以减小安装包大小。
  2. 矢量的:iconfont都是矢量图标,放大不会影响其清晰度。
  3. 可以应用文本样式:可以像文本一样改变字体图标的颜色、大小和对齐方式等。
  4. 可以通过TextSpan和文本混用。

使用Material Design字体图标

Flutter默认包含了一套Material Design的字体图标,在pubspec.yaml文件中的配置如下:

flutter:
  uses-material-design: true

Material Design所有图标可以在其官网查看:material.io/tools/icons…

示例:

String icons = "";
// accessible: &#xE914; or 0xE914 or E914
icons += "\uE914";
// error: &#xE000; or 0xE000 or E000
icons += " \uE000";
// fingerprint: &#xE90D; or 0xE90D or E90D
icons += " \uE90D";

Text(icons,
  style: TextStyle(
      fontFamily: "MaterialIcons",
      fontSize: 24.0,
      color: Colors.green
  ),
);

运行效果如下:

通过这个示例可以看到,使用图标就像使用文本一样,但是这种方式需要我们提供每个图标的码点,这对开发者来说并不友好,所以Flutter封装了一个IconDataIcon来专门显示字体图标,上面的例子也可以用如下方式实现:

child: Row(
    mainAxisAlignment: MainAxisAlignment.center,
    children: <Widget>[
        Icon(Icons.accessible, color: Colors.green),
        Icon(Icons.error, color: Colors.green),
        Icon(Icons.fingerprint, color: Colors.green)
    ],
),

Icons类中包含了所有Material Design图标的IconData静态变量定义。

使用自定义字体图标

我们也可以使用自定义字体图标,iconfont.cn上有很多字体图标素材,我们选择自己需要的图标打包下载后,会生成一些不同格式的字体文件,在Flutter中,我们使用ttf格式即可。

假设我们项目中需要使用一个书籍图标和微信图标,那么我们首先应将其打包下载后导入:

  1. 打包下载书籍图标和微信图标

  2. 选择使用ttf格式的文件

  3. 导入字体图标文件;这一步和导入字体文件相同,如我们的字体图标文件保存在项目根目录下,路径为“fonts/iconfont.ttf”:

    fonts:
      - family: iconfont  #指定一个字体名
        fonts:
          - asset: fonts/iconfont.ttf
    
  4. 为了方便使用,定义一个MyIcons类,功能和Icons类一样:将字体文件中的所有图标都定义为静态变量:

    class MyIcons{
      // book 图标
      static const IconData book = const IconData(
          0xe606,
          fontFamily: 'iconfont', 
          matchTextDirection: true
      );
      // 微信图标
      static const IconData wechat = const IconData(
          0xe64f,
          fontFamily: 'iconfont', 
          matchTextDirection: true
      );
    }
    
  5. 使用

    Row(
        mainAxisAlignment: MainAxisAlignment.center,
        children: <Widget>[
            Icon(MyIcons.book,color: Colors.blue,),
            Icon(MyIcons.wechat,color: Colors.green,),
        ],
    )
    
  6. 运行效果如下: