nc-photos/app/lib/widget/viewer.dart

1040 lines
34 KiB
Dart
Raw Normal View History

import 'dart:async';
2021-04-10 06:28:12 +02:00
import 'dart:math';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/rendering.dart';
import 'package:flutter/services.dart';
2023-04-17 18:15:29 +02:00
import 'package:flutter_bloc/flutter_bloc.dart';
2024-05-29 15:55:16 +02:00
import 'package:intl/intl.dart';
2022-01-25 11:17:19 +01:00
import 'package:kiwi/kiwi.dart';
2021-04-10 06:28:12 +02:00
import 'package:logging/logging.dart';
import 'package:nc_photos/account.dart';
2021-07-25 07:00:38 +02:00
import 'package:nc_photos/app_localizations.dart';
import 'package:nc_photos/asset.dart';
2023-04-17 18:15:29 +02:00
import 'package:nc_photos/controller/account_controller.dart';
import 'package:nc_photos/controller/collection_items_controller.dart';
2022-01-25 11:17:19 +01:00
import 'package:nc_photos/di_container.dart';
2021-09-28 22:56:44 +02:00
import 'package:nc_photos/download_handler.dart';
2023-04-17 18:15:29 +02:00
import 'package:nc_photos/entity/collection.dart';
import 'package:nc_photos/entity/collection/adapter.dart';
import 'package:nc_photos/entity/collection_item.dart';
import 'package:nc_photos/entity/file_descriptor.dart';
2021-05-06 13:36:20 +02:00
import 'package:nc_photos/entity/file_util.dart' as file_util;
2023-07-17 09:35:45 +02:00
import 'package:nc_photos/entity/pref.dart';
2022-09-07 11:37:50 +02:00
import 'package:nc_photos/flutter_util.dart';
2021-04-10 06:28:12 +02:00
import 'package:nc_photos/k.dart' as k;
import 'package:nc_photos/live_photo_util.dart';
2023-04-17 18:15:29 +02:00
import 'package:nc_photos/object_extension.dart';
2022-05-04 10:42:46 +02:00
import 'package:nc_photos/platform/features.dart' as features;
import 'package:nc_photos/share_handler.dart';
2023-04-17 18:15:29 +02:00
import 'package:nc_photos/snack_bar_manager.dart';
2021-04-10 06:28:12 +02:00
import 'package:nc_photos/theme.dart';
2024-08-10 13:13:56 +02:00
import 'package:nc_photos/widget/app_intermediate_circular_progress_indicator.dart';
2021-08-23 19:28:25 +02:00
import 'package:nc_photos/widget/disposable.dart';
import 'package:nc_photos/widget/handler/remove_selection_handler.dart';
2021-07-31 17:38:26 +02:00
import 'package:nc_photos/widget/horizontal_page_viewer.dart';
2022-07-12 22:11:27 +02:00
import 'package:nc_photos/widget/image_editor.dart';
import 'package:nc_photos/widget/image_enhancer.dart';
2021-05-02 16:18:26 +02:00
import 'package:nc_photos/widget/image_viewer.dart';
import 'package:nc_photos/widget/live_photo_viewer.dart';
import 'package:nc_photos/widget/png_icon.dart';
2021-09-14 23:00:24 +02:00
import 'package:nc_photos/widget/slideshow_dialog.dart';
import 'package:nc_photos/widget/slideshow_viewer.dart';
2021-05-06 13:36:20 +02:00
import 'package:nc_photos/widget/video_viewer.dart';
import 'package:nc_photos/widget/viewer_bottom_app_bar.dart';
2021-04-10 06:28:12 +02:00
import 'package:nc_photos/widget/viewer_detail_pane.dart';
2021-09-14 23:00:24 +02:00
import 'package:nc_photos/widget/viewer_mixin.dart';
2022-12-16 16:01:04 +01:00
import 'package:np_codegen/np_codegen.dart';
import 'package:np_common/or_null.dart';
2024-05-29 15:55:16 +02:00
import 'package:np_platform_util/np_platform_util.dart';
import 'package:video_player_platform_interface/video_player_platform_interface.dart';
2022-12-16 16:01:04 +01:00
part 'viewer.g.dart';
2021-04-10 06:28:12 +02:00
2023-04-17 18:15:29 +02:00
class ViewerCollectionData {
const ViewerCollectionData(this.collection, this.items);
final Collection collection;
final List<CollectionItem> items;
}
2021-04-10 06:28:12 +02:00
class ViewerArguments {
2023-04-17 18:15:29 +02:00
const ViewerArguments(
2021-08-13 22:18:35 +02:00
this.account,
this.streamFiles,
this.startIndex, {
2023-04-17 18:15:29 +02:00
this.fromCollection,
2021-08-13 22:18:35 +02:00
});
2021-04-10 06:28:12 +02:00
final Account account;
final List<FileDescriptor> streamFiles;
2021-04-10 06:28:12 +02:00
final int startIndex;
2023-04-17 18:15:29 +02:00
final ViewerCollectionData? fromCollection;
2021-04-10 06:28:12 +02:00
}
class Viewer extends StatefulWidget {
static const routeName = "/viewer";
2022-09-07 11:37:50 +02:00
static Route buildRoute(ViewerArguments args) =>
CustomizableMaterialPageRoute(
transitionDuration: k.heroDurationNormal,
reverseTransitionDuration: k.heroDurationNormal,
builder: (_) => Viewer.fromArgs(args),
2021-07-23 22:05:57 +02:00
);
2021-09-15 08:58:06 +02:00
const Viewer({
2024-05-28 17:10:33 +02:00
super.key,
2021-07-23 22:05:57 +02:00
required this.account,
required this.streamFiles,
required this.startIndex,
2023-04-17 18:15:29 +02:00
this.fromCollection,
2024-05-28 17:10:33 +02:00
});
2021-04-10 06:28:12 +02:00
2021-07-23 22:05:57 +02:00
Viewer.fromArgs(ViewerArguments args, {Key? key})
2021-04-10 06:28:12 +02:00
: this(
key: key,
account: args.account,
streamFiles: args.streamFiles,
startIndex: args.startIndex,
2023-04-17 18:15:29 +02:00
fromCollection: args.fromCollection,
2021-04-10 06:28:12 +02:00
);
@override
createState() => _ViewerState();
final Account account;
final List<FileDescriptor> streamFiles;
2021-04-10 06:28:12 +02:00
final int startIndex;
2021-08-13 22:18:35 +02:00
2023-04-17 18:15:29 +02:00
/// Data of the collection these files belongs to, or null
final ViewerCollectionData? fromCollection;
2021-04-10 06:28:12 +02:00
}
2022-12-16 16:01:04 +01:00
@npLog
2021-09-14 23:00:24 +02:00
class _ViewerState extends State<Viewer>
with DisposableManagerMixin<Viewer>, ViewerControllersMixin<Viewer> {
@override
void initState() {
super.initState();
_streamFilesView = widget.streamFiles;
}
2021-04-10 06:28:12 +02:00
@override
build(BuildContext context) {
2022-11-12 10:55:33 +01:00
final originalBrightness = Theme.of(context).brightness;
return Theme(
data: buildDarkTheme(context),
2023-08-04 21:51:28 +02:00
child: AnnotatedRegion<SystemUiOverlayStyle>(
value: const SystemUiOverlayStyle(
systemNavigationBarColor: Colors.black,
systemNavigationBarIconBrightness: Brightness.dark,
),
child: Scaffold(
extendBodyBehindAppBar: true,
extendBody: true,
appBar: _isShowAppBar ? _buildAppBar(context) : null,
bottomNavigationBar: _isShowAppBar && !_isDetailPaneActive
? _buildBottomAppBar(context)
: null,
2023-08-04 21:51:28 +02:00
body: Builder(
builder: (context) => _buildContent(context, originalBrightness),
),
2021-07-31 17:38:26 +02:00
),
2021-04-10 06:28:12 +02:00
),
);
}
2022-11-12 10:55:33 +01:00
Widget _buildContent(BuildContext context, Brightness originalBrightness) {
2021-05-02 16:18:26 +02:00
return GestureDetector(
onTap: () {
setState(() {
_setShowActionBar(!_isShowAppBar);
});
2021-04-10 06:28:12 +02:00
},
2021-05-02 16:18:26 +02:00
child: Stack(
children: [
Container(color: Colors.black),
2021-07-31 17:38:26 +02:00
if (!_isViewerLoaded ||
2021-09-14 23:03:09 +02:00
_pageStates[_viewerController.currentPage]?.hasLoaded != true)
2021-09-15 08:58:06 +02:00
const Align(
2021-05-02 16:18:26 +02:00
alignment: Alignment.center,
2024-08-10 13:13:56 +02:00
child: AppIntermediateCircularProgressIndicator(),
2021-04-10 06:28:12 +02:00
),
2021-07-31 17:38:26 +02:00
HorizontalPageViewer(
2023-05-30 18:48:58 +02:00
key: _pageViewerKey,
pageCount: _streamFilesView.length,
2022-11-12 10:55:33 +01:00
pageBuilder: (context, i) =>
_buildPage(context, i, originalBrightness),
2021-07-31 17:38:26 +02:00
initialPage: widget.startIndex,
controller: _viewerController,
viewportFraction: _viewportFraction,
canSwitchPage: _canSwitchPage(),
onPageChanged: (from, to) {
setState(() {
_pageStates[from]?.shouldPlayLivePhoto = false;
});
2021-08-05 12:38:43 +02:00
},
2021-05-02 16:18:26 +02:00
),
if (_isShowAppBar)
Container(
// + status bar height
height: kToolbarHeight + MediaQuery.of(context).padding.top,
decoration: const BoxDecoration(
gradient: LinearGradient(
begin: Alignment(0, -1),
end: Alignment(0, 1),
colors: [
Color.fromARGB(192, 0, 0, 0),
Color.fromARGB(0, 0, 0, 0),
],
),
),
),
2021-05-02 16:18:26 +02:00
],
2021-04-10 06:28:12 +02:00
),
);
}
AppBar _buildAppBar(BuildContext context) {
2022-01-25 11:17:19 +01:00
final index =
_isViewerLoaded ? _viewerController.currentPage : widget.startIndex;
final file = _streamFilesView[index];
2024-05-29 15:55:16 +02:00
final isCentered = getRawPlatform() == NpPlatform.iOs;
return AppBar(
backgroundColor: Colors.transparent,
elevation: 0,
2024-05-29 15:55:16 +02:00
title: _isDetailPaneActive
? null
: _AppBarTitle(
file: file,
isCentered: isCentered,
),
titleSpacing: 0,
centerTitle: isCentered,
actions: [
if (!_isDetailPaneActive && _canOpenDetailPane()) ...[
if (getLivePhotoTypeFromFile(file) != null)
if (_pageStates[index]?.shouldPlayLivePhoto ?? false)
IconButton(
icon: const Icon(Icons.motion_photos_pause_outlined),
onPressed: () => _onPauseMotionPhotosPressed(index),
)
else
IconButton(
icon: const PngIcon(icMotionPhotosPlay24dp),
onPressed: () => _onPlayMotionPhotosPressed(index),
),
(_pageStates[index]?.favoriteOverride ?? file.fdIsFavorite) == true
? IconButton(
icon: const Icon(Icons.star),
tooltip: L10n.global().unfavoriteTooltip,
onPressed: () => _onUnfavoritePressed(index),
)
: IconButton(
icon: const Icon(Icons.star_border),
tooltip: L10n.global().favoriteTooltip,
onPressed: () => _onFavoritePressed(index),
2021-04-10 06:28:12 +02:00
),
IconButton(
icon: const Icon(Icons.more_vert),
tooltip: L10n.global().detailsTooltip,
onPressed: _onDetailsPressed,
2021-04-10 06:28:12 +02:00
),
],
2021-04-10 06:28:12 +02:00
],
);
}
Widget _buildBottomAppBar(BuildContext context) {
2022-05-04 10:42:46 +02:00
final index =
_isViewerLoaded ? _viewerController.currentPage : widget.startIndex;
final file = _streamFilesView[index];
return ViewerBottomAppBar(
children: [
IconButton(
icon: const Icon(Icons.share_outlined),
tooltip: L10n.global().shareTooltip,
onPressed: () => _onSharePressed(context),
),
if (features.isSupportEnhancement &&
ImageEnhancer.isSupportedFormat(file)) ...[
IconButton(
icon: const Icon(Icons.tune_outlined),
tooltip: L10n.global().editTooltip,
onPressed: () => _onEditPressed(context),
2021-04-10 06:28:12 +02:00
),
IconButton(
icon: const Icon(Icons.auto_fix_high_outlined),
tooltip: L10n.global().enhanceTooltip,
onPressed: () => _onEnhancePressed(context),
),
],
IconButton(
icon: const Icon(Icons.download_outlined),
tooltip: L10n.global().downloadTooltip,
onPressed: _onDownloadPressed,
2021-04-10 06:28:12 +02:00
),
if (widget.fromCollection == null)
IconButton(
icon: const Icon(Icons.delete_outlined),
tooltip: L10n.global().deleteTooltip,
onPressed: () => _onDeletePressed(context),
),
],
2021-04-10 06:28:12 +02:00
);
}
2022-11-12 10:55:33 +01:00
Widget _buildPage(
BuildContext context, int index, Brightness originalBrightness) {
2021-04-10 06:28:12 +02:00
if (_pageStates[index] == null) {
_onCreateNewPage(context, index);
2021-07-23 22:05:57 +02:00
} else if (!_pageStates[index]!.scrollController.hasClients) {
2021-04-10 06:28:12 +02:00
// the page has been moved out of view and is now coming back
_log.fine("[_buildPage] Recreating page#$index");
_onRecreatePageAfterMovedOut(context, index);
}
if (kDebugMode) {
_log.info("[_buildPage] $index");
}
return FractionallySizedBox(
2021-07-31 17:38:26 +02:00
widthFactor: 1 / _viewportFraction,
2021-04-10 06:28:12 +02:00
child: NotificationListener<ScrollNotification>(
onNotification: (notif) => _onPageContentScrolled(notif, index),
2022-01-21 19:50:33 +01:00
child: ScrollConfiguration(
behavior: ScrollConfiguration.of(context).copyWith(scrollbars: false),
child: SingleChildScrollView(
controller: _pageStates[index]!.scrollController,
physics: !_isZoomed ? null : const NeverScrollableScrollPhysics(),
child: Stack(
children: [
_buildItemView(context, index),
IgnorePointer(
ignoring: !_isShowDetailPane,
child: Visibility(
visible: !_isZoomed,
child: AnimatedOpacity(
opacity: _isShowDetailPane ? 1 : 0,
duration: k.animationDurationNormal,
onEnd: () {
if (!_isShowDetailPane) {
setState(() {
_isDetailPaneActive = false;
});
}
},
2022-11-12 10:55:33 +01:00
child: Theme(
data: buildTheme(context, originalBrightness),
2022-11-12 10:55:33 +01:00
child: Builder(
builder: (context) {
return Container(
alignment: Alignment.topLeft,
constraints: BoxConstraints(
minHeight: MediaQuery.of(context).size.height,
),
decoration: BoxDecoration(
color: Theme.of(context).colorScheme.surface,
borderRadius: const BorderRadius.vertical(
top: Radius.circular(4),
),
),
margin: EdgeInsets.only(
top: _calcDetailPaneOffset(index),
),
// this visibility widget avoids loading the detail pane
// until it's actually opened, otherwise swiping between
// photos will slow down severely
child: Visibility(
visible: _isShowDetailPane,
child: ViewerDetailPane(
account: widget.account,
fd: _streamFilesView[index],
2023-04-17 18:15:29 +02:00
fromCollection: widget.fromCollection?.run(
(d) => ViewerSingleCollectionData(
d.collection, d.items[index])),
onRemoveFromCollectionPressed:
_onRemoveFromCollectionPressed,
2022-11-12 10:55:33 +01:00
onArchivePressed: _onArchivePressed,
onUnarchivePressed: _onUnarchivePressed,
onSlideshowPressed: _onSlideshowPressed,
onDeletePressed: _onDeletePressed,
2022-11-12 10:55:33 +01:00
),
),
);
},
2022-01-21 19:50:33 +01:00
),
2021-11-26 11:10:18 +01:00
),
2021-04-10 06:28:12 +02:00
),
),
),
2022-01-21 19:50:33 +01:00
],
),
2021-04-10 06:28:12 +02:00
),
),
),
);
}
Widget _buildItemView(BuildContext context, int index) {
final file = _streamFilesView[index];
2021-05-06 13:36:20 +02:00
if (file_util.isSupportedImageFormat(file)) {
final shouldPlayLivePhoto = _pageStates[index]!.shouldPlayLivePhoto;
if (shouldPlayLivePhoto) {
final livePhotoType = getLivePhotoTypeFromFile(file);
if (livePhotoType != null) {
return _buildLivePhotoView(context, index, livePhotoType);
} else {
_log.warning("[_buildItemView] Not a live photo");
}
}
2021-05-06 13:36:20 +02:00
return _buildImageView(context, index);
} else if (file_util.isSupportedVideoFormat(file)) {
return _buildVideoView(context, index);
} else {
_log.shout("[_buildItemView] Unknown file format: ${file.fdMime}");
2021-07-23 22:05:57 +02:00
_pageStates[index]!.itemHeight = 0;
2021-05-06 13:36:20 +02:00
return Container();
}
}
Widget _buildImageView(BuildContext context, int index) {
return RemoteImageViewer(
2021-05-02 16:18:26 +02:00
account: widget.account,
file: _streamFilesView[index],
2021-05-02 16:18:26 +02:00
canZoom: _canZoom(),
onLoaded: () => _onImageLoaded(index),
onHeightChanged: (height) => _updateItemHeight(index, height),
onZoomStarted: () {
setState(() {
_isZoomed = true;
});
},
onZoomEnded: () {
setState(() {
_isZoomed = false;
});
},
2021-04-10 06:28:12 +02:00
);
}
2021-05-06 13:36:20 +02:00
Widget _buildVideoView(BuildContext context, int index) {
return VideoViewer(
account: widget.account,
file: _streamFilesView[index],
2021-05-06 13:36:20 +02:00
onLoaded: () => _onVideoLoaded(index),
onHeightChanged: (height) => _updateItemHeight(index, height),
onPlay: _onVideoPlay,
onPause: _onVideoPause,
isControlVisible: _isShowAppBar && !_isDetailPaneActive,
canPlay: !_isDetailPaneActive,
2024-08-20 19:47:55 +02:00
canZoom: _canZoom(),
onZoomStarted: () {
setState(() {
_isZoomed = true;
});
},
onZoomEnded: () {
setState(() {
_isZoomed = false;
});
},
2021-05-06 13:36:20 +02:00
);
}
Widget _buildLivePhotoView(
BuildContext context, int index, LivePhotoType livePhotoType) {
return LivePhotoViewer(
account: widget.account,
file: _streamFilesView[index],
onLoaded: () => _onVideoLoaded(index),
onHeightChanged: (height) => _updateItemHeight(index, height),
canPlay: !_isDetailPaneActive,
livePhotoType: livePhotoType,
onLoadFailure: () {
if (mounted) {
setState(() {
_pageStates[index]!.shouldPlayLivePhoto = false;
});
}
},
);
}
2021-04-10 06:28:12 +02:00
bool _onPageContentScrolled(ScrollNotification notification, int index) {
if (!_canOpenDetailPane()) {
return false;
}
2021-12-08 21:18:48 +01:00
if (notification is ScrollStartNotification) {
_scrollStartPosition =
_pageStates[index]?.scrollController.position.pixels;
}
2021-04-10 06:28:12 +02:00
if (notification is ScrollEndNotification) {
2021-12-08 21:18:48 +01:00
_scrollStartPosition = null;
2021-07-23 22:05:57 +02:00
final scrollPos = _pageStates[index]!.scrollController.position;
2021-04-10 06:28:12 +02:00
if (scrollPos.pixels == 0) {
setState(() {
_onDetailPaneClosed();
});
} else if (scrollPos.pixels <
_calcDetailPaneOpenedScrollPosition(index) - 1) {
2021-04-10 06:28:12 +02:00
if (scrollPos.userScrollDirection == ScrollDirection.reverse) {
// upward, open the pane to its minimal size
Future.delayed(Duration.zero, () {
setState(() {
2021-07-31 17:38:26 +02:00
_openDetailPane(_viewerController.currentPage,
2021-04-10 06:28:12 +02:00
shouldAnimate: true);
});
});
} else if (scrollPos.userScrollDirection == ScrollDirection.forward) {
2021-04-10 06:28:12 +02:00
// downward, close the pane
Future.delayed(Duration.zero, () {
2021-07-31 17:38:26 +02:00
_closeDetailPane(_viewerController.currentPage,
2021-07-23 22:05:57 +02:00
shouldAnimate: true);
2021-04-10 06:28:12 +02:00
});
}
}
2021-11-26 11:10:18 +01:00
} else if (notification is ScrollUpdateNotification) {
if (!_isShowDetailPane) {
Future.delayed(Duration.zero, () {
setState(() {
_isShowDetailPane = true;
_isDetailPaneActive = true;
2021-11-26 11:10:18 +01:00
});
});
}
2021-04-10 06:28:12 +02:00
}
2021-12-08 21:18:48 +01:00
if (notification is OverscrollNotification) {
if (_scrollStartPosition == 0) {
// start at top
_overscrollSum += notification.overscroll;
2022-02-13 20:34:44 +01:00
if (_overscrollSum < -144) {
2021-12-08 21:18:48 +01:00
// and scroll downwards
Navigator.of(context).pop();
}
}
} else {
_overscrollSum = 0;
}
2021-04-10 06:28:12 +02:00
return false;
}
2021-05-02 16:18:26 +02:00
void _onImageLoaded(int index) {
2021-04-10 06:28:12 +02:00
// currently pageview doesn't pre-load pages, we do it manually
// don't pre-load if user already navigated away
2021-07-31 17:38:26 +02:00
if (_viewerController.currentPage == index &&
2021-07-23 22:05:57 +02:00
!_pageStates[index]!.hasLoaded) {
2021-05-02 16:18:26 +02:00
_log.info("[_onImageLoaded] Pre-loading nearby images");
2021-04-10 06:28:12 +02:00
if (index > 0) {
final prevFile = _streamFilesView[index - 1];
2021-05-06 13:36:20 +02:00
if (file_util.isSupportedImageFormat(prevFile)) {
RemoteImageViewer.preloadImage(widget.account, prevFile);
2021-05-06 13:36:20 +02:00
}
2021-04-10 06:28:12 +02:00
}
if (index + 1 < _streamFilesView.length) {
final nextFile = _streamFilesView[index + 1];
2021-05-06 13:36:20 +02:00
if (file_util.isSupportedImageFormat(nextFile)) {
RemoteImageViewer.preloadImage(widget.account, nextFile);
2021-05-06 13:36:20 +02:00
}
2021-04-10 06:28:12 +02:00
}
}
2021-08-05 12:38:43 +02:00
setState(() {
_pageStates[index]!.hasLoaded = true;
_isViewerLoaded = true;
});
2021-04-10 06:28:12 +02:00
}
2021-05-06 13:36:20 +02:00
void _onVideoLoaded(int index) {
2021-08-05 12:38:43 +02:00
setState(() {
_pageStates[index]!.hasLoaded = true;
_isViewerLoaded = true;
});
2021-05-06 13:36:20 +02:00
}
void _onVideoPlay() {
setState(() {
_setShowActionBar(false);
});
}
void _onVideoPause() {
setState(() {
_setShowActionBar(true);
});
}
2021-04-10 06:28:12 +02:00
/// Called when the page is being built for the first time
void _onCreateNewPage(BuildContext context, int index) {
_pageStates[index] = _PageState(ScrollController(
initialScrollOffset: _isShowDetailPane && !_isClosingDetailPane
? _calcDetailPaneOpenedScrollPosition(index)
: 0,
));
2021-04-10 06:28:12 +02:00
}
/// Called when the page is being built after previously moved out of view
void _onRecreatePageAfterMovedOut(BuildContext context, int index) {
_pageStates[index]!.setScrollController(ScrollController(
initialScrollOffset: _isShowDetailPane && !_isClosingDetailPane
? _calcDetailPaneOpenedScrollPosition(index)
: 0,
));
2021-04-10 06:28:12 +02:00
if (_isShowDetailPane && !_isClosingDetailPane) {
2022-06-20 13:49:58 +02:00
WidgetsBinding.instance.addPostFrameCallback((_) {
2022-12-11 06:01:13 +01:00
if (mounted && _pageStates[index]!.itemHeight != null) {
2021-04-10 06:28:12 +02:00
setState(() {
_openDetailPane(index);
});
}
});
} else {
2022-06-20 13:49:58 +02:00
WidgetsBinding.instance.addPostFrameCallback((_) {
2021-07-23 22:05:57 +02:00
_pageStates[index]!.scrollController.jumpTo(0);
2021-04-10 06:28:12 +02:00
});
}
}
void _onPlayMotionPhotosPressed(int index) {
setState(() {
_pageStates[index]!.shouldPlayLivePhoto = true;
});
}
void _onPauseMotionPhotosPressed(int index) {
setState(() {
_pageStates[index]!.shouldPlayLivePhoto = false;
});
}
2022-01-25 11:17:19 +01:00
Future<void> _onFavoritePressed(int index) async {
if (_pageStates[index]!.isProcessingFavorite) {
_log.fine("[_onFavoritePressed] Process ongoing, ignored");
return;
}
setState(() {
_pageStates[index]!.favoriteOverride = true;
});
_pageStates[index]!.isProcessingFavorite = true;
final fd = _streamFilesView[_viewerController.currentPage];
2022-01-25 11:17:19 +01:00
try {
await context.read<AccountController>().filesController.updateProperty(
[fd],
isFavorite: true,
errorBuilder: (fileIds) {
if (mounted) {
SnackBarManager().showSnackBar(SnackBar(
content: Text(L10n.global().unfavoriteFailureNotification),
duration: k.snackBarDurationNormal,
));
setState(() {
_pageStates[index]!.favoriteOverride = false;
});
}
return null;
},
);
} finally {
_pageStates[index]!.isProcessingFavorite = false;
2022-01-25 11:17:19 +01:00
}
}
Future<void> _onUnfavoritePressed(int index) async {
if (_pageStates[index]!.isProcessingFavorite) {
_log.fine("[_onUnfavoritePressed] Process ongoing, ignored");
return;
}
setState(() {
_pageStates[index]!.favoriteOverride = false;
});
_pageStates[index]!.isProcessingFavorite = true;
final fd = _streamFilesView[_viewerController.currentPage];
2022-01-25 11:17:19 +01:00
try {
await context.read<AccountController>().filesController.updateProperty(
[fd],
isFavorite: false,
errorBuilder: (fileIds) {
if (mounted) {
SnackBarManager().showSnackBar(SnackBar(
content: Text(L10n.global().unfavoriteFailureNotification),
duration: k.snackBarDurationNormal,
));
setState(() {
_pageStates[index]!.favoriteOverride = true;
});
}
return null;
},
);
} finally {
_pageStates[index]!.isProcessingFavorite = false;
2022-01-25 11:17:19 +01:00
}
}
2021-04-10 06:28:12 +02:00
void _onDetailsPressed() {
if (!_isDetailPaneActive) {
setState(() {
2021-07-31 17:38:26 +02:00
_openDetailPane(_viewerController.currentPage, shouldAnimate: true);
2021-04-10 06:28:12 +02:00
});
}
}
void _onSharePressed(BuildContext context) {
final c = KiwiContainer().resolve<DiContainer>();
final file = _streamFilesView[_viewerController.currentPage];
2021-10-08 20:39:53 +02:00
ShareHandler(
c,
2021-10-08 20:39:53 +02:00
context: context,
).shareFiles(widget.account, [file]);
}
2022-07-12 22:11:27 +02:00
void _onEditPressed(BuildContext context) {
final file = _streamFilesView[_viewerController.currentPage];
2022-07-12 22:11:27 +02:00
if (!file_util.isSupportedImageFormat(file)) {
_log.shout("[_onEditPressed] Video file not supported");
return;
}
_log.info("[_onEditPressed] Edit file: ${file.fdPath}");
2022-07-12 22:11:27 +02:00
Navigator.of(context).pushNamed(ImageEditor.routeName,
arguments: ImageEditorArguments(widget.account, file));
}
2022-05-04 10:42:46 +02:00
void _onEnhancePressed(BuildContext context) {
final file = _streamFilesView[_viewerController.currentPage];
2022-05-04 10:42:46 +02:00
if (!file_util.isSupportedImageFormat(file)) {
_log.shout("[_onEnhancePressed] Video file not supported");
return;
}
final c = KiwiContainer().resolve<DiContainer>();
2022-05-04 10:42:46 +02:00
_log.info("[_onEnhancePressed] Enhance file: ${file.fdPath}");
Navigator.of(context).pushNamed(ImageEnhancer.routeName,
arguments: ImageEnhancerArguments(
widget.account, file, c.pref.isSaveEditResultToServerOr()));
2022-05-04 10:42:46 +02:00
}
2021-09-28 22:56:44 +02:00
void _onDownloadPressed() {
final c = KiwiContainer().resolve<DiContainer>();
final file = _streamFilesView[_viewerController.currentPage];
_log.info("[_onDownloadPressed] Downloading file: ${file.fdPath}");
DownloadHandler(c).downloadFiles(widget.account, [file]);
2021-04-10 06:28:12 +02:00
}
void _onDeletePressed(BuildContext context) {
final index = _viewerController.currentPage;
final file = _streamFilesView[index];
_log.info("[_onDeletePressed] Removing file: ${file.fdPath}");
unawaited(RemoveSelectionHandler(
filesController: context.read<AccountController>().filesController,
)(
account: widget.account,
selection: [file],
isRemoveOpened: true,
isMoveToTrash: true,
));
_removeCurrentItemFromStream(context, index);
}
void _onArchivePressed(BuildContext context) {
final index = _viewerController.currentPage;
final file = _streamFilesView[index];
_log.info("[_onArchivePressed] Archive file: ${file.fdPath}");
context.read<AccountController>().filesController.updateProperty(
[file],
isArchived: const OrNull(true),
errorBuilder: (fileIds) {
if (mounted) {
SnackBarManager().showSnackBar(SnackBar(
content: Text(L10n.global().archiveSelectedFailureNotification(1)),
duration: k.snackBarDurationNormal,
));
}
return null;
},
);
_removeCurrentItemFromStream(context, index);
}
void _onUnarchivePressed(BuildContext context) {
final index = _viewerController.currentPage;
final file = _streamFilesView[index];
_log.info("[_onUnarchivePressed] Unarchive file: ${file.fdPath}");
context.read<AccountController>().filesController.updateProperty(
[file],
isArchived: const OrNull(false),
errorBuilder: (fileIds) {
if (mounted) {
SnackBarManager().showSnackBar(SnackBar(
content:
Text(L10n.global().unarchiveSelectedFailureNotification(1)),
duration: k.snackBarDurationNormal,
));
}
return null;
},
);
_removeCurrentItemFromStream(context, index);
}
2023-04-17 18:15:29 +02:00
Future<void> _onRemoveFromCollectionPressed(BuildContext context) async {
assert(CollectionAdapter.of(KiwiContainer().resolve<DiContainer>(),
widget.account, widget.fromCollection!.collection)
.isPermitted(CollectionCapability.manualItem));
final index = _viewerController.currentPage;
final file = _streamFilesView[index];
2023-04-17 18:15:29 +02:00
_log.info("[_onRemoveFromCollectionPressed] Remove file: ${file.fdPath}");
try {
final itemsController = _findCollectionItemsController(context);
final item = itemsController.stream.value.items
.whereType<CollectionFileItem>()
.firstWhere((i) => i.file.compareServerIdentity(file));
await itemsController.removeItems([item]);
} catch (e, stackTrace) {
_log.shout("[_onRemoveFromCollectionPressed] Failed while updating album",
e, stackTrace);
SnackBarManager().showSnackBar(SnackBar(
content: Text(L10n.global().removeSelectedFromAlbumFailureNotification),
duration: k.snackBarDurationNormal,
));
}
_removeCurrentItemFromStream(context, index);
}
void _removeCurrentItemFromStream(BuildContext context, int index) {
if (_streamFilesView.length == 1) {
2021-04-10 06:28:12 +02:00
Navigator.of(context).pop();
} else {
if (index >= _streamFilesView.length - 1) {
// last item, go back
_viewerController
.previousPage(
duration: k.animationDurationNormal,
curve: Curves.easeInOut,
)
.then((_) {
if (mounted) {
setState(() {
_streamFilesEditable.removeAt(index);
});
}
});
} else {
_viewerController
.nextPage(
duration: k.animationDurationNormal,
curve: Curves.easeInOut,
)
.then((_) {
if (mounted) {
setState(() {
_streamFilesEditable.removeAt(index);
});
// a page is removed, length - 1
_viewerController.jumpToPage(index);
}
});
}
2021-04-10 06:28:12 +02:00
}
}
2022-07-28 20:45:43 +02:00
Future<void> _onSlideshowPressed() async {
2021-09-14 23:00:24 +02:00
final result = await showDialog<SlideshowConfig>(
context: context,
builder: (_) => SlideshowDialog(
2021-10-27 22:40:54 +02:00
duration: Duration(seconds: Pref().getSlideshowDurationOr(5)),
isShuffle: Pref().isSlideshowShuffleOr(false),
isRepeat: Pref().isSlideshowRepeatOr(false),
2022-10-16 10:48:39 +02:00
isReverse: Pref().isSlideshowReverseOr(false),
2021-09-14 23:00:24 +02:00
),
);
if (result == null) {
return;
}
2022-07-28 18:59:26 +02:00
unawaited(Pref().setSlideshowDuration(result.duration.inSeconds));
unawaited(Pref().setSlideshowShuffle(result.isShuffle));
unawaited(Pref().setSlideshowRepeat(result.isRepeat));
2022-10-16 10:48:39 +02:00
unawaited(Pref().setSlideshowReverse(result.isReverse));
2022-07-28 18:59:26 +02:00
unawaited(
Navigator.of(context).pushNamed(
SlideshowViewer.routeName,
arguments: SlideshowViewerArguments(widget.account, widget.streamFiles,
_viewerController.currentPage, result),
),
2021-09-14 23:00:24 +02:00
);
}
2021-04-10 06:28:12 +02:00
double _calcDetailPaneOffset(int index) {
if (_pageStates[index]?.itemHeight == null) {
return MediaQuery.of(context).size.height;
} else {
2021-07-23 22:05:57 +02:00
return _pageStates[index]!.itemHeight! +
(MediaQuery.of(context).size.height -
_pageStates[index]!.itemHeight!) /
2021-04-10 06:28:12 +02:00
2 -
4;
}
}
double _calcDetailPaneOpenedScrollPosition(int index) {
// distance of the detail pane from the top edge
const distanceFromTop = 196;
return max(_calcDetailPaneOffset(index) - distanceFromTop, 0);
}
void _updateItemHeight(int index, double height) {
2021-07-23 22:05:57 +02:00
if (_pageStates[index]!.itemHeight != height) {
2021-04-10 06:28:12 +02:00
_log.fine("[_updateItemHeight] New height of item#$index: $height");
setState(() {
2021-07-23 22:05:57 +02:00
_pageStates[index]!.itemHeight = height;
2021-04-10 06:28:12 +02:00
if (_isDetailPaneActive) {
_openDetailPane(index);
}
});
}
}
void _setShowActionBar(bool flag) {
_isShowAppBar = flag;
if (flag) {
2021-09-15 17:13:38 +02:00
SystemChrome.setEnabledSystemUIMode(SystemUiMode.manual,
overlays: SystemUiOverlay.values);
} else {
2021-09-15 17:13:38 +02:00
SystemChrome.setEnabledSystemUIMode(SystemUiMode.immersive);
}
2021-04-10 06:28:12 +02:00
}
void _openDetailPane(int index, {bool shouldAnimate = false}) {
if (!_canOpenDetailPane()) {
_log.warning("[_openDetailPane] Can't open detail pane right now");
return;
}
_isShowDetailPane = true;
_isDetailPaneActive = true;
if (shouldAnimate) {
2021-07-23 22:05:57 +02:00
_pageStates[index]!.scrollController.animateTo(
2021-04-10 06:28:12 +02:00
_calcDetailPaneOpenedScrollPosition(index),
duration: k.animationDurationNormal,
curve: Curves.easeOut);
} else {
2021-07-23 22:05:57 +02:00
_pageStates[index]!
2021-04-10 06:28:12 +02:00
.scrollController
.jumpTo(_calcDetailPaneOpenedScrollPosition(index));
}
}
void _closeDetailPane(int index, {bool shouldAnimate = false}) {
_isClosingDetailPane = true;
if (shouldAnimate) {
2021-07-23 22:05:57 +02:00
_pageStates[index]!.scrollController.animateTo(0,
2021-04-10 06:28:12 +02:00
duration: k.animationDurationNormal, curve: Curves.easeOut);
}
}
void _onDetailPaneClosed() {
_isShowDetailPane = false;
_isClosingDetailPane = false;
}
2023-04-17 18:15:29 +02:00
CollectionItemsController _findCollectionItemsController(
BuildContext context) {
return context
.read<AccountController>()
.collectionsController
.stream
.value
.data
.firstWhere((d) =>
d.collection.compareIdentity(widget.fromCollection!.collection))
.controller;
}
2021-05-02 16:18:26 +02:00
bool _canSwitchPage() => !_isZoomed;
bool _canOpenDetailPane() => !_isZoomed;
2021-04-10 06:28:12 +02:00
bool _canZoom() => !_isDetailPaneActive;
List<FileDescriptor> get _streamFilesEditable {
if (!_isStreamFilesCopy) {
_streamFilesView = List.of(_streamFilesView);
_isStreamFilesCopy = true;
}
return _streamFilesView;
}
2021-04-10 06:28:12 +02:00
var _isShowAppBar = true;
var _isShowDetailPane = false;
var _isDetailPaneActive = false;
var _isClosingDetailPane = false;
2021-05-02 16:18:26 +02:00
var _isZoomed = false;
2021-04-10 06:28:12 +02:00
2021-07-31 17:38:26 +02:00
final _viewerController = HorizontalPageViewerController();
bool _isViewerLoaded = false;
2021-04-10 06:28:12 +02:00
final _pageStates = <int, _PageState>{};
2021-12-08 21:18:48 +01:00
double? _scrollStartPosition;
var _overscrollSum = 0.0;
late List<FileDescriptor> _streamFilesView;
bool _isStreamFilesCopy = false;
2023-05-30 18:48:58 +02:00
final _pageViewerKey = GlobalKey();
2021-07-31 17:38:26 +02:00
static const _viewportFraction = 1.05;
2021-04-10 06:28:12 +02:00
}
class _PageState {
_PageState(this.scrollController);
void setScrollController(ScrollController c) {
scrollController = c;
}
2021-04-10 06:28:12 +02:00
ScrollController scrollController;
2021-07-23 22:05:57 +02:00
double? itemHeight;
2021-05-02 16:18:26 +02:00
bool hasLoaded = false;
2022-01-25 11:17:19 +01:00
bool isProcessingFavorite = false;
bool? favoriteOverride;
bool shouldPlayLivePhoto = false;
2021-04-10 06:28:12 +02:00
}
2024-05-29 15:55:16 +02:00
class _AppBarTitle extends StatelessWidget {
const _AppBarTitle({
required this.file,
required this.isCentered,
});
@override
Widget build(BuildContext context) {
final locale = Localizations.localeOf(context).languageCode;
final localTime = file.fdDateTime.toLocal();
return Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment:
isCentered ? CrossAxisAlignment.center : CrossAxisAlignment.start,
children: [
Text(
(localTime.year == DateTime.now().year
? DateFormat.MMMd(locale)
: DateFormat.yMMMd(locale))
.format(localTime),
style: Theme.of(context).textTheme.titleMedium,
),
Text(
DateFormat.jm(locale).format(localTime),
style: Theme.of(context).textTheme.bodySmall,
),
],
);
}
final FileDescriptor file;
final bool isCentered;
}