487 lines
14 KiB
Dart
487 lines
14 KiB
Dart
library interactiveviewer_gallery;
|
|
|
|
import 'dart:io';
|
|
|
|
import 'package:cached_network_image/cached_network_image.dart';
|
|
import 'package:dio/dio.dart';
|
|
import 'package:flutter/material.dart';
|
|
import 'package:flutter/services.dart';
|
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart';
|
|
import 'package:get/get.dart';
|
|
import 'package:path_provider/path_provider.dart';
|
|
import 'package:pilipala/utils/download.dart';
|
|
import 'package:share_plus/share_plus.dart';
|
|
import 'package:status_bar_control/status_bar_control.dart';
|
|
import 'custom_dismissible.dart';
|
|
import 'interactive_viewer_boundary.dart';
|
|
|
|
/// Builds a carousel controlled by a [PageView] for the tweet media sources.
|
|
///
|
|
/// Used for showing a full screen view of the [TweetMedia] sources.
|
|
///
|
|
/// The sources can be panned and zoomed interactively using an
|
|
/// [InteractiveViewer].
|
|
/// An [InteractiveViewerBoundary] is used to detect when the boundary of the
|
|
/// source is hit after zooming in to disable or enable the swiping gesture of
|
|
/// the [PageView].
|
|
///
|
|
typedef IndexedFocusedWidgetBuilder = Widget Function(
|
|
BuildContext context, int index, bool isFocus, bool enablePageView);
|
|
|
|
typedef IndexedTagStringBuilder = String Function(int index);
|
|
|
|
class InteractiveviewerGallery<T> extends StatefulWidget {
|
|
const InteractiveviewerGallery({
|
|
required this.sources,
|
|
required this.initIndex,
|
|
this.itemBuilder,
|
|
this.maxScale = 4.5,
|
|
this.minScale = 1.0,
|
|
this.onPageChanged,
|
|
this.onDismissed,
|
|
Key? key,
|
|
}) : super(key: key);
|
|
|
|
/// The sources to show.
|
|
final List<T> sources;
|
|
|
|
/// The index of the first source in [sources] to show.
|
|
final int initIndex;
|
|
|
|
/// The item content
|
|
final IndexedFocusedWidgetBuilder? itemBuilder;
|
|
|
|
final double maxScale;
|
|
|
|
final double minScale;
|
|
|
|
final ValueChanged<int>? onPageChanged;
|
|
|
|
final ValueChanged<int>? onDismissed;
|
|
|
|
@override
|
|
State<InteractiveviewerGallery> createState() =>
|
|
_InteractiveviewerGalleryState();
|
|
}
|
|
|
|
class _InteractiveviewerGalleryState extends State<InteractiveviewerGallery>
|
|
with SingleTickerProviderStateMixin {
|
|
PageController? _pageController;
|
|
TransformationController? _transformationController;
|
|
|
|
/// The controller to animate the transformation value of the
|
|
/// [InteractiveViewer] when it should reset.
|
|
late AnimationController _animationController;
|
|
Animation<Matrix4>? _animation;
|
|
|
|
/// `true` when an source is zoomed in and not at the at a horizontal boundary
|
|
/// to disable the [PageView].
|
|
bool _enablePageView = true;
|
|
|
|
/// `true` when an source is zoomed in to disable the [CustomDismissible].
|
|
bool _enableDismiss = true;
|
|
|
|
late Offset _doubleTapLocalPosition;
|
|
|
|
int? currentIndex;
|
|
|
|
@override
|
|
void initState() {
|
|
super.initState();
|
|
|
|
_pageController = PageController(initialPage: widget.initIndex);
|
|
|
|
_transformationController = TransformationController();
|
|
|
|
_animationController = AnimationController(
|
|
vsync: this,
|
|
duration: const Duration(milliseconds: 300),
|
|
)
|
|
..addListener(() {
|
|
_transformationController!.value =
|
|
_animation?.value ?? Matrix4.identity();
|
|
})
|
|
..addStatusListener((AnimationStatus status) {
|
|
if (status == AnimationStatus.completed && !_enableDismiss) {
|
|
setState(() {
|
|
_enableDismiss = true;
|
|
});
|
|
}
|
|
});
|
|
|
|
currentIndex = widget.initIndex;
|
|
setStatusBar();
|
|
}
|
|
|
|
setStatusBar() async {
|
|
if (Platform.isIOS || Platform.isAndroid) {
|
|
await StatusBarControl.setHidden(true,
|
|
animation: StatusBarAnimation.FADE);
|
|
}
|
|
}
|
|
|
|
@override
|
|
void dispose() {
|
|
_pageController!.dispose();
|
|
_animationController.dispose();
|
|
try {
|
|
StatusBarControl.setHidden(false, animation: StatusBarAnimation.FADE);
|
|
} catch (_) {}
|
|
super.dispose();
|
|
}
|
|
|
|
/// When the source gets scaled up, the swipe up / down to dismiss gets
|
|
/// disabled.
|
|
///
|
|
/// When the scale resets, the dismiss and the page view swiping gets enabled.
|
|
void _onScaleChanged(double scale) {
|
|
final bool initialScale = scale <= widget.minScale;
|
|
|
|
if (initialScale) {
|
|
if (!_enableDismiss) {
|
|
setState(() {
|
|
_enableDismiss = true;
|
|
});
|
|
}
|
|
|
|
if (!_enablePageView) {
|
|
setState(() {
|
|
_enablePageView = true;
|
|
});
|
|
}
|
|
} else {
|
|
if (_enableDismiss) {
|
|
setState(() {
|
|
_enableDismiss = false;
|
|
});
|
|
}
|
|
|
|
if (_enablePageView) {
|
|
setState(() {
|
|
_enablePageView = false;
|
|
});
|
|
}
|
|
}
|
|
}
|
|
|
|
/// When the left boundary has been hit after scaling up the source, the page
|
|
/// view swiping gets enabled if it has a page to swipe to.
|
|
void _onLeftBoundaryHit() {
|
|
if (!_enablePageView && _pageController!.page!.floor() > 0) {
|
|
setState(() {
|
|
_enablePageView = true;
|
|
});
|
|
}
|
|
}
|
|
|
|
/// When the right boundary has been hit after scaling up the source, the page
|
|
/// view swiping gets enabled if it has a page to swipe to.
|
|
void _onRightBoundaryHit() {
|
|
if (!_enablePageView &&
|
|
_pageController!.page!.floor() < widget.sources.length - 1) {
|
|
setState(() {
|
|
_enablePageView = true;
|
|
});
|
|
}
|
|
}
|
|
|
|
/// When the source has been scaled up and no horizontal boundary has been hit,
|
|
/// the page view swiping gets disabled.
|
|
void _onNoBoundaryHit() {
|
|
if (_enablePageView) {
|
|
setState(() {
|
|
_enablePageView = false;
|
|
});
|
|
}
|
|
}
|
|
|
|
/// When the page view changed its page, the source will animate back into the
|
|
/// original scale if it was scaled up.
|
|
///
|
|
/// Additionally the swipe up / down to dismiss gets enabled.
|
|
void _onPageChanged(int page) {
|
|
setState(() {
|
|
currentIndex = page;
|
|
});
|
|
widget.onPageChanged?.call(page);
|
|
if (_transformationController!.value != Matrix4.identity()) {
|
|
// animate the reset for the transformation of the interactive viewer
|
|
|
|
_animation = Matrix4Tween(
|
|
begin: _transformationController!.value,
|
|
end: Matrix4.identity(),
|
|
).animate(
|
|
CurveTween(curve: Curves.easeOut).animate(_animationController),
|
|
);
|
|
|
|
_animationController.forward(from: 0);
|
|
}
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return InteractiveViewerBoundary(
|
|
controller: _transformationController,
|
|
boundaryWidth: MediaQuery.of(context).size.width,
|
|
onScaleChanged: _onScaleChanged,
|
|
onLeftBoundaryHit: _onLeftBoundaryHit,
|
|
onRightBoundaryHit: _onRightBoundaryHit,
|
|
onNoBoundaryHit: _onNoBoundaryHit,
|
|
maxScale: widget.maxScale,
|
|
minScale: widget.minScale,
|
|
child: Stack(children: [
|
|
CustomDismissible(
|
|
onDismissed: () {
|
|
Navigator.of(context).pop();
|
|
widget.onDismissed?.call(_pageController!.page!.floor());
|
|
},
|
|
enabled: _enableDismiss,
|
|
child: PageView.builder(
|
|
onPageChanged: _onPageChanged,
|
|
controller: _pageController,
|
|
physics:
|
|
_enablePageView ? null : const NeverScrollableScrollPhysics(),
|
|
itemCount: widget.sources.length,
|
|
itemBuilder: (BuildContext context, int index) {
|
|
return GestureDetector(
|
|
onDoubleTapDown: (TapDownDetails details) {
|
|
_doubleTapLocalPosition = details.localPosition;
|
|
},
|
|
onDoubleTap: onDoubleTap,
|
|
onLongPress: onLongPress,
|
|
child: widget.itemBuilder != null
|
|
? widget.itemBuilder!(
|
|
context,
|
|
index,
|
|
index == currentIndex,
|
|
_enablePageView,
|
|
)
|
|
: _itemBuilder(widget.sources, index),
|
|
);
|
|
},
|
|
),
|
|
),
|
|
Positioned(
|
|
bottom: 0,
|
|
left: 0,
|
|
right: 0,
|
|
child: Container(
|
|
padding: EdgeInsets.fromLTRB(
|
|
12, 8, 20, MediaQuery.of(context).padding.bottom + 8),
|
|
decoration: _enablePageView
|
|
? BoxDecoration(
|
|
gradient: LinearGradient(
|
|
begin: Alignment.topCenter,
|
|
end: Alignment.bottomCenter,
|
|
colors: [
|
|
Colors.transparent,
|
|
Colors.black.withOpacity(0.3)
|
|
],
|
|
),
|
|
)
|
|
: null,
|
|
child: Row(
|
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
|
children: [
|
|
IconButton(
|
|
icon: const Icon(Icons.close, color: Colors.white),
|
|
onPressed: () {
|
|
Navigator.of(context).pop();
|
|
widget.onDismissed?.call(_pageController!.page!.floor());
|
|
},
|
|
),
|
|
widget.sources.length > 1
|
|
? Text(
|
|
"${currentIndex! + 1}/${widget.sources.length}",
|
|
style: const TextStyle(color: Colors.white),
|
|
)
|
|
: const SizedBox(),
|
|
PopupMenuButton(
|
|
itemBuilder: (context) {
|
|
return [
|
|
PopupMenuItem(
|
|
value: 0,
|
|
onTap: () => onShareImg(widget.sources[currentIndex!]),
|
|
child: const Text("分享图片"),
|
|
),
|
|
PopupMenuItem(
|
|
value: 1,
|
|
onTap: () {
|
|
onCopyImg(widget.sources[currentIndex!].toString());
|
|
},
|
|
child: const Text("复制图片"),
|
|
),
|
|
PopupMenuItem(
|
|
value: 2,
|
|
onTap: () {
|
|
DownloadUtils.downloadImg(
|
|
widget.sources[currentIndex!]);
|
|
},
|
|
child: const Text("保存图片"),
|
|
),
|
|
];
|
|
},
|
|
child: const Icon(Icons.more_horiz, color: Colors.white),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
),
|
|
]),
|
|
);
|
|
}
|
|
|
|
// 图片分享
|
|
void onShareImg(String imgUrl) async {
|
|
SmartDialog.showLoading();
|
|
var response = await Dio()
|
|
.get(imgUrl, options: Options(responseType: ResponseType.bytes));
|
|
final temp = await getTemporaryDirectory();
|
|
SmartDialog.dismiss();
|
|
String imgName =
|
|
"plpl_pic_${DateTime.now().toString().split('-').join()}.jpg";
|
|
var path = '${temp.path}/$imgName';
|
|
File(path).writeAsBytesSync(response.data);
|
|
Share.shareXFiles([XFile(path)], subject: imgUrl);
|
|
}
|
|
|
|
// 复制图片
|
|
void onCopyImg(String imgUrl) {
|
|
Clipboard.setData(
|
|
ClipboardData(text: widget.sources[currentIndex!].toString()))
|
|
.then((value) {
|
|
SmartDialog.showToast('已复制到粘贴板');
|
|
}).catchError((err) {
|
|
SmartDialog.showNotify(
|
|
msg: err.toString(),
|
|
notifyType: NotifyType.error,
|
|
);
|
|
});
|
|
}
|
|
|
|
Widget _itemBuilder(sources, index) {
|
|
return GestureDetector(
|
|
behavior: HitTestBehavior.opaque,
|
|
onTap: () {
|
|
if (_enablePageView) {
|
|
Navigator.of(context).pop();
|
|
}
|
|
},
|
|
child: Center(
|
|
child: Hero(
|
|
tag: sources[index],
|
|
child: CachedNetworkImage(
|
|
fadeInDuration: const Duration(milliseconds: 0),
|
|
imageUrl: sources[index],
|
|
fit: BoxFit.contain,
|
|
),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
onDoubleTap() {
|
|
Matrix4 matrix = _transformationController!.value.clone();
|
|
double currentScale = matrix.row0.x;
|
|
|
|
double targetScale = widget.minScale;
|
|
|
|
if (currentScale <= widget.minScale) {
|
|
targetScale = widget.maxScale * 0.7;
|
|
}
|
|
|
|
double offSetX = targetScale == 1.0
|
|
? 0.0
|
|
: -_doubleTapLocalPosition.dx * (targetScale - 1);
|
|
double offSetY = targetScale == 1.0
|
|
? 0.0
|
|
: -_doubleTapLocalPosition.dy * (targetScale - 1);
|
|
|
|
matrix = Matrix4.fromList([
|
|
targetScale,
|
|
matrix.row1.x,
|
|
matrix.row2.x,
|
|
matrix.row3.x,
|
|
matrix.row0.y,
|
|
targetScale,
|
|
matrix.row2.y,
|
|
matrix.row3.y,
|
|
matrix.row0.z,
|
|
matrix.row1.z,
|
|
targetScale,
|
|
matrix.row3.z,
|
|
offSetX,
|
|
offSetY,
|
|
matrix.row2.w,
|
|
matrix.row3.w
|
|
]);
|
|
|
|
_animation = Matrix4Tween(
|
|
begin: _transformationController!.value,
|
|
end: matrix,
|
|
).animate(
|
|
CurveTween(curve: Curves.easeOut).animate(_animationController),
|
|
);
|
|
_animationController
|
|
.forward(from: 0)
|
|
.whenComplete(() => _onScaleChanged(targetScale));
|
|
}
|
|
|
|
onLongPress() {
|
|
showModalBottomSheet(
|
|
context: context,
|
|
useRootNavigator: true,
|
|
isScrollControlled: true,
|
|
builder: (context) {
|
|
return Container(
|
|
padding:
|
|
EdgeInsets.only(bottom: MediaQuery.of(context).padding.bottom),
|
|
child: Column(
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
InkWell(
|
|
onTap: () => Get.back(),
|
|
child: Container(
|
|
height: 35,
|
|
padding: const EdgeInsets.only(bottom: 2),
|
|
child: Center(
|
|
child: Container(
|
|
width: 32,
|
|
height: 3,
|
|
decoration: BoxDecoration(
|
|
color: Theme.of(context).colorScheme.outline,
|
|
borderRadius:
|
|
const BorderRadius.all(Radius.circular(3))),
|
|
),
|
|
),
|
|
),
|
|
),
|
|
ListTile(
|
|
onTap: () {
|
|
onShareImg(widget.sources[currentIndex!]);
|
|
Navigator.of(context).pop();
|
|
},
|
|
title: const Text('分享图片'),
|
|
),
|
|
ListTile(
|
|
onTap: () {
|
|
onCopyImg(widget.sources[currentIndex!].toString());
|
|
Navigator.of(context).pop();
|
|
},
|
|
title: const Text('复制图片'),
|
|
),
|
|
ListTile(
|
|
onTap: () {
|
|
DownloadUtils.downloadImg(widget.sources[currentIndex!]);
|
|
Navigator.of(context).pop();
|
|
},
|
|
title: const Text('保存图片'),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
},
|
|
);
|
|
}
|
|
}
|