图片

Flutter中,我们可以通过Image组件来加载并显示图片,Image的数据源可以是asset、文件、内存以及网络。
image.png

ImageProvider

ImageProvider 是一个抽象类,主要定义了图片数据获取的接口load(),从不同的数据源获取图片需要实现不同的ImageProvider

  • AssetImage 是实现了从Asset中加载图片的 ImageProvider
  • NetworkImage 实现了从网络加载图片的 ImageProvider

Image

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

从asset中加载图片

  1. 在工程根目录下创建一个 images 目录,并将图片 avatar.png 拷贝到该目录。
  2. 在 pubspec.yaml 中的 flutter 部分添加如下内容:

    1. assets:
    2. - images/avatar.png

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

  3. 加载该图片

    1. Image(
    2. image: AssetImage("images/avatar.png"),
    3. width: 100.0
    4. );

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

  1. Image.asset("images/avatar.png",
  2. width: 100.0,
  3. )
  4. #参数属性
  5. (new) Image Image.asset(String name,
  6. {Key key,
  7. AssetBundle bundle,
  8. Widget Function(BuildContext,
  9. Widget,
  10. int,
  11. bool) frameBuilder,
  12. String semanticLabel,
  13. bool excludeFromSemantics = false,
  14. double scale,
  15. double width,
  16. double height,
  17. Color color,
  18. BlendMode colorBlendMode,
  19. BoxFit fit,
  20. AlignmentGeometry alignment = Alignment.center,
  21. ImageRepeat repeat = ImageRepeat.noRepeat,
  22. Rect centerSlice,
  23. bool matchTextDirection = false,
  24. bool gaplessPlayback = false,
  25. String package,
  26. FilterQuality filterQuality = FilterQuality.low,
  27. int cacheWidth,
  28. int cacheHeight})

从网络加载图片

  1. Image(
  2. image: NetworkImage(
  3. "http://n.sinaimg.cn/sinakd10112/175/w1080h1495/20200326/f2ac-irkazzv3170145.jpg"),
  4. width: 100.0,
  5. fit:BoxFit.cover // 充满父容器
  6. )

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

  1. Image.network(
  2. "http://n.sinaimg.cn/sinakd10112/175/w1080h1495/20200326/f2ac-irkazzv3170145.jpg",
  3. width: 100.0,
  4. fit:BoxFit.cover // 充满父容器
  5. )

运行上面两个示例,图片加载成功后如图3-17所示:
image.png

image参数属性

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

  1. (new) Image Image({Key key,
  2. ImageProvider<dynamic> image,
  3. Widget Function(BuildContext,Widget,int,bool) frameBuilder,
  4. Widget Function(BuildContext,Widget,ImageChunkEvent) loadingBuilder,
  5. String semanticLabel,
  6. bool excludeFromSemantics = false,
  7. double width,
  8. double height,
  9. Color color,
  10. BlendMode colorBlendMode,
  11. BoxFit fit,
  12. AlignmentGeometry alignment = Alignment.center,
  13. ImageRepeat repeat = ImageRepeat.noRepeat,
  14. Rect centerSlice,
  15. bool matchTextDirection = false,
  16. bool gaplessPlayback = false,
  17. FilterQuality filterQuality = FilterQuality.low})
  • width height 用于设置图片的宽、高,当不指定宽高时,图片会根据当前父容器的限制,尽可能的显示其原始大小,如果只设置width、height的其中一个,那么另一个属性默认会按比例缩放,但可以通过下面介绍的fit属性来指定适应规则。
  • fit 该属性用于在图片的显示空间和图片本身大小不同时指定图片的适应模式。适应模式是在BoxFit中定义,它是一个枚举类型,有如下值:
    • fill 会拉伸填充满显示空间,图片本身长宽比会发生变化,图片会变形。
    • cover 会按图片的长宽比放大后居中填满显示空间,图片不会变形,超出显示空间部分会被剪裁。
    • contain 这是图片的默认适应规则,图片会在保证图片本身长宽比不变的情况下缩放以适应当前显示空间,图片不会变形。
    • fitWidth 图片的宽度会缩放到显示空间的宽度,高度会按比例缩放,然后居中显示,图片不会变形,超出显示空间部分会被剪裁。
    • fitHeight 图片的高度会缩放到显示空间的高度,宽度会按比例缩放,然后居中显示,图片不会变形,超出显示空间部分会被剪裁。
    • none 图片没有适应策略,会在显示空间内显示图片,如果图片比显示空间大,则显示空间只会显示图片中间部分
      图片及icon - 图5
  • color和colorBlendMode 在图片绘制时可以对每一个像素进行颜色混合处理,color指定混合色,而colorBlendMode指定混合模式,下面是一个简单的示例:

    1. Image(
    2. image: AssetImage("images/avatar.png"),
    3. width: 100.0,
    4. color: Colors.blue,
    5. colorBlendMode: BlendMode.difference,
    6. );

    运行效果如图(彩色):
    图片及icon - 图6

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

    1. Image(
    2. image: AssetImage("images/avatar.png"),
    3. width: 100.0,
    4. height: 200.0,
    5. repeat: ImageRepeat.repeatY ,
    6. )
  • 运行后效果如图3-20所示:
    图片及icon - 图7

完整的示例代码如下:

  1. import 'package:flutter/material.dart';
  2. class ImageAndIconRoute extends StatelessWidget {
  3. @override
  4. Widget build(BuildContext context) {
  5. var img=AssetImage("imgs/avatar.png");
  6. return SingleChildScrollView(
  7. child: Column(
  8. children: <Image>[
  9. Image(
  10. image: img,
  11. height: 50.0,
  12. width: 100.0,
  13. fit: BoxFit.fill,
  14. ),
  15. Image(
  16. image: img,
  17. height: 50,
  18. width: 50.0,
  19. fit: BoxFit.contain,
  20. ),
  21. Image(
  22. image: img,
  23. width: 100.0,
  24. height: 50.0,
  25. fit: BoxFit.cover,
  26. ),
  27. Image(
  28. image: img,
  29. width: 100.0,
  30. height: 50.0,
  31. fit: BoxFit.fitWidth,
  32. ),
  33. Image(
  34. image: img,
  35. width: 100.0,
  36. height: 50.0,
  37. fit: BoxFit.fitHeight,
  38. ),
  39. Image(
  40. image: img,
  41. width: 100.0,
  42. height: 50.0,
  43. fit: BoxFit.scaleDown,
  44. ),
  45. Image(
  46. image: img,
  47. height: 50.0,
  48. width: 100.0,
  49. fit: BoxFit.none,
  50. ),
  51. Image(
  52. image: img,
  53. width: 100.0,
  54. color: Colors.blue,
  55. colorBlendMode: BlendMode.difference,
  56. fit: BoxFit.fill,
  57. ),
  58. Image(
  59. image: img,
  60. width: 100.0,
  61. height: 200.0,
  62. repeat: ImageRepeat.repeatY ,
  63. )
  64. ].map((e){
  65. return Row(
  66. children: <Widget>[
  67. Padding(
  68. padding: EdgeInsets.all(16.0),
  69. child: SizedBox(
  70. width: 100,
  71. child: e,
  72. ),
  73. ),
  74. Text(e.fit.toString())
  75. ],
  76. );
  77. }).toList()
  78. ),
  79. );
  80. }
  81. }

Image缓存

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

Flutter 设置图片铺满全屏

1. 设置 Container 的 width 和 height 为屏幕的宽高

  1. Container(
  2. width: MediaQuery.of(context).size.width, // 屏幕宽度
  3. height: MediaQuery.of(context).size.height, // 屏幕高度
  4. child: Image.asset(
  5. "assets/icons/bg.png",
  6. fit: BoxFit.cover,
  7. ),
  8. )

2. 层级组件 Stack

  1. Stack(
  2. children: [
  3. Positioned.fill(
  4. child: Image.asset(
  5. "assets/icons/bg.png",
  6. fit: BoxFit.cover,
  7. ),
  8. ),
  9. ],
  10. )

3.ConstrainedBox 的 constraints 属性设置图片充满全屏

  1. ConstrainedBox(
  2. constraints: BoxConstraints.expand(), // 自适应屏幕
  3. child: Image.asset(
  4. "assets/icons/bg.png",
  5. fit: BoxFit.cover,
  6. ),
  7. )

Icon 文字图片

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

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

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

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

使用Material Design字体图标

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

  1. flutter:
  2. uses-material-design: true

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

我们看一个简单的例子:

  1. String icons = "\uE914\uE000\uE90D";
  2. Text(icons,
  3. style: TextStyle(
  4. fontFamily: "MaterialIcons",
  5. fontSize: 24.0,
  6. color: Colors.green
  7. ),
  8. );

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

  1. Row(
  2. mainAxisAlignment: MainAxisAlignment.center,
  3. children: <Widget>[
  4. Icon(Icons.accessible,color: Colors.green,),
  5. Icon(Icons.error,color: Colors.green,),
  6. Icon(Icons.fingerprint,color: Colors.green,),
  7. ],
  8. )

Icons 类中包含了所有 Material Design 图标IconData静态变量定义。
Icon参数属性

  1. (new) Icon Icon(IconData icon,
  2. {Key key,
  3. double size,
  4. Color color,
  5. String semanticLabel,
  6. TextDirection textDirection})

使用自定义字体图标

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

  1. 导入字体图标文件;这一步和导入字体文件相同,假设我们的字体图标文件保存在项目根目录下,路径为”fonts/iconfont.ttf“,在 pubspec.yaml 中设置

    1. fonts:
    2. - family: myIcon #指定一个字体名
    3. fonts:
    4. - asset: fonts/iconfont.ttf
  2. 为了使用方便,我们定义一个 MyIcons 类,功能和 Icons 类一样:将字体文件中的所有图标都定义成静态变量: ```dart class MyIcons{ // book 图标 static const IconData book = const IconData( 0xe614, fontFamily: ‘myIcon’, matchTextDirection: true ); // 微信图标 static const IconData wechat = const IconData( 0xec7d,
    fontFamily: ‘myIcon’, matchTextDirection: true ); }

IconData参数属性,matchTextDirection:此图标是否应在从右向左的环境中自动镜像

IconData IconData(int codePoint, {String fontFamily, String fontPackage, bool matchTextDirection = false})

  1. 3. 使用
  2. ```dart
  3. Row(
  4. mainAxisAlignment: MainAxisAlignment.center,
  5. children: <Widget>[
  6. Icon(MyIcons.book,color: Colors.purple,),
  7. Icon(MyIcons.wechat,color: Colors.green,),
  8. ],
  9. )

图片及icon - 图9