Flutter07-基础组件-图片

1.图片

1.1.Image

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

1.1.1.从asset中加载图片

  1. 在工程根目录下创建一个images目录,并将图片 avatar.png 拷贝到该目录。

  2. 在pubspec.yaml中的flutter部分添加如下内容:

assets:
    - images/avatar.png

注意: 由于 yaml 文件对缩进严格,所以必须严格按照每一层两个空格的方式进行缩进,此处 assets 前面应有两个空格。

  1. 加载该图片
Image(
  image: AssetImage("assets/images/avatar.jpeg"),
  width: 100.0
);

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

Image.asset(
  "assets/images/avatar.jpeg",
  width: 100.0,
),

1.1.2.从网络加载图片

Image(
  image: NetworkImage(
    "https://avatars2.githubusercontent.com/u/20411648?s=460&v=4"),
  width: 100.0,
),

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

Image.network(
  "https://avatars2.githubusercontent.com/u/20411648?s=460&v=4",
  width: 100.0,
)

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

image-20240217154111470

1.1.3.参数

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:会拉伸填充满显示空间,图片本身长宽比会发生变化,图片会变形。
    • contain:这是图片的默认适应规则,图片会在保证图片本身长宽比不变的情况下缩放以适应当前显示空间,图片不会变形。
    • cover:会按图片的长宽比放大后居中填满显示空间,图片不会变形,超出显示空间部分会被剪裁。
    • fitWidth:图片的宽度会缩放到显示空间的宽度,高度会按比例缩放,然后居中显示,图片不会变形,超出显示空间部分会被剪裁。
    • fitHeight:图片的高度会缩放到显示空间的高度,宽度会按比例缩放,然后居中显示,图片不会变形,超出显示空间部分会被剪裁。
    • none:图片没有适应策略,会在显示空间内显示图片,如果图片比显示空间大,则显示空间只会显示图片中间部分。

我们对一个宽高相同的头像图片应用不同的fit值,效果如图所示: image-20240308212808375

void main(List<String> args) {
  var img = const AssetImage('assets/images/avatar.jpeg');
  runApp(MaterialApp(
    title: 'My app',
    home: Scaffold(
      appBar: AppBar(
        title: const Text('图片伸缩'),
      ),
      body: SingleChildScrollView(
        child: Column(
          children: [
            Row(
              children: [
                Image(
                  image: img,
                  width: 100.0,
                  height: 100.0,
                  fit: BoxFit.fill,
                ),
                const Text('BoxFit.fill'),
              ],
            ),
            Row(
              children: [
                Image(
                  image: img,
                  width: 100.0,
                  height: 100.0,
                  fit: BoxFit.contain,
                ),
                const Text('BoxFit.contain'),
              ],
            ),
            Row(
              children: [
                Image(
                  image: img,
                  width: 100.0,
                  height: 100.0,
                  fit: BoxFit.cover,
                ),
                const Text('BoxFit.cover'),
              ],
            ),

            Row(
              children: [
                Image(
                  image: img,
                  width: 100.0,
                  height: 100.0,
                  fit: BoxFit.fitWidth,
                ),
                const Text('BoxFit.fitWidth'),
              ],
            ),

            Row(
              children: [
                Image(
                  image: img,
                  width: 100.0,
                  height: 100.0,
                  fit: BoxFit.fitHeight,
                ),
                const Text('BoxFit.fitHeight'),
              ],
            ),

            Row(
              children: [
                Image(
                  image: img,
                  width: 100.0,
                  height: 100.0,
                  fit: BoxFit.none,
                ),
                const Text('BoxFit.none'),
              ],
            ),
          ],
        ),
      ),
    ),
  ));
}
  • colorcolorBlendMode:在图片绘制时可以对每一个像素进行颜色混合处理,color指定混合色,而colorBlendMode指定混合模式,下面是一个简单的示例:
Image(
  image: img,
  width: 100.0,
  height: 100.0,
  color: Colors.blue,
  colorBlendMode: BlendMode.difference,
)
  • 运行效果如图所示:

image-20240217154435797

  • repeat:当图片本身大小小于显示空间时,指定图片的重复规则。简单示例如下:
Image(
  image: img,
  width: 100.0,
  height: 200.0,
  repeat: ImageRepeat.repeatY,
)

运行后效果如图所示:

image-20240217154455076

1.2.ICON

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

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

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

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

1.2.1.使用Material Design字体图标

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

flutter:
  uses-material-design: true

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

我们看一个简单的例子:

Row(
  children: const [
    Text(
      '\uE03e',
      style: TextStyle(
        fontFamily: "MaterialIcons",
        fontSize: 24.0,
        color: Colors.green,
      ),
    ),
    Text(
      '\uE237',
      style: TextStyle(
        fontFamily: "MaterialIcons",
        fontSize: 24.0,
        color: Colors.green,
      ),
    ),
    Text(
      '\uE287',
      style: TextStyle(
        fontFamily: "MaterialIcons",
        fontSize: 24.0,
        color: Colors.green,
      ),
    ),
  ],
)

运行效果如图所示:

image-20240217154700072

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

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静态变量定义。

1.2.2.使用自定义字体图标

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

iconfont字体图标下载教程,https://blog.csdn.net/wingchiehpih/article/details/105875227

  • 导入字体图标文件;这一步和导入字体文件相同,假设我们的字体图标文件保存在项目根目录下,路径为"fonts/iconfont.ttf":
fonts:
  - family: myIcon  #指定一个字体名
    fonts:
      - asset: fonts/iconfont.ttf
  • 为了使用方便,我们定义一个MyIcons类,功能和Icons类一样:将字体文件中的所有图标都定义成静态变量:
class MyIcons {
  static const IconData member =
      IconData(0xe60d, fontFamily: 'myIcon', matchTextDirection: true);

  static const IconData service =
      IconData(0xe608, fontFamily: 'myIcon', matchTextDirection: true);
}
  • 使用
Row(
  mainAxisAlignment: MainAxisAlignment.center,
  children: const <Widget>[
    Icon(MyIcons.member, color: Colors.purple),
    Icon(MyIcons.service, color: Colors.purple),
  ],
)

运行后效果如图所示:

image-20240217155030309