diff --git a/packages/flet/lib/src/controls/image.dart b/packages/flet/lib/src/controls/image.dart index cb591eec7..b7e66119d 100644 --- a/packages/flet/lib/src/controls/image.dart +++ b/packages/flet/lib/src/controls/image.dart @@ -3,10 +3,11 @@ import 'package:flutter/material.dart'; import '../extensions/control.dart'; import '../models/control.dart'; import '../utils/borders.dart'; -import '../utils/box.dart'; +import '../utils/animations.dart'; import '../utils/colors.dart'; import '../utils/images.dart'; import '../utils/numbers.dart'; +import '../utils/time.dart'; import '../widgets/error.dart'; import 'base_controls.dart'; @@ -15,10 +16,7 @@ class ImageControl extends StatelessWidget { static const String svgTag = " xmlns=\"http://www.w3.org/2000/svg\""; - const ImageControl({ - super.key, - required this.control, - }); + const ImageControl({super.key, required this.control}); @override Widget build(BuildContext context) { @@ -29,25 +27,73 @@ class ImageControl extends StatelessWidget { return const ErrorControl("Image must have \"src\" specified."); } + final width = control.getDouble("width"); + final height = control.getDouble("height"); + final fit = control.getBoxFit("fit"); + final repeat = control.getImageRepeat("repeat", ImageRepeat.noRepeat)!; + final color = control.getColor("color", context); + final colorBlendMode = control.getBlendMode("color_blend_mode"); + final semanticsLabel = control.getString("semantics_label"); + final gaplessPlayback = control.getBool("gapless_playback"); + final excludeFromSemantics = + control.getBool("exclude_from_semantics", false)!; + final filterQuality = + control.getFilterQuality("filter_quality", FilterQuality.medium)!; + final cacheWidth = control.getInt("cache_width"); + final cacheHeight = control.getInt("cache_height"); + final antiAlias = control.getBool("anti_alias", false)!; + final errorContent = control.buildWidget("error_content"); + + // Optional placeholder shown while the image is loading. + Widget? placeholder; + final placeholderSrc = control.get("placeholder_src"); + if (placeholderSrc != null) { + placeholder = buildImage( + context: context, + src: placeholderSrc, + width: width, + height: height, + fit: control.getBoxFit("placeholder_fit", fit), + repeat: repeat, + color: color, + colorBlendMode: colorBlendMode, + semanticsLabel: semanticsLabel, + gaplessPlayback: gaplessPlayback, + excludeFromSemantics: excludeFromSemantics, + filterQuality: filterQuality, + cacheWidth: cacheWidth, + cacheHeight: cacheHeight, + antiAlias: antiAlias, + errorCtrl: errorContent, + ); + } + + final fadeConfig = ImageFadeConfig( + placeholder: placeholder, + fadeInDuration: control.getDuration("fade_in_duration"), + fadeOutDuration: control.getDuration("fade_out_duration"), + fadeInCurve: control.getCurve("fade_in_curve"), + fadeOutCurve: control.getCurve("fade_out_curve")); + Widget? image = buildImage( context: context, src: rawSrc, - width: control.getDouble("width"), - height: control.getDouble("height"), - cacheWidth: control.getInt("cache_width"), - cacheHeight: control.getInt("cache_height"), - antiAlias: control.getBool("anti_alias", false)!, - repeat: control.getImageRepeat("repeat", ImageRepeat.noRepeat)!, - fit: control.getBoxFit("fit"), - colorBlendMode: control.getBlendMode("color_blend_mode"), - color: control.getColor("color", context), - semanticsLabel: control.getString("semantics_label"), - gaplessPlayback: control.getBool("gapless_playback"), - excludeFromSemantics: control.getBool("exclude_from_semantics", false)!, - filterQuality: - control.getFilterQuality("filter_quality", FilterQuality.medium)!, + width: width, + height: height, + cacheWidth: cacheWidth, + cacheHeight: cacheHeight, + antiAlias: antiAlias, + repeat: repeat, + fit: fit, + colorBlendMode: colorBlendMode, + color: color, + semanticsLabel: semanticsLabel, + gaplessPlayback: gaplessPlayback, + excludeFromSemantics: excludeFromSemantics, + filterQuality: filterQuality, disabled: control.disabled, - errorCtrl: control.buildWidget("error_content"), + errorCtrl: errorContent, + fadeConfig: fadeConfig.enabled ? fadeConfig : null, ); return LayoutControl( control: control, diff --git a/packages/flet/lib/src/utils/images.dart b/packages/flet/lib/src/utils/images.dart index 404e04b81..c7e8bd18b 100644 --- a/packages/flet/lib/src/utils/images.dart +++ b/packages/flet/lib/src/utils/images.dart @@ -129,6 +129,7 @@ Widget buildImage({ bool excludeFromSemantics = false, FilterQuality filterQuality = FilterQuality.low, bool disabled = false, + ImageFadeConfig? fadeConfig, }) { const String svgTag = " xmlns=\"http://www.w3.org/2000/svg\""; @@ -143,31 +144,41 @@ Widget buildImage({ try { // SVG bytes if (arrayIndexOf(bytes, Uint8List.fromList(utf8.encode(svgTag))) != -1) { - return SvgPicture.memory(bytes, - width: width, - height: height, - excludeFromSemantics: excludeFromSemantics, - fit: fit ?? BoxFit.contain, - colorFilter: color != null - ? ColorFilter.mode(color, colorBlendMode ?? BlendMode.srcIn) - : null, - semanticsLabel: semanticsLabel); + return SvgPicture.memory( + bytes, + width: width, + height: height, + excludeFromSemantics: excludeFromSemantics, + fit: fit ?? BoxFit.contain, + colorFilter: color != null + ? ColorFilter.mode(color, colorBlendMode ?? BlendMode.srcIn) + : null, + semanticsLabel: semanticsLabel, + ); } else { // other image bytes - return Image.memory(bytes, - width: width, - height: height, - repeat: repeat, - fit: fit, - color: color, - cacheHeight: cacheHeight, - cacheWidth: cacheWidth, - filterQuality: filterQuality, - isAntiAlias: antiAlias, - colorBlendMode: colorBlendMode, - gaplessPlayback: gaplessPlayback ?? false, - excludeFromSemantics: excludeFromSemantics, - semanticLabel: semanticsLabel); + return Image.memory( + bytes, + width: width, + height: height, + repeat: repeat, + fit: fit, + color: color, + cacheHeight: cacheHeight, + cacheWidth: cacheWidth, + filterQuality: filterQuality, + isAntiAlias: antiAlias, + colorBlendMode: colorBlendMode, + gaplessPlayback: gaplessPlayback ?? false, + excludeFromSemantics: excludeFromSemantics, + semanticLabel: semanticsLabel, + frameBuilder: (BuildContext context, Widget child, int? frame, + bool wasSyncLoaded) => + fadeConfig != null && fadeConfig.enabled + ? fadeConfig.wrapFrame(child, frame, wasSyncLoaded, + width: width, height: height) + : child, + ); } } catch (ex) { return ErrorControl("Error decoding base64: ${ex.toString()}"); @@ -175,15 +186,20 @@ Widget buildImage({ } else if (resolvedSrc.hasUri) { var stringSrc = resolvedSrc.uri!; if (stringSrc.contains(svgTag)) { - return SvgPicture.memory(Uint8List.fromList(utf8.encode(stringSrc)), - width: width, - height: height, - fit: fit ?? BoxFit.contain, - excludeFromSemantics: excludeFromSemantics, - colorFilter: color != null - ? ColorFilter.mode(color, colorBlendMode ?? BlendMode.srcIn) - : null, - semanticsLabel: semanticsLabel); + return SvgPicture.memory( + Uint8List.fromList(utf8.encode(stringSrc)), + width: width, + height: height, + fit: fit ?? BoxFit.contain, + excludeFromSemantics: excludeFromSemantics, + colorFilter: color != null + ? ColorFilter.mode(color, colorBlendMode ?? BlendMode.srcIn) + : null, + semanticsLabel: semanticsLabel, + errorBuilder: errorCtrl != null + ? (context, error, stackTrace) => errorCtrl + : null, + ); } else { var assetSrc = FletBackend.of(context).getAssetSource(stringSrc); if (assetSrc.isFile) { @@ -214,10 +230,14 @@ Widget buildImage({ gaplessPlayback: gaplessPlayback ?? false, colorBlendMode: colorBlendMode, semanticLabel: semanticsLabel, + frameBuilder: (BuildContext context, Widget child, int? frame, + bool wasSyncLoaded) => + fadeConfig != null && fadeConfig.enabled + ? fadeConfig.wrapFrame(child, frame, wasSyncLoaded, + width: width, height: height) + : child, errorBuilder: errorCtrl != null - ? (context, error, stackTrace) { - return errorCtrl; - } + ? (context, error, stackTrace) => errorCtrl : null, ); } @@ -234,6 +254,9 @@ Widget buildImage({ ? ColorFilter.mode(color, colorBlendMode ?? BlendMode.srcIn) : null, semanticsLabel: semanticsLabel, + errorBuilder: errorCtrl != null + ? (context, error, stackTrace) => errorCtrl + : null, ); } else { // other image URL @@ -252,10 +275,14 @@ Widget buildImage({ gaplessPlayback: gaplessPlayback ?? false, colorBlendMode: colorBlendMode, semanticLabel: semanticsLabel, + frameBuilder: (BuildContext context, Widget child, int? frame, + bool wasSyncLoaded) => + fadeConfig != null && fadeConfig.enabled + ? fadeConfig.wrapFrame(child, frame, wasSyncLoaded, + width: width, height: height) + : child, errorBuilder: errorCtrl != null - ? (context, error, stackTrace) { - return errorCtrl; - } + ? (context, error, stackTrace) => errorCtrl : null, ); } @@ -266,6 +293,79 @@ Widget buildImage({ return const ErrorControl("A valid src value must be specified."); } +class ImageFadeConfig { + const ImageFadeConfig( + {this.placeholder, + this.fadeInDuration, + this.fadeOutDuration, + this.fadeInCurve, + this.fadeOutCurve}); + + final Widget? placeholder; + final Duration? fadeInDuration; + final Duration? fadeOutDuration; + final Curve? fadeInCurve; + final Curve? fadeOutCurve; + + /// Returns true if any fade-related option is set. + bool get enabled => + placeholder != null || + fadeInDuration != null || + fadeOutDuration != null || + fadeInCurve != null || + fadeOutCurve != null; + + Duration get resolvedFadeInDuration => + fadeInDuration ?? const Duration(milliseconds: 250); + + Duration get resolvedFadeOutDuration => + fadeOutDuration ?? const Duration(milliseconds: 150); + + Curve get resolvedFadeInCurve => fadeInCurve ?? Curves.easeInOut; + + Curve get resolvedFadeOutCurve => fadeOutCurve ?? Curves.easeOut; + + /// Wraps an [Image] frame with a placeholder-to-image fade transition. + /// + /// - Shows [placeholder] (or a transparent placeholder that preserves layout) + /// until the first frame is available. + /// - Fades the loaded image in using [fadeInDuration]/[fadeInCurve]. + /// - Fades the placeholder out using [fadeOutDuration]/[fadeOutCurve]. + Widget wrapFrame(Widget image, int? frame, bool wasSyncLoaded, + {double? width, double? height}) { + if (!enabled) { + return image; + } + + final isLoaded = frame != null || wasSyncLoaded; + final placeholderWidget = placeholder != null + ? SizedBox(width: width, height: height, child: placeholder) + // Defaults to an invisible version to preserve layout. + : SizedBox( + width: width, + height: height, + child: Opacity(opacity: 0, child: image)); + + return AnimatedSwitcher( + duration: isLoaded ? resolvedFadeInDuration : resolvedFadeOutDuration, + switchInCurve: resolvedFadeInCurve, + switchOutCurve: resolvedFadeOutCurve, + layoutBuilder: (Widget? current, List previousChildren) => + Stack( + alignment: Alignment.center, + children: [ + ...previousChildren, + if (current != null) current, + ], + ), + child: isLoaded + ? KeyedSubtree(key: const ValueKey("image-loaded"), child: image) + : KeyedSubtree( + key: const ValueKey("image-placeholder"), + child: placeholderWidget)); + } +} + class ResolvedAssetSource { const ResolvedAssetSource({this.bytes, this.uri, this.error}); diff --git a/sdk/python/examples/controls/image/fade_in.py b/sdk/python/examples/controls/image/fade_in.py new file mode 100644 index 000000000..b17af37d1 --- /dev/null +++ b/sdk/python/examples/controls/image/fade_in.py @@ -0,0 +1,20 @@ +import flet as ft + + +def main(page: ft.Page): + page.add( + ft.Image( + src="https://picsum.photos/320/200?random=2", + width=360, + height=220, + fit=ft.BoxFit.COVER, + # blue image + placeholder_src="", # noqa: E501 + fade_in_duration=400, + fade_out_duration=200, + ), + ) + + +if __name__ == "__main__": + ft.run(main) diff --git a/sdk/python/packages/flet/docs/controls/image.md b/sdk/python/packages/flet/docs/controls/image.md index 40b913231..86735d756 100644 --- a/sdk/python/packages/flet/docs/controls/image.md +++ b/sdk/python/packages/flet/docs/controls/image.md @@ -19,6 +19,12 @@ example_media: ../examples/controls/image/media {{ image(example_media + "/gallery.gif", width="80%") }} +### Fade-in images with a placeholder + +```python +--8<-- "{{ examples }}/fade_in.py" +``` + ### Displaying images from base64 strings and byte data ```python diff --git a/sdk/python/packages/flet/integration_tests/controls/core/golden/macos/image/placeholder_1.png b/sdk/python/packages/flet/integration_tests/controls/core/golden/macos/image/placeholder_1.png new file mode 100644 index 000000000..5dd519f7b Binary files /dev/null and b/sdk/python/packages/flet/integration_tests/controls/core/golden/macos/image/placeholder_1.png differ diff --git a/sdk/python/packages/flet/integration_tests/controls/core/golden/macos/image/placeholder_2.png b/sdk/python/packages/flet/integration_tests/controls/core/golden/macos/image/placeholder_2.png new file mode 100644 index 000000000..8c32183fd Binary files /dev/null and b/sdk/python/packages/flet/integration_tests/controls/core/golden/macos/image/placeholder_2.png differ diff --git a/sdk/python/packages/flet/integration_tests/controls/core/test_image.py b/sdk/python/packages/flet/integration_tests/controls/core/test_image.py index dfd8a6098..62d37cb85 100644 --- a/sdk/python/packages/flet/integration_tests/controls/core/test_image.py +++ b/sdk/python/packages/flet/integration_tests/controls/core/test_image.py @@ -99,3 +99,44 @@ async def test_src_bytes(flet_app: ftt.FletTestApp, request): pump_times=1, pump_duration=1000, ) + + +@pytest.mark.skip(reason="The test is flaky on CI") +@pytest.mark.asyncio(loop_scope="module") +async def test_placeholder_1(flet_app: ftt.FletTestApp, request): + await flet_app.assert_control_screenshot( + request.node.name, + ft.Image( + src="/minion.png", + width=100, + height=100, + fit=ft.BoxFit.CONTAIN, + placeholder_src=base64_image, + fade_in_duration=1000, + fade_out_duration=250, + fade_in_curve=ft.AnimationCurve.EASE_IN_OUT, + fade_out_curve=ft.AnimationCurve.EASE_OUT, + ), + pump_times=1, + pump_duration=50, + ) + + +@pytest.mark.asyncio(loop_scope="module") +async def test_placeholder_2(flet_app: ftt.FletTestApp, request): + await flet_app.assert_control_screenshot( + request.node.name, + ft.Image( + src="/minion.png", + width=100, + height=100, + fit=ft.BoxFit.CONTAIN, + placeholder_src=base64_image, + fade_in_duration=1000, + fade_out_duration=250, + fade_in_curve=ft.AnimationCurve.EASE_IN_OUT, + fade_out_curve=ft.AnimationCurve.EASE_OUT, + ), + pump_times=3, + pump_duration=1000, + ) diff --git a/sdk/python/packages/flet/src/flet/controls/core/image.py b/sdk/python/packages/flet/src/flet/controls/core/image.py index a2f1206a5..941663740 100644 --- a/sdk/python/packages/flet/src/flet/controls/core/image.py +++ b/sdk/python/packages/flet/src/flet/controls/core/image.py @@ -1,9 +1,11 @@ from typing import Optional, Union +from flet.controls.animation import AnimationCurve from flet.controls.base_control import control from flet.controls.border_radius import BorderRadiusValue from flet.controls.box import BoxFit, FilterQuality from flet.controls.control import Control +from flet.controls.duration import DurationValue from flet.controls.layout_control import LayoutControl from flet.controls.types import ( BlendMode, @@ -100,6 +102,56 @@ class Image(LayoutControl): The rendering quality of the image. """ + placeholder_src: Optional[Union[str, bytes]] = None + """ + A placeholder displayed while the image is loading. + + It can be one of the following: + - A URL or local [asset file](https://flet.dev/docs/cookbook/assets) path; + - A base64 string; + - Raw bytes. + + Note: + SVG sources are currently not supported as placeholders. If provided, + this property will be ignored and the [`src`][(c).] will be + displayed directly instead. + """ + + placeholder_fit: Optional[BoxFit] = None + """ + How to inscribe the placeholder into its space. + + Falls back to [`fit`][(c).] when omitted. + """ + + fade_in_duration: Optional[DurationValue] = None + """ + Duration of the fade-in animation when the image loads. + + Defaults to 250 milliseconds when any fade option is set. + """ + + fade_out_duration: Optional[DurationValue] = None + """ + Duration of the fade-out animation for the placeholder. + + Defaults to 150 milliseconds when any fade option is set. + """ + + fade_in_curve: Optional[AnimationCurve] = None + """ + Animation curve used for the fade-in transition. + + Defaults to `AnimationCurve.EASE_IN_OUT` when a fade is enabled. + """ + + fade_out_curve: Optional[AnimationCurve] = None + """ + Animation curve used for the fade-out transition. + + Defaults to `AnimationCurve.EASE_OUT` when a fade is enabled. + """ + cache_width: Optional[int] = None """ The size at which this image should be decoded.