2023-02-01 17:59:34 +01:00
|
|
|
import 'package:flutter/widgets.dart';
|
|
|
|
|
|
|
|
import '../photo_view.dart';
|
|
|
|
import 'core/photo_view_core.dart';
|
|
|
|
import 'photo_view_default_widgets.dart';
|
|
|
|
import 'utils/photo_view_utils.dart';
|
|
|
|
|
|
|
|
class ImageWrapper extends StatefulWidget {
|
|
|
|
const ImageWrapper({
|
2024-01-27 17:14:32 +01:00
|
|
|
super.key,
|
2023-02-01 17:59:34 +01:00
|
|
|
required this.imageProvider,
|
|
|
|
required this.loadingBuilder,
|
|
|
|
required this.backgroundDecoration,
|
|
|
|
required this.gaplessPlayback,
|
|
|
|
required this.heroAttributes,
|
|
|
|
required this.scaleStateChangedCallback,
|
|
|
|
required this.enableRotation,
|
|
|
|
required this.controller,
|
|
|
|
required this.scaleStateController,
|
|
|
|
required this.maxScale,
|
|
|
|
required this.minScale,
|
|
|
|
required this.initialScale,
|
|
|
|
required this.basePosition,
|
|
|
|
required this.scaleStateCycle,
|
|
|
|
required this.onTapUp,
|
|
|
|
required this.onTapDown,
|
|
|
|
required this.onDragStart,
|
|
|
|
required this.onDragEnd,
|
|
|
|
required this.onDragUpdate,
|
|
|
|
required this.onScaleEnd,
|
2024-05-02 17:37:39 +02:00
|
|
|
required this.onLongPressStart,
|
2023-02-01 17:59:34 +01:00
|
|
|
required this.outerSize,
|
|
|
|
required this.gestureDetectorBehavior,
|
|
|
|
required this.tightMode,
|
|
|
|
required this.filterQuality,
|
|
|
|
required this.disableGestures,
|
|
|
|
required this.errorBuilder,
|
|
|
|
required this.enablePanAlways,
|
2023-09-18 05:57:05 +02:00
|
|
|
required this.index,
|
2024-01-27 17:14:32 +01:00
|
|
|
});
|
2023-02-01 17:59:34 +01:00
|
|
|
|
|
|
|
final ImageProvider imageProvider;
|
|
|
|
final LoadingBuilder? loadingBuilder;
|
|
|
|
final ImageErrorWidgetBuilder? errorBuilder;
|
|
|
|
final BoxDecoration backgroundDecoration;
|
|
|
|
final bool gaplessPlayback;
|
|
|
|
final PhotoViewHeroAttributes? heroAttributes;
|
|
|
|
final ValueChanged<PhotoViewScaleState>? scaleStateChangedCallback;
|
|
|
|
final bool enableRotation;
|
|
|
|
final dynamic maxScale;
|
|
|
|
final dynamic minScale;
|
|
|
|
final dynamic initialScale;
|
|
|
|
final PhotoViewControllerBase controller;
|
|
|
|
final PhotoViewScaleStateController scaleStateController;
|
|
|
|
final Alignment? basePosition;
|
|
|
|
final ScaleStateCycle? scaleStateCycle;
|
|
|
|
final PhotoViewImageTapUpCallback? onTapUp;
|
|
|
|
final PhotoViewImageTapDownCallback? onTapDown;
|
|
|
|
final PhotoViewImageDragStartCallback? onDragStart;
|
|
|
|
final PhotoViewImageDragEndCallback? onDragEnd;
|
|
|
|
final PhotoViewImageDragUpdateCallback? onDragUpdate;
|
|
|
|
final PhotoViewImageScaleEndCallback? onScaleEnd;
|
2024-05-02 17:37:39 +02:00
|
|
|
final PhotoViewImageLongPressStartCallback? onLongPressStart;
|
2023-02-01 17:59:34 +01:00
|
|
|
final Size outerSize;
|
|
|
|
final HitTestBehavior? gestureDetectorBehavior;
|
|
|
|
final bool? tightMode;
|
|
|
|
final FilterQuality? filterQuality;
|
|
|
|
final bool? disableGestures;
|
|
|
|
final bool? enablePanAlways;
|
2023-09-18 05:57:05 +02:00
|
|
|
final int index;
|
2023-02-01 17:59:34 +01:00
|
|
|
|
|
|
|
@override
|
|
|
|
createState() => _ImageWrapperState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _ImageWrapperState extends State<ImageWrapper> {
|
|
|
|
ImageStreamListener? _imageStreamListener;
|
|
|
|
ImageStream? _imageStream;
|
|
|
|
ImageChunkEvent? _loadingProgress;
|
|
|
|
ImageInfo? _imageInfo;
|
|
|
|
bool _loading = true;
|
|
|
|
Size? _imageSize;
|
|
|
|
Object? _lastException;
|
|
|
|
StackTrace? _lastStack;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
super.dispose();
|
|
|
|
_stopImageStream();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void didChangeDependencies() {
|
|
|
|
_resolveImage();
|
|
|
|
super.didChangeDependencies();
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
void didUpdateWidget(ImageWrapper oldWidget) {
|
|
|
|
super.didUpdateWidget(oldWidget);
|
|
|
|
if (widget.imageProvider != oldWidget.imageProvider) {
|
|
|
|
_resolveImage();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// retrieve image from the provider
|
|
|
|
void _resolveImage() {
|
|
|
|
final ImageStream newStream = widget.imageProvider.resolve(
|
|
|
|
const ImageConfiguration(),
|
|
|
|
);
|
|
|
|
_updateSourceStream(newStream);
|
|
|
|
}
|
|
|
|
|
|
|
|
ImageStreamListener _getOrCreateListener() {
|
|
|
|
void handleImageChunk(ImageChunkEvent event) {
|
|
|
|
setState(() {
|
|
|
|
_loadingProgress = event;
|
|
|
|
_lastException = null;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
void handleImageFrame(ImageInfo info, bool synchronousCall) {
|
|
|
|
setupCB() {
|
|
|
|
_imageSize = Size(
|
|
|
|
info.image.width.toDouble(),
|
|
|
|
info.image.height.toDouble(),
|
|
|
|
);
|
|
|
|
_loading = false;
|
|
|
|
_imageInfo = _imageInfo;
|
|
|
|
|
|
|
|
_loadingProgress = null;
|
|
|
|
_lastException = null;
|
|
|
|
_lastStack = null;
|
|
|
|
}
|
2023-09-18 05:57:05 +02:00
|
|
|
|
2023-02-01 17:59:34 +01:00
|
|
|
synchronousCall ? setupCB() : setState(setupCB);
|
|
|
|
}
|
|
|
|
|
|
|
|
void handleError(dynamic error, StackTrace? stackTrace) {
|
|
|
|
setState(() {
|
|
|
|
_loading = false;
|
|
|
|
_lastException = error;
|
|
|
|
_lastStack = stackTrace;
|
|
|
|
});
|
|
|
|
assert(() {
|
|
|
|
if (widget.errorBuilder == null) {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}());
|
|
|
|
}
|
|
|
|
|
|
|
|
_imageStreamListener = ImageStreamListener(
|
|
|
|
handleImageFrame,
|
|
|
|
onChunk: handleImageChunk,
|
|
|
|
onError: handleError,
|
|
|
|
);
|
|
|
|
|
|
|
|
return _imageStreamListener!;
|
|
|
|
}
|
|
|
|
|
|
|
|
void _updateSourceStream(ImageStream newStream) {
|
|
|
|
if (_imageStream?.key == newStream.key) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
_imageStream?.removeListener(_imageStreamListener!);
|
|
|
|
_imageStream = newStream;
|
|
|
|
_imageStream!.addListener(_getOrCreateListener());
|
|
|
|
}
|
|
|
|
|
|
|
|
void _stopImageStream() {
|
|
|
|
_imageStream?.removeListener(_imageStreamListener!);
|
|
|
|
}
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
if (_loading) {
|
|
|
|
return _buildLoading(context);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_lastException != null) {
|
|
|
|
return _buildError(context);
|
|
|
|
}
|
|
|
|
|
|
|
|
final scaleBoundaries = ScaleBoundaries(
|
|
|
|
widget.minScale ?? 0.0,
|
|
|
|
widget.maxScale ?? double.infinity,
|
|
|
|
widget.initialScale ?? PhotoViewComputedScale.contained,
|
|
|
|
widget.outerSize,
|
|
|
|
_imageSize!,
|
|
|
|
);
|
|
|
|
|
|
|
|
return PhotoViewCore(
|
|
|
|
imageProvider: widget.imageProvider,
|
|
|
|
backgroundDecoration: widget.backgroundDecoration,
|
|
|
|
gaplessPlayback: widget.gaplessPlayback,
|
|
|
|
enableRotation: widget.enableRotation,
|
|
|
|
heroAttributes: widget.heroAttributes,
|
|
|
|
basePosition: widget.basePosition ?? Alignment.center,
|
|
|
|
controller: widget.controller,
|
|
|
|
scaleStateController: widget.scaleStateController,
|
|
|
|
scaleStateCycle: widget.scaleStateCycle ?? defaultScaleStateCycle,
|
|
|
|
scaleBoundaries: scaleBoundaries,
|
|
|
|
onTapUp: widget.onTapUp,
|
|
|
|
onTapDown: widget.onTapDown,
|
|
|
|
onDragStart: widget.onDragStart,
|
|
|
|
onDragEnd: widget.onDragEnd,
|
|
|
|
onDragUpdate: widget.onDragUpdate,
|
|
|
|
onScaleEnd: widget.onScaleEnd,
|
2024-05-02 17:37:39 +02:00
|
|
|
onLongPressStart: widget.onLongPressStart,
|
2023-02-01 17:59:34 +01:00
|
|
|
gestureDetectorBehavior: widget.gestureDetectorBehavior,
|
|
|
|
tightMode: widget.tightMode ?? false,
|
|
|
|
filterQuality: widget.filterQuality ?? FilterQuality.none,
|
|
|
|
disableGestures: widget.disableGestures ?? false,
|
|
|
|
enablePanAlways: widget.enablePanAlways ?? false,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget _buildLoading(BuildContext context) {
|
|
|
|
if (widget.loadingBuilder != null) {
|
2023-09-18 05:57:05 +02:00
|
|
|
return widget.loadingBuilder!(context, _loadingProgress, widget.index);
|
2023-02-01 17:59:34 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return PhotoViewDefaultLoading(
|
|
|
|
event: _loadingProgress,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget _buildError(
|
|
|
|
BuildContext context,
|
|
|
|
) {
|
|
|
|
if (widget.errorBuilder != null) {
|
|
|
|
return widget.errorBuilder!(context, _lastException!, _lastStack);
|
|
|
|
}
|
|
|
|
return PhotoViewDefaultError(
|
|
|
|
decoration: widget.backgroundDecoration,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class CustomChildWrapper extends StatelessWidget {
|
|
|
|
const CustomChildWrapper({
|
2024-01-27 17:14:32 +01:00
|
|
|
super.key,
|
2023-02-01 17:59:34 +01:00
|
|
|
this.child,
|
|
|
|
required this.childSize,
|
|
|
|
required this.backgroundDecoration,
|
|
|
|
this.heroAttributes,
|
|
|
|
this.scaleStateChangedCallback,
|
|
|
|
required this.enableRotation,
|
|
|
|
required this.controller,
|
|
|
|
required this.scaleStateController,
|
|
|
|
required this.maxScale,
|
|
|
|
required this.minScale,
|
|
|
|
required this.initialScale,
|
|
|
|
required this.basePosition,
|
|
|
|
required this.scaleStateCycle,
|
|
|
|
this.onTapUp,
|
|
|
|
this.onTapDown,
|
|
|
|
this.onDragStart,
|
|
|
|
this.onDragEnd,
|
|
|
|
this.onDragUpdate,
|
|
|
|
this.onScaleEnd,
|
2024-05-02 17:37:39 +02:00
|
|
|
this.onLongPressStart,
|
2023-02-01 17:59:34 +01:00
|
|
|
required this.outerSize,
|
|
|
|
this.gestureDetectorBehavior,
|
|
|
|
required this.tightMode,
|
|
|
|
required this.filterQuality,
|
|
|
|
required this.disableGestures,
|
|
|
|
required this.enablePanAlways,
|
2024-01-27 17:14:32 +01:00
|
|
|
});
|
2023-02-01 17:59:34 +01:00
|
|
|
|
|
|
|
final Widget? child;
|
|
|
|
final Size? childSize;
|
|
|
|
final Decoration backgroundDecoration;
|
|
|
|
final PhotoViewHeroAttributes? heroAttributes;
|
|
|
|
final ValueChanged<PhotoViewScaleState>? scaleStateChangedCallback;
|
|
|
|
final bool enableRotation;
|
|
|
|
|
|
|
|
final PhotoViewControllerBase controller;
|
|
|
|
final PhotoViewScaleStateController scaleStateController;
|
|
|
|
|
|
|
|
final dynamic maxScale;
|
|
|
|
final dynamic minScale;
|
|
|
|
final dynamic initialScale;
|
|
|
|
|
|
|
|
final Alignment? basePosition;
|
|
|
|
final ScaleStateCycle? scaleStateCycle;
|
|
|
|
final PhotoViewImageTapUpCallback? onTapUp;
|
|
|
|
final PhotoViewImageTapDownCallback? onTapDown;
|
|
|
|
final PhotoViewImageDragStartCallback? onDragStart;
|
|
|
|
final PhotoViewImageDragEndCallback? onDragEnd;
|
|
|
|
final PhotoViewImageDragUpdateCallback? onDragUpdate;
|
|
|
|
final PhotoViewImageScaleEndCallback? onScaleEnd;
|
2024-05-02 17:37:39 +02:00
|
|
|
final PhotoViewImageLongPressStartCallback? onLongPressStart;
|
2023-02-01 17:59:34 +01:00
|
|
|
final Size outerSize;
|
|
|
|
final HitTestBehavior? gestureDetectorBehavior;
|
|
|
|
final bool? tightMode;
|
|
|
|
final FilterQuality? filterQuality;
|
|
|
|
final bool? disableGestures;
|
|
|
|
final bool? enablePanAlways;
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
final scaleBoundaries = ScaleBoundaries(
|
|
|
|
minScale ?? 0.0,
|
|
|
|
maxScale ?? double.infinity,
|
|
|
|
initialScale ?? PhotoViewComputedScale.contained,
|
|
|
|
outerSize,
|
|
|
|
childSize ?? outerSize,
|
|
|
|
);
|
|
|
|
|
|
|
|
return PhotoViewCore.customChild(
|
|
|
|
customChild: child,
|
|
|
|
backgroundDecoration: backgroundDecoration,
|
|
|
|
enableRotation: enableRotation,
|
|
|
|
heroAttributes: heroAttributes,
|
|
|
|
controller: controller,
|
|
|
|
scaleStateController: scaleStateController,
|
|
|
|
scaleStateCycle: scaleStateCycle ?? defaultScaleStateCycle,
|
|
|
|
basePosition: basePosition ?? Alignment.center,
|
|
|
|
scaleBoundaries: scaleBoundaries,
|
|
|
|
onTapUp: onTapUp,
|
|
|
|
onTapDown: onTapDown,
|
|
|
|
onDragStart: onDragStart,
|
|
|
|
onDragEnd: onDragEnd,
|
|
|
|
onDragUpdate: onDragUpdate,
|
|
|
|
onScaleEnd: onScaleEnd,
|
2024-05-02 17:37:39 +02:00
|
|
|
onLongPressStart: onLongPressStart,
|
2023-02-01 17:59:34 +01:00
|
|
|
gestureDetectorBehavior: gestureDetectorBehavior,
|
|
|
|
tightMode: tightMode ?? false,
|
|
|
|
filterQuality: filterQuality ?? FilterQuality.none,
|
|
|
|
disableGestures: disableGestures ?? false,
|
|
|
|
enablePanAlways: enablePanAlways ?? false,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|