import 'dart:async'; import 'dart:math'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter/rendering.dart'; import 'package:flutter/services.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:intl/intl.dart'; import 'package:kiwi/kiwi.dart'; import 'package:logging/logging.dart'; import 'package:nc_photos/account.dart'; import 'package:nc_photos/app_localizations.dart'; import 'package:nc_photos/asset.dart'; import 'package:nc_photos/controller/account_controller.dart'; import 'package:nc_photos/controller/collection_items_controller.dart'; import 'package:nc_photos/di_container.dart'; import 'package:nc_photos/download_handler.dart'; 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'; import 'package:nc_photos/entity/file_util.dart' as file_util; import 'package:nc_photos/entity/pref.dart'; import 'package:nc_photos/flutter_util.dart'; import 'package:nc_photos/k.dart' as k; import 'package:nc_photos/live_photo_util.dart'; import 'package:nc_photos/object_extension.dart'; import 'package:nc_photos/platform/features.dart' as features; import 'package:nc_photos/share_handler.dart'; import 'package:nc_photos/snack_bar_manager.dart'; import 'package:nc_photos/theme.dart'; import 'package:nc_photos/widget/app_intermediate_circular_progress_indicator.dart'; import 'package:nc_photos/widget/disposable.dart'; import 'package:nc_photos/widget/handler/remove_selection_handler.dart'; import 'package:nc_photos/widget/horizontal_page_viewer.dart'; import 'package:nc_photos/widget/image_editor.dart'; import 'package:nc_photos/widget/image_enhancer.dart'; 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'; import 'package:nc_photos/widget/slideshow_dialog.dart'; import 'package:nc_photos/widget/slideshow_viewer.dart'; import 'package:nc_photos/widget/video_viewer.dart'; import 'package:nc_photos/widget/viewer_bottom_app_bar.dart'; import 'package:nc_photos/widget/viewer_detail_pane.dart'; import 'package:nc_photos/widget/viewer_mixin.dart'; import 'package:np_codegen/np_codegen.dart'; import 'package:np_common/or_null.dart'; import 'package:np_platform_util/np_platform_util.dart'; import 'package:video_player_platform_interface/video_player_platform_interface.dart'; part 'viewer.g.dart'; class ViewerCollectionData { const ViewerCollectionData(this.collection, this.items); final Collection collection; final List items; } class ViewerArguments { const ViewerArguments( this.account, this.streamFiles, this.startIndex, { this.fromCollection, }); final Account account; final List streamFiles; final int startIndex; final ViewerCollectionData? fromCollection; } class Viewer extends StatefulWidget { static const routeName = "/viewer"; static Route buildRoute(ViewerArguments args) => CustomizableMaterialPageRoute( transitionDuration: k.heroDurationNormal, reverseTransitionDuration: k.heroDurationNormal, builder: (_) => Viewer.fromArgs(args), ); const Viewer({ super.key, required this.account, required this.streamFiles, required this.startIndex, this.fromCollection, }); Viewer.fromArgs(ViewerArguments args, {Key? key}) : this( key: key, account: args.account, streamFiles: args.streamFiles, startIndex: args.startIndex, fromCollection: args.fromCollection, ); @override createState() => _ViewerState(); final Account account; final List streamFiles; final int startIndex; /// Data of the collection these files belongs to, or null final ViewerCollectionData? fromCollection; } @npLog class _ViewerState extends State with DisposableManagerMixin, ViewerControllersMixin { @override void initState() { super.initState(); _streamFilesView = widget.streamFiles; } @override build(BuildContext context) { final originalBrightness = Theme.of(context).brightness; return Theme( data: buildDarkTheme(context), child: AnnotatedRegion( 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, body: Builder( builder: (context) => _buildContent(context, originalBrightness), ), ), ), ); } Widget _buildContent(BuildContext context, Brightness originalBrightness) { return GestureDetector( onTap: () { setState(() { _setShowActionBar(!_isShowAppBar); }); }, child: Stack( children: [ Container(color: Colors.black), if (!_isViewerLoaded || _pageStates[_viewerController.currentPage]?.hasLoaded != true) const Align( alignment: Alignment.center, child: AppIntermediateCircularProgressIndicator(), ), HorizontalPageViewer( key: _pageViewerKey, pageCount: _streamFilesView.length, pageBuilder: (context, i) => _buildPage(context, i, originalBrightness), initialPage: widget.startIndex, controller: _viewerController, viewportFraction: _viewportFraction, canSwitchPage: _canSwitchPage(), onPageChanged: (from, to) { setState(() { _pageStates[from]?.shouldPlayLivePhoto = false; }); }, ), 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), ], ), ), ), ], ), ); } AppBar _buildAppBar(BuildContext context) { final index = _isViewerLoaded ? _viewerController.currentPage : widget.startIndex; final file = _streamFilesView[index]; final isCentered = getRawPlatform() == NpPlatform.iOs; return AppBar( backgroundColor: Colors.transparent, elevation: 0, 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), ), IconButton( icon: const Icon(Icons.more_vert), tooltip: L10n.global().detailsTooltip, onPressed: _onDetailsPressed, ), ], ], ); } Widget _buildBottomAppBar(BuildContext context) { 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), ), 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, ), if (widget.fromCollection == null) IconButton( icon: const Icon(Icons.delete_outlined), tooltip: L10n.global().deleteTooltip, onPressed: () => _onDeletePressed(context), ), ], ); } Widget _buildPage( BuildContext context, int index, Brightness originalBrightness) { if (_pageStates[index] == null) { _onCreateNewPage(context, index); } else if (!_pageStates[index]!.scrollController.hasClients) { // 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( widthFactor: 1 / _viewportFraction, child: NotificationListener( onNotification: (notif) => _onPageContentScrolled(notif, index), 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; }); } }, child: Theme( data: buildTheme(context, originalBrightness), 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], fromCollection: widget.fromCollection?.run( (d) => ViewerSingleCollectionData( d.collection, d.items[index])), onRemoveFromCollectionPressed: _onRemoveFromCollectionPressed, onArchivePressed: _onArchivePressed, onUnarchivePressed: _onUnarchivePressed, onSlideshowPressed: _onSlideshowPressed, onDeletePressed: _onDeletePressed, ), ), ); }, ), ), ), ), ), ], ), ), ), ), ); } Widget _buildItemView(BuildContext context, int index) { final file = _streamFilesView[index]; 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"); } } return _buildImageView(context, index); } else if (file_util.isSupportedVideoFormat(file)) { return _buildVideoView(context, index); } else { _log.shout("[_buildItemView] Unknown file format: ${file.fdMime}"); _pageStates[index]!.itemHeight = 0; return Container(); } } Widget _buildImageView(BuildContext context, int index) { return RemoteImageViewer( account: widget.account, file: _streamFilesView[index], canZoom: _canZoom(), onLoaded: () => _onImageLoaded(index), onHeightChanged: (height) => _updateItemHeight(index, height), onZoomStarted: () { setState(() { _isZoomed = true; }); }, onZoomEnded: () { setState(() { _isZoomed = false; }); }, ); } Widget _buildVideoView(BuildContext context, int index) { return VideoViewer( account: widget.account, file: _streamFilesView[index], onLoaded: () => _onVideoLoaded(index), onHeightChanged: (height) => _updateItemHeight(index, height), onPlay: _onVideoPlay, onPause: _onVideoPause, isControlVisible: _isShowAppBar && !_isDetailPaneActive, canPlay: !_isDetailPaneActive, canZoom: _canZoom(), onZoomStarted: () { setState(() { _isZoomed = true; }); }, onZoomEnded: () { setState(() { _isZoomed = false; }); }, ); } 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; }); } }, ); } bool _onPageContentScrolled(ScrollNotification notification, int index) { if (!_canOpenDetailPane()) { return false; } if (notification is ScrollStartNotification) { _scrollStartPosition = _pageStates[index]?.scrollController.position.pixels; } if (notification is ScrollEndNotification) { _scrollStartPosition = null; final scrollPos = _pageStates[index]!.scrollController.position; if (scrollPos.pixels == 0) { setState(() { _onDetailPaneClosed(); }); } else if (scrollPos.pixels < _calcDetailPaneOpenedScrollPosition(index) - 1) { if (scrollPos.userScrollDirection == ScrollDirection.reverse) { // upward, open the pane to its minimal size Future.delayed(Duration.zero, () { setState(() { _openDetailPane(_viewerController.currentPage, shouldAnimate: true); }); }); } else if (scrollPos.userScrollDirection == ScrollDirection.forward) { // downward, close the pane Future.delayed(Duration.zero, () { _closeDetailPane(_viewerController.currentPage, shouldAnimate: true); }); } } } else if (notification is ScrollUpdateNotification) { if (!_isShowDetailPane) { Future.delayed(Duration.zero, () { setState(() { _isShowDetailPane = true; _isDetailPaneActive = true; }); }); } } if (notification is OverscrollNotification) { if (_scrollStartPosition == 0) { // start at top _overscrollSum += notification.overscroll; if (_overscrollSum < -144) { // and scroll downwards Navigator.of(context).pop(); } } } else { _overscrollSum = 0; } return false; } void _onImageLoaded(int index) { // currently pageview doesn't pre-load pages, we do it manually // don't pre-load if user already navigated away if (_viewerController.currentPage == index && !_pageStates[index]!.hasLoaded) { _log.info("[_onImageLoaded] Pre-loading nearby images"); if (index > 0) { final prevFile = _streamFilesView[index - 1]; if (file_util.isSupportedImageFormat(prevFile)) { RemoteImageViewer.preloadImage(widget.account, prevFile); } } if (index + 1 < _streamFilesView.length) { final nextFile = _streamFilesView[index + 1]; if (file_util.isSupportedImageFormat(nextFile)) { RemoteImageViewer.preloadImage(widget.account, nextFile); } } } setState(() { _pageStates[index]!.hasLoaded = true; _isViewerLoaded = true; }); } void _onVideoLoaded(int index) { setState(() { _pageStates[index]!.hasLoaded = true; _isViewerLoaded = true; }); } void _onVideoPlay() { setState(() { _setShowActionBar(false); }); } void _onVideoPause() { setState(() { _setShowActionBar(true); }); } /// 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, )); } /// 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, )); if (_isShowDetailPane && !_isClosingDetailPane) { WidgetsBinding.instance.addPostFrameCallback((_) { if (mounted && _pageStates[index]!.itemHeight != null) { setState(() { _openDetailPane(index); }); } }); } else { WidgetsBinding.instance.addPostFrameCallback((_) { _pageStates[index]!.scrollController.jumpTo(0); }); } } void _onPlayMotionPhotosPressed(int index) { setState(() { _pageStates[index]!.shouldPlayLivePhoto = true; }); } void _onPauseMotionPhotosPressed(int index) { setState(() { _pageStates[index]!.shouldPlayLivePhoto = false; }); } Future _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]; try { await context.read().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; } } Future _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]; try { await context.read().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; } } void _onDetailsPressed() { if (!_isDetailPaneActive) { setState(() { _openDetailPane(_viewerController.currentPage, shouldAnimate: true); }); } } void _onSharePressed(BuildContext context) { final c = KiwiContainer().resolve(); final file = _streamFilesView[_viewerController.currentPage]; ShareHandler( c, context: context, ).shareFiles(widget.account, [file]); } void _onEditPressed(BuildContext context) { final file = _streamFilesView[_viewerController.currentPage]; if (!file_util.isSupportedImageFormat(file)) { _log.shout("[_onEditPressed] Video file not supported"); return; } _log.info("[_onEditPressed] Edit file: ${file.fdPath}"); Navigator.of(context).pushNamed(ImageEditor.routeName, arguments: ImageEditorArguments(widget.account, file)); } void _onEnhancePressed(BuildContext context) { final file = _streamFilesView[_viewerController.currentPage]; if (!file_util.isSupportedImageFormat(file)) { _log.shout("[_onEnhancePressed] Video file not supported"); return; } final c = KiwiContainer().resolve(); _log.info("[_onEnhancePressed] Enhance file: ${file.fdPath}"); Navigator.of(context).pushNamed(ImageEnhancer.routeName, arguments: ImageEnhancerArguments( widget.account, file, c.pref.isSaveEditResultToServerOr())); } void _onDownloadPressed() { final c = KiwiContainer().resolve(); final file = _streamFilesView[_viewerController.currentPage]; _log.info("[_onDownloadPressed] Downloading file: ${file.fdPath}"); DownloadHandler(c).downloadFiles(widget.account, [file]); } void _onDeletePressed(BuildContext context) { final index = _viewerController.currentPage; final file = _streamFilesView[index]; _log.info("[_onDeletePressed] Removing file: ${file.fdPath}"); unawaited(RemoveSelectionHandler( filesController: context.read().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().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().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); } Future _onRemoveFromCollectionPressed(BuildContext context) async { assert(CollectionAdapter.of(KiwiContainer().resolve(), widget.account, widget.fromCollection!.collection) .isPermitted(CollectionCapability.manualItem)); final index = _viewerController.currentPage; final file = _streamFilesView[index]; _log.info("[_onRemoveFromCollectionPressed] Remove file: ${file.fdPath}"); try { final itemsController = _findCollectionItemsController(context); final item = itemsController.stream.value.items .whereType() .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) { 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); } }); } } } Future _onSlideshowPressed() async { final result = await showDialog( context: context, builder: (_) => SlideshowDialog( duration: Duration(seconds: Pref().getSlideshowDurationOr(5)), isShuffle: Pref().isSlideshowShuffleOr(false), isRepeat: Pref().isSlideshowRepeatOr(false), isReverse: Pref().isSlideshowReverseOr(false), ), ); if (result == null) { return; } unawaited(Pref().setSlideshowDuration(result.duration.inSeconds)); unawaited(Pref().setSlideshowShuffle(result.isShuffle)); unawaited(Pref().setSlideshowRepeat(result.isRepeat)); unawaited(Pref().setSlideshowReverse(result.isReverse)); unawaited( Navigator.of(context).pushNamed( SlideshowViewer.routeName, arguments: SlideshowViewerArguments(widget.account, widget.streamFiles, _viewerController.currentPage, result), ), ); } double _calcDetailPaneOffset(int index) { if (_pageStates[index]?.itemHeight == null) { return MediaQuery.of(context).size.height; } else { return _pageStates[index]!.itemHeight! + (MediaQuery.of(context).size.height - _pageStates[index]!.itemHeight!) / 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) { if (_pageStates[index]!.itemHeight != height) { _log.fine("[_updateItemHeight] New height of item#$index: $height"); setState(() { _pageStates[index]!.itemHeight = height; if (_isDetailPaneActive) { _openDetailPane(index); } }); } } void _setShowActionBar(bool flag) { _isShowAppBar = flag; if (flag) { SystemChrome.setEnabledSystemUIMode(SystemUiMode.manual, overlays: SystemUiOverlay.values); } else { SystemChrome.setEnabledSystemUIMode(SystemUiMode.immersive); } } 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) { _pageStates[index]!.scrollController.animateTo( _calcDetailPaneOpenedScrollPosition(index), duration: k.animationDurationNormal, curve: Curves.easeOut); } else { _pageStates[index]! .scrollController .jumpTo(_calcDetailPaneOpenedScrollPosition(index)); } } void _closeDetailPane(int index, {bool shouldAnimate = false}) { _isClosingDetailPane = true; if (shouldAnimate) { _pageStates[index]!.scrollController.animateTo(0, duration: k.animationDurationNormal, curve: Curves.easeOut); } } void _onDetailPaneClosed() { _isShowDetailPane = false; _isClosingDetailPane = false; } CollectionItemsController _findCollectionItemsController( BuildContext context) { return context .read() .collectionsController .stream .value .data .firstWhere((d) => d.collection.compareIdentity(widget.fromCollection!.collection)) .controller; } bool _canSwitchPage() => !_isZoomed; bool _canOpenDetailPane() => !_isZoomed; bool _canZoom() => !_isDetailPaneActive; List get _streamFilesEditable { if (!_isStreamFilesCopy) { _streamFilesView = List.of(_streamFilesView); _isStreamFilesCopy = true; } return _streamFilesView; } var _isShowAppBar = true; var _isShowDetailPane = false; var _isDetailPaneActive = false; var _isClosingDetailPane = false; var _isZoomed = false; final _viewerController = HorizontalPageViewerController(); bool _isViewerLoaded = false; final _pageStates = {}; double? _scrollStartPosition; var _overscrollSum = 0.0; late List _streamFilesView; bool _isStreamFilesCopy = false; final _pageViewerKey = GlobalKey(); static const _viewportFraction = 1.05; } class _PageState { _PageState(this.scrollController); void setScrollController(ScrollController c) { scrollController = c; } ScrollController scrollController; double? itemHeight; bool hasLoaded = false; bool isProcessingFavorite = false; bool? favoriteOverride; bool shouldPlayLivePhoto = false; } 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; }