
* Reland "Live image cache" (#51441) This reverts commit 419a2853a8e4ac846524afeb49b866d4cdaa7d9d. * Make evict safer
1163 lines
44 KiB
Dart
1163 lines
44 KiB
Dart
// Copyright 2014 The Flutter Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
// found in the LICENSE file.
|
|
|
|
import 'dart:async';
|
|
import 'dart:io' show File;
|
|
import 'dart:typed_data';
|
|
|
|
import 'package:flutter/foundation.dart';
|
|
import 'package:flutter/painting.dart';
|
|
import 'package:flutter/scheduler.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:flutter/semantics.dart';
|
|
|
|
import 'basic.dart';
|
|
import 'binding.dart';
|
|
import 'disposable_build_context.dart';
|
|
import 'framework.dart';
|
|
import 'localizations.dart';
|
|
import 'media_query.dart';
|
|
import 'scroll_aware_image_provider.dart';
|
|
import 'ticker_provider.dart';
|
|
|
|
export 'package:flutter/painting.dart' show
|
|
AssetImage,
|
|
ExactAssetImage,
|
|
FileImage,
|
|
FilterQuality,
|
|
ImageConfiguration,
|
|
ImageInfo,
|
|
ImageStream,
|
|
ImageProvider,
|
|
MemoryImage,
|
|
NetworkImage;
|
|
|
|
/// Creates an [ImageConfiguration] based on the given [BuildContext] (and
|
|
/// optionally size).
|
|
///
|
|
/// This is the object that must be passed to [BoxPainter.paint] and to
|
|
/// [ImageProvider.resolve].
|
|
///
|
|
/// If this is not called from a build method, then it should be reinvoked
|
|
/// whenever the dependencies change, e.g. by calling it from
|
|
/// [State.didChangeDependencies], so that any changes in the environment are
|
|
/// picked up (e.g. if the device pixel ratio changes).
|
|
///
|
|
/// See also:
|
|
///
|
|
/// * [ImageProvider], which has an example showing how this might be used.
|
|
ImageConfiguration createLocalImageConfiguration(BuildContext context, { Size size }) {
|
|
return ImageConfiguration(
|
|
bundle: DefaultAssetBundle.of(context),
|
|
devicePixelRatio: MediaQuery.of(context, nullOk: true)?.devicePixelRatio ?? 1.0,
|
|
locale: Localizations.localeOf(context, nullOk: true),
|
|
textDirection: Directionality.of(context),
|
|
size: size,
|
|
platform: defaultTargetPlatform,
|
|
);
|
|
}
|
|
|
|
/// Prefetches an image into the image cache.
|
|
///
|
|
/// Returns a [Future] that will complete when the first image yielded by the
|
|
/// [ImageProvider] is available or failed to load.
|
|
///
|
|
/// If the image is later used by an [Image] or [BoxDecoration] or [FadeInImage],
|
|
/// it will probably be loaded faster. The consumer of the image does not need
|
|
/// to use the same [ImageProvider] instance. The [ImageCache] will find the image
|
|
/// as long as both images share the same key, and the image is held by the
|
|
/// cache.
|
|
///
|
|
/// The cache may refuse to hold the image if it is disabled, the image is too
|
|
/// large, or some other criteria implemented by a custom [ImageCache]
|
|
/// implementation.
|
|
///
|
|
/// The [ImageCache] holds a reference to all images passed to [putIfAbsent] as
|
|
/// long as their [ImageStreamCompleter] has at least one listener. This method
|
|
/// will wait until the end of the frame after its future completes before
|
|
/// releasing its own listener. This gives callers a chance to listen to the
|
|
/// stream if necessary. A caller can determine if the image ended up in the
|
|
/// cache by calling [ImageProvider.obtainCacheStatus]. If it is only held as
|
|
/// [ImageCacheStatus.live], and the caller wishes to keep the resolved
|
|
/// image in memory, the caller should immediately call `provider.resolve` and
|
|
/// add a listener to the returned [ImageStream]. The image will remain pinned
|
|
/// in memory at least until the caller removes its listener from the stream,
|
|
/// even if it would not otherwise fit into the cache.
|
|
///
|
|
/// Callers should be cautious about pinning large images or a large number of
|
|
/// images in memory, as this can result in running out of memory and being
|
|
/// killed by the operating system. The lower the avilable physical memory, the
|
|
/// more susceptible callers will be to running into OOM issues. These issues
|
|
/// manifest as immediate process death, sometimes with no other error messages.
|
|
///
|
|
/// The [BuildContext] and [Size] are used to select an image configuration
|
|
/// (see [createLocalImageConfiguration]).
|
|
///
|
|
/// The `onError` argument can be used to manually handle errors while
|
|
/// pre-caching.
|
|
///
|
|
/// See also:
|
|
///
|
|
/// * [ImageCache], which holds images that may be reused.
|
|
Future<void> precacheImage(
|
|
ImageProvider provider,
|
|
BuildContext context, {
|
|
Size size,
|
|
ImageErrorListener onError,
|
|
}) {
|
|
final ImageConfiguration config = createLocalImageConfiguration(context, size: size);
|
|
final Completer<void> completer = Completer<void>();
|
|
final ImageStream stream = provider.resolve(config);
|
|
ImageStreamListener listener;
|
|
listener = ImageStreamListener(
|
|
(ImageInfo image, bool sync) {
|
|
if (!completer.isCompleted) {
|
|
completer.complete();
|
|
}
|
|
// Give callers until at least the end of the frame to subscribe to the
|
|
// image stream.
|
|
// See ImageCache._liveImages
|
|
SchedulerBinding.instance.addPostFrameCallback((Duration timeStamp) {
|
|
stream.removeListener(listener);
|
|
});
|
|
},
|
|
onError: (dynamic exception, StackTrace stackTrace) {
|
|
if (!completer.isCompleted) {
|
|
completer.complete();
|
|
}
|
|
stream.removeListener(listener);
|
|
if (onError != null) {
|
|
onError(exception, stackTrace);
|
|
} else {
|
|
FlutterError.reportError(FlutterErrorDetails(
|
|
context: ErrorDescription('image failed to precache'),
|
|
library: 'image resource service',
|
|
exception: exception,
|
|
stack: stackTrace,
|
|
silent: true,
|
|
));
|
|
}
|
|
},
|
|
);
|
|
stream.addListener(listener);
|
|
return completer.future;
|
|
}
|
|
|
|
/// Signature used by [Image.frameBuilder] to control the widget that will be
|
|
/// used when an [Image] is built.
|
|
///
|
|
/// The `child` argument contains the default image widget and is guaranteed to
|
|
/// be non-null. Typically, this builder will wrap the `child` widget in some
|
|
/// way and return the wrapped widget. If this builder returns `child` directly,
|
|
/// it will yield the same result as if [Image.frameBuilder] was null.
|
|
///
|
|
/// The `frame` argument specifies the index of the current image frame being
|
|
/// rendered. It will be null before the first image frame is ready, and zero
|
|
/// for the first image frame. For single-frame images, it will never be greater
|
|
/// than zero. For multi-frame images (such as animated GIFs), it will increase
|
|
/// by one every time a new image frame is shown (including when the image
|
|
/// animates in a loop).
|
|
///
|
|
/// The `wasSynchronouslyLoaded` argument specifies whether the image was
|
|
/// available synchronously (on the same
|
|
/// [rendering pipeline frame](rendering/RendererBinding/drawFrame.html) as the
|
|
/// `Image` widget itself was created) and thus able to be painted immediately.
|
|
/// If this is false, then there was one or more rendering pipeline frames where
|
|
/// the image wasn't yet available to be painted. For multi-frame images (such
|
|
/// as animated GIFs), the value of this argument will be the same for all image
|
|
/// frames. In other words, if the first image frame was available immediately,
|
|
/// then this argument will be true for all image frames.
|
|
///
|
|
/// This builder must not return null.
|
|
///
|
|
/// See also:
|
|
///
|
|
/// * [Image.frameBuilder], which makes use of this signature in the [Image]
|
|
/// widget.
|
|
typedef ImageFrameBuilder = Widget Function(
|
|
BuildContext context,
|
|
Widget child,
|
|
int frame,
|
|
bool wasSynchronouslyLoaded,
|
|
);
|
|
|
|
/// Signature used by [Image.loadingBuilder] to build a representation of the
|
|
/// image's loading progress.
|
|
///
|
|
/// This is useful for images that are incrementally loaded (e.g. over a local
|
|
/// file system or a network), and the application wishes to give the user an
|
|
/// indication of when the image will be displayed.
|
|
///
|
|
/// The `child` argument contains the default image widget and is guaranteed to
|
|
/// be non-null. Typically, this builder will wrap the `child` widget in some
|
|
/// way and return the wrapped widget. If this builder returns `child` directly,
|
|
/// it will yield the same result as if [Image.loadingBuilder] was null.
|
|
///
|
|
/// The `loadingProgress` argument contains the current progress towards loading
|
|
/// the image. This argument will be non-null while the image is loading, but it
|
|
/// will be null in the following cases:
|
|
///
|
|
/// * When the widget is first rendered before any bytes have been loaded.
|
|
/// * When an image has been fully loaded and is available to be painted.
|
|
///
|
|
/// If callers are implementing custom [ImageProvider] and [ImageStream]
|
|
/// instances (which is very rare), it's possible to produce image streams that
|
|
/// continue to fire image chunk events after an image frame has been loaded.
|
|
/// In such cases, the `child` parameter will represent the current
|
|
/// fully-loaded image frame.
|
|
///
|
|
/// This builder must not return null.
|
|
///
|
|
/// See also:
|
|
///
|
|
/// * [Image.loadingBuilder], which makes use of this signature in the [Image]
|
|
/// widget.
|
|
/// * [ImageChunkListener], a lower-level signature for listening to raw
|
|
/// [ImageChunkEvent]s.
|
|
typedef ImageLoadingBuilder = Widget Function(
|
|
BuildContext context,
|
|
Widget child,
|
|
ImageChunkEvent loadingProgress,
|
|
);
|
|
|
|
/// A widget that displays an image.
|
|
///
|
|
/// Several constructors are provided for the various ways that an image can be
|
|
/// specified:
|
|
///
|
|
/// * [new Image], for obtaining an image from an [ImageProvider].
|
|
/// * [new Image.asset], for obtaining an image from an [AssetBundle]
|
|
/// using a key.
|
|
/// * [new Image.network], for obtaining an image from a URL.
|
|
/// * [new Image.file], for obtaining an image from a [File].
|
|
/// * [new Image.memory], for obtaining an image from a [Uint8List].
|
|
///
|
|
/// The following image formats are supported: {@macro flutter.dart:ui.imageFormats}
|
|
///
|
|
/// To automatically perform pixel-density-aware asset resolution, specify the
|
|
/// image using an [AssetImage] and make sure that a [MaterialApp], [WidgetsApp],
|
|
/// or [MediaQuery] widget exists above the [Image] widget in the widget tree.
|
|
///
|
|
/// The image is painted using [paintImage], which describes the meanings of the
|
|
/// various fields on this class in more detail.
|
|
///
|
|
/// {@tool snippet}
|
|
/// The default constructor can be used with any [ImageProvider], such as a
|
|
/// [NetworkImage], to display an image from the internet.
|
|
///
|
|
/// 
|
|
///
|
|
/// ```dart
|
|
/// const Image(
|
|
/// image: NetworkImage('https://flutter.github.io/assets-for-api-docs/assets/widgets/owl.jpg'),
|
|
/// )
|
|
/// ```
|
|
/// {@end-tool}
|
|
///
|
|
/// {@tool snippet}
|
|
/// The [Image] Widget also provides several constructors to display different
|
|
/// types of images for convenience. In this example, use the [Image.network]
|
|
/// constructor to display an image from the internet.
|
|
///
|
|
/// 
|
|
///
|
|
/// ```dart
|
|
/// Image.network('https://flutter.github.io/assets-for-api-docs/assets/widgets/owl-2.jpg')
|
|
/// ```
|
|
/// {@end-tool}
|
|
///
|
|
/// The [Image.asset], [Image.network], [Image.file], and [Image.memory]
|
|
/// constructors allow a custom decode size to be specified through
|
|
/// [cacheWidth] and [cacheHeight] parameters. The engine will decode the
|
|
/// image to the specified size, which is primarily intended to reduce the
|
|
/// memory usage of [ImageCache].
|
|
///
|
|
/// In the case where a network image is used on the Web platform, the
|
|
/// [cacheWidth] and [cacheHeight] parameters are ignored as the Web engine
|
|
/// delegates image decoding of network images to the Web, which does not support
|
|
/// custom decode sizes.
|
|
///
|
|
/// See also:
|
|
///
|
|
/// * [Icon], which shows an image from a font.
|
|
/// * [new Ink.image], which is the preferred way to show an image in a
|
|
/// material application (especially if the image is in a [Material] and will
|
|
/// have an [InkWell] on top of it).
|
|
/// * [Image](dart-ui/Image-class.html), the class in the [dart:ui] library.
|
|
///
|
|
class Image extends StatefulWidget {
|
|
/// Creates a widget that displays an image.
|
|
///
|
|
/// To show an image from the network or from an asset bundle, consider using
|
|
/// [new Image.network] and [new Image.asset] respectively.
|
|
///
|
|
/// The [image], [alignment], [repeat], and [matchTextDirection] arguments
|
|
/// must not be null.
|
|
///
|
|
/// Either the [width] and [height] arguments should be specified, or the
|
|
/// widget should be placed in a context that sets tight layout constraints.
|
|
/// Otherwise, the image dimensions will change as the image is loaded, which
|
|
/// will result in ugly layout changes.
|
|
///
|
|
/// Use [filterQuality] to change the quality when scaling an image.
|
|
/// Use the [FilterQuality.low] quality setting to scale the image,
|
|
/// which corresponds to bilinear interpolation, rather than the default
|
|
/// [FilterQuality.none] which corresponds to nearest-neighbor.
|
|
///
|
|
/// If [excludeFromSemantics] is true, then [semanticLabel] will be ignored.
|
|
const Image({
|
|
Key key,
|
|
@required this.image,
|
|
this.frameBuilder,
|
|
this.loadingBuilder,
|
|
this.semanticLabel,
|
|
this.excludeFromSemantics = false,
|
|
this.width,
|
|
this.height,
|
|
this.color,
|
|
this.colorBlendMode,
|
|
this.fit,
|
|
this.alignment = Alignment.center,
|
|
this.repeat = ImageRepeat.noRepeat,
|
|
this.centerSlice,
|
|
this.matchTextDirection = false,
|
|
this.gaplessPlayback = false,
|
|
this.filterQuality = FilterQuality.low,
|
|
}) : assert(image != null),
|
|
assert(alignment != null),
|
|
assert(repeat != null),
|
|
assert(filterQuality != null),
|
|
assert(matchTextDirection != null),
|
|
super(key: key);
|
|
|
|
/// Creates a widget that displays an [ImageStream] obtained from the network.
|
|
///
|
|
/// The [src], [scale], and [repeat] arguments must not be null.
|
|
///
|
|
/// Either the [width] and [height] arguments should be specified, or the
|
|
/// widget should be placed in a context that sets tight layout constraints.
|
|
/// Otherwise, the image dimensions will change as the image is loaded, which
|
|
/// will result in ugly layout changes.
|
|
///
|
|
/// All network images are cached regardless of HTTP headers.
|
|
///
|
|
/// An optional [headers] argument can be used to send custom HTTP headers
|
|
/// with the image request.
|
|
///
|
|
/// Use [filterQuality] to change the quality when scaling an image.
|
|
/// Use the [FilterQuality.low] quality setting to scale the image,
|
|
/// which corresponds to bilinear interpolation, rather than the default
|
|
/// [FilterQuality.none] which corresponds to nearest-neighbor.
|
|
///
|
|
/// If [excludeFromSemantics] is true, then [semanticLabel] will be ignored.
|
|
///
|
|
/// If [cacheWidth] or [cacheHeight] are provided, it indicates to the
|
|
/// engine that the image should be decoded at the specified size. The image
|
|
/// will be rendered to the constraints of the layout or [width] and [height]
|
|
/// regardless of these parameters. These parameters are primarily intended
|
|
/// to reduce the memory usage of [ImageCache].
|
|
///
|
|
/// In the case where the network image is on the Web platform, the [cacheWidth]
|
|
/// and [cacheHeight] parameters are ignored as the web engine delegates
|
|
/// image decoding to the web which does not support custom decode sizes.
|
|
//
|
|
// TODO(garyq): We should eventually support custom decoding of network images
|
|
// on Web as well, see https://github.com/flutter/flutter/issues/42789.
|
|
Image.network(
|
|
String src, {
|
|
Key key,
|
|
double scale = 1.0,
|
|
this.frameBuilder,
|
|
this.loadingBuilder,
|
|
this.semanticLabel,
|
|
this.excludeFromSemantics = false,
|
|
this.width,
|
|
this.height,
|
|
this.color,
|
|
this.colorBlendMode,
|
|
this.fit,
|
|
this.alignment = Alignment.center,
|
|
this.repeat = ImageRepeat.noRepeat,
|
|
this.centerSlice,
|
|
this.matchTextDirection = false,
|
|
this.gaplessPlayback = false,
|
|
this.filterQuality = FilterQuality.low,
|
|
Map<String, String> headers,
|
|
int cacheWidth,
|
|
int cacheHeight,
|
|
}) : image = ResizeImage.resizeIfNeeded(cacheWidth, cacheHeight, NetworkImage(src, scale: scale, headers: headers)),
|
|
assert(alignment != null),
|
|
assert(repeat != null),
|
|
assert(matchTextDirection != null),
|
|
assert(cacheWidth == null || cacheWidth > 0),
|
|
assert(cacheHeight == null || cacheHeight > 0),
|
|
super(key: key);
|
|
|
|
/// Creates a widget that displays an [ImageStream] obtained from a [File].
|
|
///
|
|
/// The [file], [scale], and [repeat] arguments must not be null.
|
|
///
|
|
/// Either the [width] and [height] arguments should be specified, or the
|
|
/// widget should be placed in a context that sets tight layout constraints.
|
|
/// Otherwise, the image dimensions will change as the image is loaded, which
|
|
/// will result in ugly layout changes.
|
|
///
|
|
/// On Android, this may require the
|
|
/// `android.permission.READ_EXTERNAL_STORAGE` permission.
|
|
///
|
|
/// Use [filterQuality] to change the quality when scaling an image.
|
|
/// Use the [FilterQuality.low] quality setting to scale the image,
|
|
/// which corresponds to bilinear interpolation, rather than the default
|
|
/// [FilterQuality.none] which corresponds to nearest-neighbor.
|
|
///
|
|
/// If [excludeFromSemantics] is true, then [semanticLabel] will be ignored.
|
|
///
|
|
/// If [cacheWidth] or [cacheHeight] are provided, it indicates to the
|
|
/// engine that the image must be decoded at the specified size. The image
|
|
/// will be rendered to the constraints of the layout or [width] and [height]
|
|
/// regardless of these parameters. These parameters are primarily intended
|
|
/// to reduce the memory usage of [ImageCache].
|
|
Image.file(
|
|
File file, {
|
|
Key key,
|
|
double scale = 1.0,
|
|
this.frameBuilder,
|
|
this.semanticLabel,
|
|
this.excludeFromSemantics = false,
|
|
this.width,
|
|
this.height,
|
|
this.color,
|
|
this.colorBlendMode,
|
|
this.fit,
|
|
this.alignment = Alignment.center,
|
|
this.repeat = ImageRepeat.noRepeat,
|
|
this.centerSlice,
|
|
this.matchTextDirection = false,
|
|
this.gaplessPlayback = false,
|
|
this.filterQuality = FilterQuality.low,
|
|
int cacheWidth,
|
|
int cacheHeight,
|
|
}) : image = ResizeImage.resizeIfNeeded(cacheWidth, cacheHeight, FileImage(file, scale: scale)),
|
|
loadingBuilder = null,
|
|
assert(alignment != null),
|
|
assert(repeat != null),
|
|
assert(filterQuality != null),
|
|
assert(matchTextDirection != null),
|
|
assert(cacheWidth == null || cacheWidth > 0),
|
|
assert(cacheHeight == null || cacheHeight > 0),
|
|
super(key: key);
|
|
|
|
|
|
// TODO(ianh): Implement the following (see ../services/image_resolution.dart):
|
|
//
|
|
// * If [width] and [height] are both specified, and [scale] is not, then
|
|
// size-aware asset resolution will be attempted also, with the given
|
|
// dimensions interpreted as logical pixels.
|
|
//
|
|
// * If the images have platform, locale, or directionality variants, the
|
|
// current platform, locale, and directionality are taken into account
|
|
// during asset resolution as well.
|
|
/// Creates a widget that displays an [ImageStream] obtained from an asset
|
|
/// bundle. The key for the image is given by the `name` argument.
|
|
///
|
|
/// The `package` argument must be non-null when displaying an image from a
|
|
/// package and null otherwise. See the `Assets in packages` section for
|
|
/// details.
|
|
///
|
|
/// If the `bundle` argument is omitted or null, then the
|
|
/// [DefaultAssetBundle] will be used.
|
|
///
|
|
/// By default, the pixel-density-aware asset resolution will be attempted. In
|
|
/// addition:
|
|
///
|
|
/// * If the `scale` argument is provided and is not null, then the exact
|
|
/// asset specified will be used. To display an image variant with a specific
|
|
/// density, the exact path must be provided (e.g. `images/2x/cat.png`).
|
|
///
|
|
/// If [excludeFromSemantics] is true, then [semanticLabel] will be ignored.
|
|
///
|
|
/// If [cacheWidth] or [cacheHeight] are provided, it indicates to the
|
|
/// engine that the image must be decoded at the specified size. The image
|
|
/// will be rendered to the constraints of the layout or [width] and [height]
|
|
/// regardless of these parameters. These parameters are primarily intended
|
|
/// to reduce the memory usage of [ImageCache].
|
|
///
|
|
/// The [name] and [repeat] arguments must not be null.
|
|
///
|
|
/// Either the [width] and [height] arguments should be specified, or the
|
|
/// widget should be placed in a context that sets tight layout constraints.
|
|
/// Otherwise, the image dimensions will change as the image is loaded, which
|
|
/// will result in ugly layout changes.
|
|
///
|
|
/// Use [filterQuality] to change the quality when scaling an image.
|
|
/// Use the [FilterQuality.low] quality setting to scale the image,
|
|
/// which corresponds to bilinear interpolation, rather than the default
|
|
/// [FilterQuality.none] which corresponds to nearest-neighbor.
|
|
///
|
|
/// {@tool snippet}
|
|
///
|
|
/// Suppose that the project's `pubspec.yaml` file contains the following:
|
|
///
|
|
/// ```yaml
|
|
/// flutter:
|
|
/// assets:
|
|
/// - images/cat.png
|
|
/// - images/2x/cat.png
|
|
/// - images/3.5x/cat.png
|
|
/// ```
|
|
/// {@end-tool}
|
|
///
|
|
/// On a screen with a device pixel ratio of 2.0, the following widget would
|
|
/// render the `images/2x/cat.png` file:
|
|
///
|
|
/// ```dart
|
|
/// Image.asset('images/cat.png')
|
|
/// ```
|
|
///
|
|
/// This corresponds to the file that is in the project's `images/2x/`
|
|
/// directory with the name `cat.png` (the paths are relative to the
|
|
/// `pubspec.yaml` file).
|
|
///
|
|
/// On a device with a 4.0 device pixel ratio, the `images/3.5x/cat.png` asset
|
|
/// would be used. On a device with a 1.0 device pixel ratio, the
|
|
/// `images/cat.png` resource would be used.
|
|
///
|
|
/// The `images/cat.png` image can be omitted from disk (though it must still
|
|
/// be present in the manifest). If it is omitted, then on a device with a 1.0
|
|
/// device pixel ratio, the `images/2x/cat.png` image would be used instead.
|
|
///
|
|
///
|
|
/// ## Assets in packages
|
|
///
|
|
/// To create the widget with an asset from a package, the [package] argument
|
|
/// must be provided. For instance, suppose a package called `my_icons` has
|
|
/// `icons/heart.png` .
|
|
///
|
|
/// {@tool snippet}
|
|
/// Then to display the image, use:
|
|
///
|
|
/// ```dart
|
|
/// Image.asset('icons/heart.png', package: 'my_icons')
|
|
/// ```
|
|
/// {@end-tool}
|
|
///
|
|
/// Assets used by the package itself should also be displayed using the
|
|
/// [package] argument as above.
|
|
///
|
|
/// If the desired asset is specified in the `pubspec.yaml` of the package, it
|
|
/// is bundled automatically with the app. In particular, assets used by the
|
|
/// package itself must be specified in its `pubspec.yaml`.
|
|
///
|
|
/// A package can also choose to have assets in its 'lib/' folder that are not
|
|
/// specified in its `pubspec.yaml`. In this case for those images to be
|
|
/// bundled, the app has to specify which ones to include. For instance a
|
|
/// package named `fancy_backgrounds` could have:
|
|
///
|
|
/// ```
|
|
/// lib/backgrounds/background1.png
|
|
/// lib/backgrounds/background2.png
|
|
/// lib/backgrounds/background3.png
|
|
/// ```
|
|
///
|
|
/// To include, say the first image, the `pubspec.yaml` of the app should
|
|
/// specify it in the assets section:
|
|
///
|
|
/// ```yaml
|
|
/// assets:
|
|
/// - packages/fancy_backgrounds/backgrounds/background1.png
|
|
/// ```
|
|
///
|
|
/// The `lib/` is implied, so it should not be included in the asset path.
|
|
///
|
|
///
|
|
/// See also:
|
|
///
|
|
/// * [AssetImage], which is used to implement the behavior when the scale is
|
|
/// omitted.
|
|
/// * [ExactAssetImage], which is used to implement the behavior when the
|
|
/// scale is present.
|
|
/// * <https://flutter.dev/assets-and-images/>, an introduction to assets in
|
|
/// Flutter.
|
|
Image.asset(
|
|
String name, {
|
|
Key key,
|
|
AssetBundle bundle,
|
|
this.frameBuilder,
|
|
this.semanticLabel,
|
|
this.excludeFromSemantics = false,
|
|
double scale,
|
|
this.width,
|
|
this.height,
|
|
this.color,
|
|
this.colorBlendMode,
|
|
this.fit,
|
|
this.alignment = Alignment.center,
|
|
this.repeat = ImageRepeat.noRepeat,
|
|
this.centerSlice,
|
|
this.matchTextDirection = false,
|
|
this.gaplessPlayback = false,
|
|
String package,
|
|
this.filterQuality = FilterQuality.low,
|
|
int cacheWidth,
|
|
int cacheHeight,
|
|
}) : image = ResizeImage.resizeIfNeeded(cacheWidth, cacheHeight, scale != null
|
|
? ExactAssetImage(name, bundle: bundle, scale: scale, package: package)
|
|
: AssetImage(name, bundle: bundle, package: package)
|
|
),
|
|
loadingBuilder = null,
|
|
assert(alignment != null),
|
|
assert(repeat != null),
|
|
assert(matchTextDirection != null),
|
|
assert(cacheWidth == null || cacheWidth > 0),
|
|
assert(cacheHeight == null || cacheHeight > 0),
|
|
super(key: key);
|
|
|
|
/// Creates a widget that displays an [ImageStream] obtained from a [Uint8List].
|
|
///
|
|
/// The [bytes], [scale], and [repeat] arguments must not be null.
|
|
///
|
|
/// This only accepts compressed image formats (e.g. PNG). Uncompressed
|
|
/// formats like rawRgba (the default format of [ui.Image.toByteData]) will
|
|
/// lead to exceptions.
|
|
///
|
|
/// Either the [width] and [height] arguments should be specified, or the
|
|
/// widget should be placed in a context that sets tight layout constraints.
|
|
/// Otherwise, the image dimensions will change as the image is loaded, which
|
|
/// will result in ugly layout changes.
|
|
///
|
|
/// Use [filterQuality] to change the quality when scaling an image.
|
|
/// Use the [FilterQuality.low] quality setting to scale the image,
|
|
/// which corresponds to bilinear interpolation, rather than the default
|
|
/// [FilterQuality.none] which corresponds to nearest-neighbor.
|
|
///
|
|
/// If [excludeFromSemantics] is true, then [semanticLabel] will be ignored.
|
|
///
|
|
/// If [cacheWidth] or [cacheHeight] are provided, it indicates to the
|
|
/// engine that the image must be decoded at the specified size. The image
|
|
/// will be rendered to the constraints of the layout or [width] and [height]
|
|
/// regardless of these parameters. These parameters are primarily intended
|
|
/// to reduce the memory usage of [ImageCache].
|
|
Image.memory(
|
|
Uint8List bytes, {
|
|
Key key,
|
|
double scale = 1.0,
|
|
this.frameBuilder,
|
|
this.semanticLabel,
|
|
this.excludeFromSemantics = false,
|
|
this.width,
|
|
this.height,
|
|
this.color,
|
|
this.colorBlendMode,
|
|
this.fit,
|
|
this.alignment = Alignment.center,
|
|
this.repeat = ImageRepeat.noRepeat,
|
|
this.centerSlice,
|
|
this.matchTextDirection = false,
|
|
this.gaplessPlayback = false,
|
|
this.filterQuality = FilterQuality.low,
|
|
int cacheWidth,
|
|
int cacheHeight,
|
|
}) : image = ResizeImage.resizeIfNeeded(cacheWidth, cacheHeight, MemoryImage(bytes, scale: scale)),
|
|
loadingBuilder = null,
|
|
assert(alignment != null),
|
|
assert(repeat != null),
|
|
assert(matchTextDirection != null),
|
|
assert(cacheWidth == null || cacheWidth > 0),
|
|
assert(cacheHeight == null || cacheHeight > 0),
|
|
super(key: key);
|
|
|
|
/// The image to display.
|
|
final ImageProvider image;
|
|
|
|
/// A builder function responsible for creating the widget that represents
|
|
/// this image.
|
|
///
|
|
/// If this is null, this widget will display an image that is painted as
|
|
/// soon as the first image frame is available (and will appear to "pop" in
|
|
/// if it becomes available asynchronously). Callers might use this builder to
|
|
/// add effects to the image (such as fading the image in when it becomes
|
|
/// available) or to display a placeholder widget while the image is loading.
|
|
///
|
|
/// To have finer-grained control over the way that an image's loading
|
|
/// progress is communicated to the user, see [loadingBuilder].
|
|
///
|
|
/// ## Chaining with [loadingBuilder]
|
|
///
|
|
/// If a [loadingBuilder] has _also_ been specified for an image, the two
|
|
/// builders will be chained together: the _result_ of this builder will
|
|
/// be passed as the `child` argument to the [loadingBuilder]. For example,
|
|
/// consider the following builders used in conjunction:
|
|
///
|
|
/// {@template flutter.widgets.image.chainedBuildersExample}
|
|
/// ```dart
|
|
/// Image(
|
|
/// ...
|
|
/// frameBuilder: (BuildContext context, Widget child, int frame, bool wasSynchronouslyLoaded) {
|
|
/// return Padding(
|
|
/// padding: EdgeInsets.all(8.0),
|
|
/// child: child,
|
|
/// );
|
|
/// },
|
|
/// loadingBuilder: (BuildContext context, Widget child, ImageChunkEvent loadingProgress) {
|
|
/// return Center(child: child);
|
|
/// },
|
|
/// )
|
|
/// ```
|
|
///
|
|
/// In this example, the widget hierarchy will contain the following:
|
|
///
|
|
/// ```dart
|
|
/// Center(
|
|
/// Padding(
|
|
/// padding: EdgeInsets.all(8.0),
|
|
/// child: <image>,
|
|
/// ),
|
|
/// )
|
|
/// ```
|
|
/// {@endtemplate}
|
|
///
|
|
/// {@tool dartpad --template=stateless_widget_material}
|
|
///
|
|
/// The following sample demonstrates how to use this builder to implement an
|
|
/// image that fades in once it's been loaded.
|
|
///
|
|
/// This sample contains a limited subset of the functionality that the
|
|
/// [FadeInImage] widget provides out of the box.
|
|
///
|
|
/// ```dart
|
|
/// @override
|
|
/// Widget build(BuildContext context) {
|
|
/// return DecoratedBox(
|
|
/// decoration: BoxDecoration(
|
|
/// color: Colors.white,
|
|
/// border: Border.all(),
|
|
/// borderRadius: BorderRadius.circular(20),
|
|
/// ),
|
|
/// child: Image.network(
|
|
/// 'https://flutter.github.io/assets-for-api-docs/assets/widgets/puffin.jpg',
|
|
/// frameBuilder: (BuildContext context, Widget child, int frame, bool wasSynchronouslyLoaded) {
|
|
/// if (wasSynchronouslyLoaded) {
|
|
/// return child;
|
|
/// }
|
|
/// return AnimatedOpacity(
|
|
/// child: child,
|
|
/// opacity: frame == null ? 0 : 1,
|
|
/// duration: const Duration(seconds: 1),
|
|
/// curve: Curves.easeOut,
|
|
/// );
|
|
/// },
|
|
/// ),
|
|
/// );
|
|
/// }
|
|
/// ```
|
|
/// {@end-tool}
|
|
final ImageFrameBuilder frameBuilder;
|
|
|
|
/// A builder that specifies the widget to display to the user while an image
|
|
/// is still loading.
|
|
///
|
|
/// If this is null, and the image is loaded incrementally (e.g. over a
|
|
/// network), the user will receive no indication of the progress as the
|
|
/// bytes of the image are loaded.
|
|
///
|
|
/// For more information on how to interpret the arguments that are passed to
|
|
/// this builder, see the documentation on [ImageLoadingBuilder].
|
|
///
|
|
/// ## Performance implications
|
|
///
|
|
/// If a [loadingBuilder] is specified for an image, the [Image] widget is
|
|
/// likely to be rebuilt on every
|
|
/// [rendering pipeline frame](rendering/RendererBinding/drawFrame.html) until
|
|
/// the image has loaded. This is useful for cases such as displaying a loading
|
|
/// progress indicator, but for simpler cases such as displaying a placeholder
|
|
/// widget that doesn't depend on the loading progress (e.g. static "loading"
|
|
/// text), [frameBuilder] will likely work and not incur as much cost.
|
|
///
|
|
/// ## Chaining with [frameBuilder]
|
|
///
|
|
/// If a [frameBuilder] has _also_ been specified for an image, the two
|
|
/// builders will be chained together: the `child` argument to this
|
|
/// builder will contain the _result_ of the [frameBuilder]. For example,
|
|
/// consider the following builders used in conjunction:
|
|
///
|
|
/// {@macro flutter.widgets.image.chainedBuildersExample}
|
|
///
|
|
/// {@tool dartpad --template=stateless_widget_material}
|
|
///
|
|
/// The following sample uses [loadingBuilder] to show a
|
|
/// [CircularProgressIndicator] while an image loads over the network.
|
|
///
|
|
/// ```dart
|
|
/// Widget build(BuildContext context) {
|
|
/// return DecoratedBox(
|
|
/// decoration: BoxDecoration(
|
|
/// color: Colors.white,
|
|
/// border: Border.all(),
|
|
/// borderRadius: BorderRadius.circular(20),
|
|
/// ),
|
|
/// child: Image.network(
|
|
/// 'https://example.com/image.jpg',
|
|
/// loadingBuilder: (BuildContext context, Widget child, ImageChunkEvent loadingProgress) {
|
|
/// if (loadingProgress == null)
|
|
/// return child;
|
|
/// return Center(
|
|
/// child: CircularProgressIndicator(
|
|
/// value: loadingProgress.expectedTotalBytes != null
|
|
/// ? loadingProgress.cumulativeBytesLoaded / loadingProgress.expectedTotalBytes
|
|
/// : null,
|
|
/// ),
|
|
/// );
|
|
/// },
|
|
/// ),
|
|
/// );
|
|
/// }
|
|
/// ```
|
|
/// {@end-tool}
|
|
///
|
|
/// Run against a real-world image on a slow network, the previous example
|
|
/// renders the following loading progress indicator while the image loads
|
|
/// before rendering the completed image.
|
|
///
|
|
/// {@animation 400 400 https://flutter.github.io/assets-for-api-docs/assets/widgets/loading_progress_image.mp4}
|
|
final ImageLoadingBuilder loadingBuilder;
|
|
|
|
/// If non-null, require the image to have this width.
|
|
///
|
|
/// If null, the image will pick a size that best preserves its intrinsic
|
|
/// aspect ratio.
|
|
///
|
|
/// It is strongly recommended that either both the [width] and the [height]
|
|
/// be specified, or that the widget be placed in a context that sets tight
|
|
/// layout constraints, so that the image does not change size as it loads.
|
|
/// Consider using [fit] to adapt the image's rendering to fit the given width
|
|
/// and height if the exact image dimensions are not known in advance.
|
|
final double width;
|
|
|
|
/// If non-null, require the image to have this height.
|
|
///
|
|
/// If null, the image will pick a size that best preserves its intrinsic
|
|
/// aspect ratio.
|
|
///
|
|
/// It is strongly recommended that either both the [width] and the [height]
|
|
/// be specified, or that the widget be placed in a context that sets tight
|
|
/// layout constraints, so that the image does not change size as it loads.
|
|
/// Consider using [fit] to adapt the image's rendering to fit the given width
|
|
/// and height if the exact image dimensions are not known in advance.
|
|
final double height;
|
|
|
|
/// If non-null, this color is blended with each image pixel using [colorBlendMode].
|
|
final Color color;
|
|
|
|
/// Used to set the [FilterQuality] of the image.
|
|
///
|
|
/// Use the [FilterQuality.low] quality setting to scale the image with
|
|
/// bilinear interpolation, or the [FilterQuality.none] which corresponds
|
|
/// to nearest-neighbor.
|
|
final FilterQuality filterQuality;
|
|
|
|
/// Used to combine [color] with this image.
|
|
///
|
|
/// The default is [BlendMode.srcIn]. In terms of the blend mode, [color] is
|
|
/// the source and this image is the destination.
|
|
///
|
|
/// See also:
|
|
///
|
|
/// * [BlendMode], which includes an illustration of the effect of each blend mode.
|
|
final BlendMode colorBlendMode;
|
|
|
|
/// How to inscribe the image into the space allocated during layout.
|
|
///
|
|
/// The default varies based on the other fields. See the discussion at
|
|
/// [paintImage].
|
|
final BoxFit fit;
|
|
|
|
/// How to align the image within its bounds.
|
|
///
|
|
/// The alignment aligns the given position in the image to the given position
|
|
/// in the layout bounds. For example, an [Alignment] alignment of (-1.0,
|
|
/// -1.0) aligns the image to the top-left corner of its layout bounds, while an
|
|
/// [Alignment] alignment of (1.0, 1.0) aligns the bottom right of the
|
|
/// image with the bottom right corner of its layout bounds. Similarly, an
|
|
/// alignment of (0.0, 1.0) aligns the bottom middle of the image with the
|
|
/// middle of the bottom edge of its layout bounds.
|
|
///
|
|
/// To display a subpart of an image, consider using a [CustomPainter] and
|
|
/// [Canvas.drawImageRect].
|
|
///
|
|
/// If the [alignment] is [TextDirection]-dependent (i.e. if it is a
|
|
/// [AlignmentDirectional]), then an ambient [Directionality] widget
|
|
/// must be in scope.
|
|
///
|
|
/// Defaults to [Alignment.center].
|
|
///
|
|
/// See also:
|
|
///
|
|
/// * [Alignment], a class with convenient constants typically used to
|
|
/// specify an [AlignmentGeometry].
|
|
/// * [AlignmentDirectional], like [Alignment] for specifying alignments
|
|
/// relative to text direction.
|
|
final AlignmentGeometry alignment;
|
|
|
|
/// How to paint any portions of the layout bounds not covered by the image.
|
|
final ImageRepeat repeat;
|
|
|
|
/// The center slice for a nine-patch image.
|
|
///
|
|
/// The region of the image inside the center slice will be stretched both
|
|
/// horizontally and vertically to fit the image into its destination. The
|
|
/// region of the image above and below the center slice will be stretched
|
|
/// only horizontally and the region of the image to the left and right of
|
|
/// the center slice will be stretched only vertically.
|
|
final Rect centerSlice;
|
|
|
|
/// Whether to paint the image in the direction of the [TextDirection].
|
|
///
|
|
/// If this is true, then in [TextDirection.ltr] contexts, the image will be
|
|
/// drawn with its origin in the top left (the "normal" painting direction for
|
|
/// images); and in [TextDirection.rtl] contexts, the image will be drawn with
|
|
/// a scaling factor of -1 in the horizontal direction so that the origin is
|
|
/// in the top right.
|
|
///
|
|
/// This is occasionally used with images in right-to-left environments, for
|
|
/// images that were designed for left-to-right locales. Be careful, when
|
|
/// using this, to not flip images with integral shadows, text, or other
|
|
/// effects that will look incorrect when flipped.
|
|
///
|
|
/// If this is true, there must be an ambient [Directionality] widget in
|
|
/// scope.
|
|
final bool matchTextDirection;
|
|
|
|
/// Whether to continue showing the old image (true), or briefly show nothing
|
|
/// (false), when the image provider changes.
|
|
final bool gaplessPlayback;
|
|
|
|
/// A Semantic description of the image.
|
|
///
|
|
/// Used to provide a description of the image to TalkBack on Android, and
|
|
/// VoiceOver on iOS.
|
|
final String semanticLabel;
|
|
|
|
/// Whether to exclude this image from semantics.
|
|
///
|
|
/// Useful for images which do not contribute meaningful information to an
|
|
/// application.
|
|
final bool excludeFromSemantics;
|
|
|
|
@override
|
|
_ImageState createState() => _ImageState();
|
|
|
|
@override
|
|
void debugFillProperties(DiagnosticPropertiesBuilder properties) {
|
|
super.debugFillProperties(properties);
|
|
properties.add(DiagnosticsProperty<ImageProvider>('image', image));
|
|
properties.add(DiagnosticsProperty<Function>('frameBuilder', frameBuilder));
|
|
properties.add(DiagnosticsProperty<Function>('loadingBuilder', loadingBuilder));
|
|
properties.add(DoubleProperty('width', width, defaultValue: null));
|
|
properties.add(DoubleProperty('height', height, defaultValue: null));
|
|
properties.add(ColorProperty('color', color, defaultValue: null));
|
|
properties.add(EnumProperty<BlendMode>('colorBlendMode', colorBlendMode, defaultValue: null));
|
|
properties.add(EnumProperty<BoxFit>('fit', fit, defaultValue: null));
|
|
properties.add(DiagnosticsProperty<AlignmentGeometry>('alignment', alignment, defaultValue: null));
|
|
properties.add(EnumProperty<ImageRepeat>('repeat', repeat, defaultValue: ImageRepeat.noRepeat));
|
|
properties.add(DiagnosticsProperty<Rect>('centerSlice', centerSlice, defaultValue: null));
|
|
properties.add(FlagProperty('matchTextDirection', value: matchTextDirection, ifTrue: 'match text direction'));
|
|
properties.add(StringProperty('semanticLabel', semanticLabel, defaultValue: null));
|
|
properties.add(DiagnosticsProperty<bool>('this.excludeFromSemantics', excludeFromSemantics));
|
|
properties.add(EnumProperty<FilterQuality>('filterQuality', filterQuality));
|
|
}
|
|
}
|
|
|
|
class _ImageState extends State<Image> with WidgetsBindingObserver {
|
|
ImageStream _imageStream;
|
|
ImageInfo _imageInfo;
|
|
ImageChunkEvent _loadingProgress;
|
|
bool _isListeningToStream = false;
|
|
bool _invertColors;
|
|
int _frameNumber;
|
|
bool _wasSynchronouslyLoaded;
|
|
DisposableBuildContext<State<Image>> _scrollAwareContext;
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
WidgetsBinding.instance.addObserver(this);
|
|
_scrollAwareContext = DisposableBuildContext<State<Image>>(this);
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
assert(_imageStream != null);
|
|
WidgetsBinding.instance.removeObserver(this);
|
|
_stopListeningToStream();
|
|
_scrollAwareContext.dispose();
|
|
super.dispose();
|
|
}
|
|
|
|
@override
|
|
void didChangeDependencies() {
|
|
_updateInvertColors();
|
|
_resolveImage();
|
|
|
|
if (TickerMode.of(context))
|
|
_listenToStream();
|
|
else
|
|
_stopListeningToStream();
|
|
|
|
super.didChangeDependencies();
|
|
}
|
|
|
|
@override
|
|
void didUpdateWidget(Image oldWidget) {
|
|
super.didUpdateWidget(oldWidget);
|
|
if (_isListeningToStream &&
|
|
(widget.loadingBuilder == null) != (oldWidget.loadingBuilder == null)) {
|
|
_imageStream.removeListener(_getListener(oldWidget.loadingBuilder));
|
|
_imageStream.addListener(_getListener());
|
|
}
|
|
if (widget.image != oldWidget.image)
|
|
_resolveImage();
|
|
}
|
|
|
|
@override
|
|
void didChangeAccessibilityFeatures() {
|
|
super.didChangeAccessibilityFeatures();
|
|
setState(() {
|
|
_updateInvertColors();
|
|
});
|
|
}
|
|
|
|
@override
|
|
void reassemble() {
|
|
_resolveImage(); // in case the image cache was flushed
|
|
super.reassemble();
|
|
}
|
|
|
|
void _updateInvertColors() {
|
|
_invertColors = MediaQuery.of(context, nullOk: true)?.invertColors
|
|
?? SemanticsBinding.instance.accessibilityFeatures.invertColors;
|
|
}
|
|
|
|
void _resolveImage() {
|
|
final ScrollAwareImageProvider provider = ScrollAwareImageProvider<dynamic>(
|
|
context: _scrollAwareContext,
|
|
imageProvider: widget.image,
|
|
);
|
|
final ImageStream newStream =
|
|
provider.resolve(createLocalImageConfiguration(
|
|
context,
|
|
size: widget.width != null && widget.height != null ? Size(widget.width, widget.height) : null,
|
|
));
|
|
assert(newStream != null);
|
|
_updateSourceStream(newStream);
|
|
}
|
|
|
|
ImageStreamListener _getListener([ImageLoadingBuilder loadingBuilder]) {
|
|
loadingBuilder ??= widget.loadingBuilder;
|
|
return ImageStreamListener(
|
|
_handleImageFrame,
|
|
onChunk: loadingBuilder == null ? null : _handleImageChunk,
|
|
);
|
|
}
|
|
|
|
void _handleImageFrame(ImageInfo imageInfo, bool synchronousCall) {
|
|
setState(() {
|
|
_imageInfo = imageInfo;
|
|
_loadingProgress = null;
|
|
_frameNumber = _frameNumber == null ? 0 : _frameNumber + 1;
|
|
_wasSynchronouslyLoaded |= synchronousCall;
|
|
});
|
|
}
|
|
|
|
void _handleImageChunk(ImageChunkEvent event) {
|
|
assert(widget.loadingBuilder != null);
|
|
setState(() {
|
|
_loadingProgress = event;
|
|
});
|
|
}
|
|
|
|
// Updates _imageStream to newStream, and moves the stream listener
|
|
// registration from the old stream to the new stream (if a listener was
|
|
// registered).
|
|
void _updateSourceStream(ImageStream newStream) {
|
|
if (_imageStream?.key == newStream?.key)
|
|
return;
|
|
|
|
if (_isListeningToStream)
|
|
_imageStream.removeListener(_getListener());
|
|
|
|
if (!widget.gaplessPlayback)
|
|
setState(() { _imageInfo = null; });
|
|
|
|
setState(() {
|
|
_loadingProgress = null;
|
|
_frameNumber = null;
|
|
_wasSynchronouslyLoaded = false;
|
|
});
|
|
|
|
_imageStream = newStream;
|
|
if (_isListeningToStream)
|
|
_imageStream.addListener(_getListener());
|
|
}
|
|
|
|
void _listenToStream() {
|
|
if (_isListeningToStream)
|
|
return;
|
|
_imageStream.addListener(_getListener());
|
|
_isListeningToStream = true;
|
|
}
|
|
|
|
void _stopListeningToStream() {
|
|
if (!_isListeningToStream)
|
|
return;
|
|
_imageStream.removeListener(_getListener());
|
|
_isListeningToStream = false;
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
Widget result = RawImage(
|
|
image: _imageInfo?.image,
|
|
width: widget.width,
|
|
height: widget.height,
|
|
scale: _imageInfo?.scale ?? 1.0,
|
|
color: widget.color,
|
|
colorBlendMode: widget.colorBlendMode,
|
|
fit: widget.fit,
|
|
alignment: widget.alignment,
|
|
repeat: widget.repeat,
|
|
centerSlice: widget.centerSlice,
|
|
matchTextDirection: widget.matchTextDirection,
|
|
invertColors: _invertColors,
|
|
filterQuality: widget.filterQuality,
|
|
);
|
|
|
|
if (!widget.excludeFromSemantics) {
|
|
result = Semantics(
|
|
container: widget.semanticLabel != null,
|
|
image: true,
|
|
label: widget.semanticLabel ?? '',
|
|
child: result,
|
|
);
|
|
}
|
|
|
|
if (widget.frameBuilder != null)
|
|
result = widget.frameBuilder(context, result, _frameNumber, _wasSynchronouslyLoaded);
|
|
|
|
if (widget.loadingBuilder != null)
|
|
result = widget.loadingBuilder(context, result, _loadingProgress);
|
|
|
|
return result;
|
|
}
|
|
|
|
@override
|
|
void debugFillProperties(DiagnosticPropertiesBuilder description) {
|
|
super.debugFillProperties(description);
|
|
description.add(DiagnosticsProperty<ImageStream>('stream', _imageStream));
|
|
description.add(DiagnosticsProperty<ImageInfo>('pixels', _imageInfo));
|
|
description.add(DiagnosticsProperty<ImageChunkEvent>('loadingProgress', _loadingProgress));
|
|
description.add(DiagnosticsProperty<int>('frameNumber', _frameNumber));
|
|
description.add(DiagnosticsProperty<bool>('wasSynchronouslyLoaded', _wasSynchronouslyLoaded));
|
|
}
|
|
}
|