精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

在 Flutter 中實現最佳 UX 性能的 12 個圖像技巧和最佳實踐

開發 前端
僅僅展示圖片是不夠的!您還應該為用戶提供他們需要的最佳體驗!因此,我強烈建議您創建自己的自定義圖片 widget,將它們隨意組合并自由使用!

Image widget 是 Flutter 中最常用的 widget 之一,但我相信我們沒有充分利用它的功能,僅僅顯示一個圖片是不夠的,你還應該給用戶他們需要的最佳體驗!

在這篇文章中,我將談論一些圖像技巧和最佳實踐,以獲得更好的性能和用戶體驗。

這些技巧是:

  • 1.使用 WebP 而不是 JPG/PNG
  • 2.設置寬度和高度以保留 UI 空間
  • 3.降低圖片的顯示分辨率以減少內存使用
  • 4.預加載/預緩存您的圖像,以便即時加載圖像
  • 5.加載時顯示進度指示器
  • 6.加載時顯示進度百分比指示器
  • 7.加載時顯示閃爍效果,以提高用戶體驗
  • 8.顯示 blurhash 作為占位符
  • 9.使用漸變效果來提高用戶體驗
  • 10.緩存圖像以減少網絡使用并提高性能
  • 11.注意非經常性成本
  • 12.在失敗時顯示重試按鈕
  • 結語

1.使用 WebP 而不是 JPG/PNG

WebP 是下一代圖像格式,它比 PNG 和 JPEG 小約 25%,并且比其他格式快。

這意味著,你的應用程序將使用更少的內存,構建速度更快。

這里有一些基準:

圖片圖片

圖片圖片

Image.asset(
  // 'image.jpg',
  'image.webp', // PREFER
);

2.設置寬度和高度以保留 UI 空間

它可以防止應用程序出現布局偏移

圖片圖片

之前——之后

Image.network(
  imageUrl,
  width: 200,
  height: 150,
);

3.降低圖片的顯示分辨率以減少內存使用

圖片圖片

圖片圖片

你的圖片可能會導致設備內存膨脹,這是因為,盡管它們在 UI 中占據相對較小的一部分,Flutter 還是會以全分辨率渲染它們,從而消耗大量內存。

為了避免這種問題,可以使用 cacheWidth 或 cacheHeight 參數對指定大小的圖像進行解碼。

此外,我們可以使用 Flutter 開發者工具輕松檢測超大圖像。

如果圖像過大,它會反轉圖像,使其顛倒。

注意!緩存大小不應該小于小部件的大小,否則,由于分辨率低,它看起來像素化!

Image.network(
  imageUrl,
  cacheWidth: 100,
  cacheHeight: 150,
);

4.預加載/預緩存您的圖像,以便即時加載圖像

如果你在顯示圖像之前緩存它們,Flutter 將跳過構建的處理步驟并立即顯示它們。

圖片圖片

class MyImage extends StatefulWidget {
  const MyImage({super.key});

  @override
  State createState() => _MyImageState();
}

class _MyImageState extends State {
  late final Image myImage;

  @override
  void initState() {
    super.initState();
    myImage = Image.asset('path');
  }

  @override
  void didChangeDependencies() {
    super.didChangeDependencies();
    precacheImage(myImage.image, context);
  }

  @override
  Widget build(BuildContext context) {
    return myImage;
  }
}

5.加載時顯示進度指示器

突然彈出圖像不是預期的行為,用戶可能會因為網絡連接不足而錯過圖像并向下滾動,或者可能在屏幕上看到一些空白,等等。我們應該始終通知用戶圖像正在加載。

圖片圖片

return Image.network(
  imageUrl,
  loadingBuilder: (_, child, event) {
    if (event == null) return child;
    return const Center(child: CircularProgressIndicator());
  },
);

6.加載時顯示進度百分比指示器

我們也可以顯示進度百分比,而不是無限加載,這樣對用戶來說更有用。

圖片圖片

return Image.network(
  imageUrl,
  loadingBuilder: (_, child, event) {
    if (event == null) return child;
    return Center(
      child: CircularProgressIndicator(
        value: event.cumulativeBytesLoaded / (event.expectedTotalBytes ?? 0),
      ),
    );
  },
);

7.加載時顯示閃爍效果,以提高用戶體驗

顯示進度條是好的,但并不是最好的選擇,顯示閃爍效果(Shimmer)要比顯示進度條好得多。

圖片圖片

return Image.network(
  imageUrl,
  height: 200,
  width: 350,
  loadingBuilder: (_, child, event) {
    if (event == null) return child;
    return const Shimmer(
      height: 200,
      width: 350,
     );
  }
);

// Most Basic Shimmer
class Shimmer extends StatefulWidget {
  const Shimmer({
    super.key,
    this.width,
    this.height,
    this.minOpacity = 0.015,
    this.maxOpacity = 0.15,
    this.borderRadius = const BorderRadius.all(Radius.circular(4)),
    this.child,
  });

  final double? width;
  final double? height;
  final double minOpacity;
  final double maxOpacity;
  final BorderRadius? borderRadius;
  final Widget? child;

  @override
  State createState() => _ShimmerState();
}

class _ShimmerState extends State with SingleTickerProviderStateMixin {
  late final AnimationController controller;

  @override
  void initState() {
    super.initState();
    controller = AnimationController(
      vsync: this,
      duration: const Duration(seconds: 1),
      lowerBound: widget.minOpacity,
      upperBound: widget.maxOpacity,
    )..repeat(reverse: true);
  }

  @override
  void dispose() {
    controller.dispose();
    super.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return RepaintBoundary(
      child: FadeTransition(
        opacity: controller,
        child: Container(
          width: widget.width,
          height: widget.height,
          decoration: BoxDecoration(
            color: Colors.black,
            borderRadius: widget.borderRadius,
          ),
          child: widget.child,
        ),
      ),
    );
  }
}

我創建了一個簡單的 shimmer 小部件,但你可以從官方文檔中學習如何創建高級版本的 shimmer 效果。

?

官方文檔:創建 shimmer 加載效果(https://docs.flutter.dev/cookbook/effects/shimmer-loading)

8.顯示 blurhash 作為占位符

為了改善用戶體驗,可以使用哈希代碼顯示圖像的模糊版本,而不是在圖像加載時顯示空白的灰色區域。

圖片圖片

https://blurha.sh/

return const SizedBox(
  width: 350,
  height: 200,
  child: BlurHash(
    hash: hashCode,
    imageFit: BoxFit.cover,
    image: imageUrl,
  ),
);

9.使用漸變效果來提高用戶體驗

默認情況下,圖片加載后立即顯示,這對我們的視覺體驗來說非常糟糕,為了改善這一點,我們可以用一個小的漸入動畫來顯示它們。

我們可以使用 FadeInImage 來實現這個功能。

它需要字節或資源作為占位符,在這個例子中,我將使用 transparent_image 包來獲取透明圖像字節。

我們還可以使用 cached_network_image 包來實現這一點,以及更多。

return FadeInImage.memoryNetwork(
  image: imageUrl,
  placeholder: kTransparentImage,
);

// 或者
return CachedNetworkImage(
  imageUrl: imageUrl,
);

10.緩存圖像以減少網絡使用并提高性能

為了避免每次下載相同的圖片,我們可以緩存第一次下載的圖片并重復使用,為了實現這一點,我們可以創建自己的緩存機制,或者我們可以直接使用 cached_network_image。

它緩存網絡圖像,默認情況下自動顯示它們的淡入效果,并提供了更多的圖像控制。

11.注意非經常性成本

Image widget 沒有 const 構造函數,雖然這在大多數情況下都不是問題,但我們可以通過將它包裝在自定義 widget 中來修復它。

它不僅可以讓我們的應用程序更具性能,而且我們還可以根據我們的意愿定制小部件,例如,我們可以為每個圖像小部件創建一個全局解決方案,而不是每次都處理 error/loading 情況。

enum _ImageType { asset, network }

class AppImage extends StatelessWidget {
  const AppImage.asset(
    this.image, {
    super.key,
  }) : type = _ImageType.asset;

  const AppImage.network(
    this.image, {
    super.key,
  }) : type = _ImageType.network;

  final String image;
  final _ImageType type;

  @override
  Widget build(BuildContext context) {
    const errorWidget = Icon(Icons.error);
    return switch (type) {
      _ImageType.asset => Image.asset(
          image,
          errorBuilder: (_, __, ___) => errorWidget,
        ),
      _ImageType.network => Image.network(
          image,
          errorBuilder: (_, __, ___) => errorWidget,
        ),
    };
  }
}


/// 使用
const AppImage.asset(''), // OK
const Image.asset(''), // 錯誤!!Image 沒有const構造函數

// 正如您所知,擁有 const 的小部件非常重要以獲得更好的性能。

12.在失敗時顯示重試按鈕

有時候由于網絡連接不好或其他原因,圖片無法第一次加載,顯示錯誤消息是好的,但這還不夠,如果我們想把應用的 UX 提升到另一個層次,我們應該讓用戶重新加載圖片,并繼續使用應用,而不會遇到任何麻煩。

圖片圖片

class MyImage extends StatefulWidget {
  const MyImage({super.key});

  @override
  State createState() => _MyImageState();
}

class _MyImageState extends State {
  int attempt = 0;

  @override
  Widget build(BuildContext context) {
    return CachedNetworkImage(
      imageUrl: imageUrl,
      cacheKey: '$attempt',
      height: 200,
      width: 250,
      fit: BoxFit.cover,
      errorWidget: (_, __, ___) {
        return RetryWidget(
          height: 200,
          width: 250,
          onTap: () => setState(() => attempt++),
        );
      },
    );
  }
}

// Just a basic retry button
class RetryWidget extends StatelessWidget {
  const RetryWidget({
    super.key,
    required this.height,
    required this.width,
    required this.onTap,
  });

  final double? height;
  final double? width;
  final void Function() onTap;

  @override
  Widget build(BuildContext context) {
    return GestureDetector(
      onTap: onTap,
      child: Container(
        height: height,
        width: width,
        alignment: Alignment.center,
        decoration: const BoxDecoration(color: Colors.black12),
        child: const Column(
          mainAxisSize: MainAxisSize.min,
          children: [
            Icon(Icons.image_not_supported, size: 20),
            SizedBox(height: 12),
            Padding(
              padding: EdgeInsets.symmetric(horizontal: 12),
              child: Text(
                "Image couldn't load, tap here to retry",
                textAlign: TextAlign.center,
                style: TextStyle(fontSize: 14, color: Colors.black),
              ),
            ),
          ],
        ),
      ),
    );
  }
}

結語

僅僅展示圖片是不夠的!您還應該為用戶提供他們需要的最佳體驗!因此,我強烈建議您創建自己的自定義圖片 widget,將它們隨意組合并自由使用!

原文:https://medium.com/itnext/12-image-tips-and-best-practices-for-the-best-ux-performance-in-flutter-e7a1b2b1da2a&strip=0&vwsrc=1&referer=medium-parser

責任編輯:武曉燕 來源: 獨立開發者張張
相關推薦

2023-07-07 12:26:39

攜程開發

2023-09-13 08:00:00

JavaScript循環語句

2013-12-31 09:26:31

JavaScript技巧

2023-10-16 11:27:43

2011-08-11 09:45:25

2023-07-25 11:22:31

2010-02-04 11:55:27

ibmdwDB2

2012-03-29 09:35:32

WEBCSS

2012-03-19 09:55:38

CSS

2011-12-31 10:18:33

響應設計

2010-07-06 09:07:09

2009-11-26 10:31:55

配置IPS最佳實踐

2022-05-30 07:48:11

DevOps測試策略

2013-05-17 11:43:55

主數據數據管理

2010-11-15 09:13:22

云計算開發測試

2020-09-17 06:00:21

Git

2009-07-01 17:44:46

Servlet和JSP

2018-01-12 14:37:34

Java代碼實踐

2020-06-10 09:57:23

Kubernetes日志容器

2017-03-30 22:16:21

DevOpsIT應用程序
點贊
收藏

51CTO技術棧公眾號

日本女优一区| 玛雅亚洲电影| 成人免费观看男女羞羞视频| 欧美孕妇性xx| 可以免费看av的网址| 国产成人免费视频网站视频社区| 亚洲免费在线电影| 久久国产日韩欧美| 在线观看免费视频一区| 欧美色综合网| 亚洲午夜av久久乱码| 视频区 图片区 小说区| 一区二区精品伦理...| 国产精品免费久久| 国产精品视频免费一区| 久久午夜鲁丝片| 亚洲午夜激情在线| 亚洲视频777| 红桃视频一区二区三区免费| jizz内谢中国亚洲jizz| 一区二区在线观看视频在线观看| 久久综合久久久| 精品人妻伦一区二区三区久久 | 国产精品77777| 日韩av快播网址| 久久午夜无码鲁丝片| 欧美理论电影大全| 亚洲精品97久久| 色黄视频免费看| 免费一区二区三区四区| 精品久久久久久中文字幕大豆网| 日日噜噜噜夜夜爽爽| 免费人成黄页在线观看忧物| 国产精品香蕉一区二区三区| 国产精品久久久久久久久久| 91porny在线| 欧美精品色网| 久久久国产精品亚洲一区| a毛片毛片av永久免费| 国产成人澳门| 精品久久久三级丝袜| 国产人妻精品久久久久野外| 欧美美女福利视频| 欧美综合天天夜夜久久| 男人日女人bb视频| 九色porny自拍视频在线播放| 亚洲乱码精品一二三四区日韩在线| 亚洲国产欧美一区二区三区不卡| 天堂av在线资源| 不卡的看片网站| 99精品在线直播| 国产丝袜视频在线观看| 国产一区二区三区蝌蚪| 91在线|亚洲| 国产激情无套内精对白视频| 国内精品免费在线观看| 亚洲一区二区三| 国产精品女同一区二区| 麻豆久久久久久| 国产精品视频色| 中文字字幕在线观看| 热久久一区二区| 国产精品视频自在线| 国产九色91回来了| 麻豆国产精品官网| 国产欧美一区二区三区四区| 亚洲综合精品视频| 国产麻豆9l精品三级站| 999精品视频一区二区三区| 99久久久久成人国产免费| 国产伦精品一区二区三区免费迷 | 久久精品欧美日韩精品| 四虎影视永久免费在线观看一区二区三区| 免费播放片a高清在线观看| 久久久久久久久久久久久久久99 | 欧美午夜视频一区二区| 激情网站五月天| 久久久久久一区二区三区四区别墅| 欧美三级欧美一级| 91pony九色| 成人在线视频中文字幕| 亚洲国产精品人久久电影| 丝袜美腿中文字幕| 日韩精品电影| 久久国产精品亚洲| 久草视频在线资源站| 国产情侣一区| 国产欧美日韩免费| 亚洲精品国产精品国| 91啦中文在线观看| 亚洲一区二区在线观| 日韩专区av| 日韩欧美高清在线视频| 色片在线免费观看| 成人免费在线电影网| 亚洲欧美国产日韩天堂区| 久久一级免费视频| 黄色免费成人| 国产精品热视频| 蜜桃av中文字幕| 日本一区二区三区四区在线视频| 成年人三级视频| 电影一区二区三| 91精品在线一区二区| 亚洲av片不卡无码久久| 我不卡伦不卡影院| 欧美最猛性xxxxx(亚洲精品)| 亚洲天堂网视频| 99久久综合99久久综合网站| 伊人色综合影院| 91在线超碰| 欧美美女一区二区在线观看| 免费啪视频在线观看| jiujiure精品视频播放| 久久久这里只有精品视频| 亚洲综合一区中| 成人国产亚洲欧美成人综合网| 天天久久人人| 性xxxxfreexxxxx欧美丶| 91精品国产一区二区三区蜜臀| av无码av天天av天天爽| 欧美日韩国产成人精品| 国产欧美日韩精品在线观看| 视频一区二区三区国产| 亚洲免费观看高清完整| 日韩高清第一页| 国产欧美日韩一区二区三区四区 | 欧美一区 二区 三区| 欧美va在线播放| 波兰性xxxxx极品hd| 老司机免费视频久久| 国内精品**久久毛片app| av在线网址观看| 欧美日韩免费一区二区三区视频 | 综合在线观看色| 男人的天堂日韩| 亚洲黄色录像| 97在线视频一区| 少妇喷水在线观看| 亚洲一区国产视频| 中国男女全黄大片| 欧美a级一区| 91日本在线观看| 男人的天堂在线视频免费观看| 欧洲人成人精品| 人妻av无码一区二区三区| 久久xxxx精品视频| 麻豆蜜桃91| 一区二区三区电影大全| 亚洲精品一区中文| 制服.丝袜.亚洲.中文.综合懂色| 波多野结衣中文字幕一区二区三区| 日本a在线天堂| 中文字幕av一区二区三区四区| 欧美精品一二区| 亚洲爱爱综合网| 亚洲综合精品久久| 国产精品一区二区无码对白| 极品少妇一区二区三区| 国产嫩草一区二区三区在线观看| 黄污视频在线观看| 日韩av在线免费| 精品人妻一区二区色欲产成人| 久久综合色之久久综合| 热久久精品国产| 最近国产精品视频| 国产精品久久久久久av下载红粉 | www.污污视频| 综合视频在线| 成人在线视频网址| 精品极品在线| 国产小视频91| 99久久亚洲精品日本无码| 亚洲男人的天堂在线观看| 曰本三级日本三级日本三级| 在线看片一区| 日本不卡在线播放| 亚洲欧洲一二区| 欧美国产日韩中文字幕在线| 日夜干在线视频| 欧美撒尿777hd撒尿| 国产精品 欧美激情| 成人黄色av电影| 一区二区三区国产免费| 一区二区三区午夜探花| 国产精品久久久久久久天堂第1集| 女海盗2成人h版中文字幕| 中文字幕久久久av一区| 精品女同一区二区三区| 色综合久久久网| avtt天堂在线| 久久婷婷成人综合色| 一本色道久久亚洲综合精品蜜桃| 欧美精品日本| 日本午夜精品一区二区| 日韩视频在线直播| 国产ts人妖一区二区三区| 国产在线观看a视频| 亚洲国产欧美一区二区丝袜黑人 | 精品伦理精品一区| 日日夜夜狠狠操| 一区二区在线观看免费视频播放| 中文字幕av观看| 国产又黄又大久久| 日韩视频第二页| 午夜精品国产| 欧美亚洲爱爱另类综合| 亚洲福利合集| 国产精品久久久久久亚洲调教| 天堂av中文在线| 色黄久久久久久| 青青视频在线观| 日韩免费在线观看| 伊人网av在线| 色婷婷av一区二区三区大白胸| 免费又黄又爽又色的视频| 国产欧美日韩不卡免费| 国产精品300页| 国产盗摄视频一区二区三区| 蜜臀av免费观看| 亚洲一区二区三区高清| 亚洲午夜视频在线观看| 亚洲成人激情小说| 麻豆精品新av中文字幕| 精品99在线视频| 好看不卡的中文字幕| 最新中文字幕久久| 秋霞欧美视频| 欧美日韩综合网| 欧美美女在线直播| 国产一区在线观| av日韩精品| 91精品国产综合久久久久久丝袜| 素人一区二区三区| 国产成人精品999| 大胆人体一区| 日韩美女激情视频| 中文日产幕无线码一区二区| 高清亚洲成在人网站天堂| 中文在线免费| 日韩视频亚洲视频| 日本在线天堂| 日韩性生活视频| lutube成人福利在线观看| 亚洲欧美中文日韩在线| 日本私人网站在线观看| 精品亚洲一区二区三区| 午夜视频在线播放| 日韩高清不卡av| 色香蕉在线视频| 亚洲加勒比久久88色综合 | 亚洲区中文字幕| 日韩精品一二| 亚洲天堂免费观看| 韩国中文字幕2020精品| 一级做a爰片久久毛片美女图片| 毛片免费在线| 中文字幕不卡在线视频极品| 1769在线观看| 久久精品亚洲94久久精品| 久久综合之合合综合久久| 久久中国妇女中文字幕| 日本在线视频网址| 97国产精品人人爽人人做| 免费h视频在线观看| 日本欧美精品在线| 巨胸喷奶水www久久久免费动漫| 国产精品精品视频一区二区三区| 成人国产精品| 91久久精品视频| jazzjazz国产精品久久| 国内外成人免费视频| 曰本一区二区三区视频| 亚洲精品一区二| 午夜久久黄色| 久草资源站在线观看| 首页亚洲欧美制服丝腿| 视频在线观看免费高清| 国产电影一区在线| 亚洲少妇18p| 中文字幕成人在线观看| 久久r这里只有精品| 欧美日韩国产麻豆| 伊人精品在线视频| 精品国产一区a| 国产一区二区三区福利| 久久中文字幕在线| 手机在线理论片| 国产一区二区色| 黄色美女久久久| 色婷婷精品国产一区二区三区| 91精品国产调教在线观看| 亚洲美免无码中文字幕在线| 人人超碰91尤物精品国产| 人妻精品久久久久中文字幕69| 97久久久精品综合88久久| 欧美老女人性生活视频| 亚洲一区二区三区国产| 91麻豆精品在线| 精品国产制服丝袜高跟| 欧美3p视频在线观看| 久久天天躁日日躁| 久久爱91午夜羞羞| 91中文字幕一区| 神马电影久久| 亚洲熟妇无码av在线播放| 日本欧美一区二区| 少妇被狂c下部羞羞漫画| 国产精品乱码一区二三区小蝌蚪| 日本熟妇一区二区| 3d动漫精品啪啪一区二区竹菊| 青梅竹马是消防员在线| 欧美另类在线播放| 国产精品第一| 你懂的视频在线一区二区| 亚洲综合五月| 午夜在线观看av| 91原创在线视频| 国产亚洲精品久久777777| 欧美精品久久一区| www.久久热.com| 日本免费一区二区三区视频观看| 成人看片爽爽爽| 国产精品国产三级国产专区51| 男男视频亚洲欧美| 老牛影视av老牛影视av| 亚洲成人av电影在线| 精品美女www爽爽爽视频| 日韩色av导航| 国产成人精选| 欧美一区1区三区3区公司 | 激情婷婷综合| 男女午夜激情视频| 972aa.com艺术欧美| 久久一级黄色片| 日韩欧美一级精品久久| 久草中文在线| 国产欧美在线看| 成人激情免费视频| 嫩草影院国产精品| 国产亚洲美州欧州综合国| 日本熟女毛茸茸| 亚洲女成人图区| 老司机2019福利精品视频导航| 国产精品推荐精品| 亚洲激情av| 天堂www中文在线资源| 亚洲一本大道在线| 日韩一级在线播放| 国内揄拍国内精品少妇国语| 97品白浆高清久久久久久| 日本大胆人体视频| 国产91在线看| 国产福利拍拍拍| 日韩毛片在线看| 亚洲va中文在线播放免费| 日韩中文字幕一区二区| 日本午夜一区二区| 日韩黄色中文字幕| 欧美另类一区二区三区| wwwav在线| 国产成人av一区二区三区| 国产一区亚洲| 亚洲成人av免费在线观看| 日韩欧美精品在线观看| 国产中文字幕在线| 国产日韩欧美在线观看| 亚洲精品一区二区妖精| 爱情岛论坛亚洲自拍| 亚洲高清在线精品| 日本一级在线观看| 国产精品综合久久久| 91精品高清| 中国免费黄色片| 日本高清无吗v一区| 欧美jizzhd欧美| 国产伦精品一区二区三区在线| 国产欧美在线| 三级影片在线观看| 精品国产91乱码一区二区三区| 欧美日韩在线观看首页| 日韩中文字幕av在线| 国产精品99久久久久久宅男| www.av视频在线观看| 亚洲偷熟乱区亚洲香蕉av| 国产精品xnxxcom| 欧美精品久久久久久久自慰| 国产日韩亚洲欧美综合| aaa级黄色片| 欧美一级免费看| 亚洲九九在线| 国产精品无码一区二区三区| 欧美日韩亚洲高清一区二区| 三级资源在线| 日韩影院一区| 成人18精品视频| 一级特黄aaa大片在线观看| 性欧美xxxx交| 久久久五月天| 91国模少妇一区二区三区|