import 'dart:async'; import 'package:cached_network_image/cached_network_image.dart'; import 'package:cached_network_image_platform_interface/cached_network_image_platform_interface.dart'; import 'package:copy_with/copy_with.dart'; import 'package:flutter/material.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_staggered_grid_view/flutter_staggered_grid_view.dart'; import 'package:logging/logging.dart'; import 'package:nc_photos/account.dart'; import 'package:nc_photos/app_localizations.dart'; import 'package:nc_photos/bloc_util.dart'; import 'package:nc_photos/cache_manager_util.dart'; import 'package:nc_photos/controller/account_controller.dart'; import 'package:nc_photos/controller/collections_controller.dart'; import 'package:nc_photos/controller/pref_controller.dart'; import 'package:nc_photos/entity/album/provider.dart'; import 'package:nc_photos/entity/collection.dart'; import 'package:nc_photos/entity/collection/content_provider/album.dart'; import 'package:nc_photos/entity/collection/content_provider/nc_album.dart'; import 'package:nc_photos/entity/collection/util.dart' as collection_util; import 'package:nc_photos/entity/pref.dart'; import 'package:nc_photos/exception_event.dart'; import 'package:nc_photos/exception_util.dart' as exception_util; import 'package:nc_photos/k.dart' as k; import 'package:nc_photos/np_api_util.dart'; import 'package:nc_photos/platform/features.dart' as features; import 'package:nc_photos/snack_bar_manager.dart'; import 'package:nc_photos/theme.dart'; import 'package:nc_photos/theme/dimension.dart'; import 'package:nc_photos/widget/album_importer.dart'; import 'package:nc_photos/widget/archive_browser.dart'; import 'package:nc_photos/widget/collection_browser.dart'; import 'package:nc_photos/widget/collection_grid_item.dart'; import 'package:nc_photos/widget/enhanced_photo_browser.dart'; import 'package:nc_photos/widget/fancy_option_picker.dart'; import 'package:nc_photos/widget/home_app_bar.dart'; import 'package:nc_photos/widget/navigation_bar_blur_filter.dart'; import 'package:nc_photos/widget/new_collection_dialog.dart'; import 'package:nc_photos/widget/page_visibility_mixin.dart'; import 'package:nc_photos/widget/selectable_item_list.dart'; import 'package:nc_photos/widget/selection_app_bar.dart'; import 'package:nc_photos/widget/sharing_browser.dart'; import 'package:nc_photos/widget/trashbin_browser.dart'; import 'package:np_codegen/np_codegen.dart'; import 'package:to_string/to_string.dart'; part 'home_collections.g.dart'; part 'home_collections/bloc.dart'; part 'home_collections/state_event.dart'; part 'home_collections/type.dart'; typedef _BlocBuilder = BlocBuilder<_Bloc, _State>; /// Show and manage a list of [Collection]s class HomeCollections extends StatelessWidget { const HomeCollections({ super.key, }); @override Widget build(BuildContext context) { return BlocProvider( create: (context) => _Bloc( account: context.read().account, controller: context.read().collectionsController, prefController: context.read(), ), child: const _WrappedHomeCollections(), ); } } class _WrappedHomeCollections extends StatefulWidget { const _WrappedHomeCollections(); @override State createState() => _WrappedHomeCollectionsState(); } @npLog class _WrappedHomeCollectionsState extends State<_WrappedHomeCollections> with RouteAware, PageVisibilityMixin { @override void initState() { super.initState(); _bloc.add(const _LoadCollections()); } @override Widget build(BuildContext context) { return MultiBlocListener( listeners: [ BlocListener<_Bloc, _State>( listenWhen: (previous, current) => previous.collections != current.collections, listener: (context, state) { _bloc.add(_TransformItems(state.collections)); }, ), BlocListener<_Bloc, _State>( listenWhen: (previous, current) => previous.error != current.error, listener: (context, state) { if (state.error != null && isPageVisible()) { SnackBarManager().showSnackBar(SnackBar( content: Text(exception_util.toUserString(state.error!.error)), duration: k.snackBarDurationNormal, )); } }, ), BlocListener<_Bloc, _State>( listenWhen: (previous, current) => previous.removeError != current.removeError, listener: (context, state) { if (state.removeError != null && isPageVisible()) { SnackBarManager().showSnackBar(SnackBar( content: Text(L10n.global().removeCollectionsFailedNotification), duration: k.snackBarDurationNormal, )); } }, ), ], child: Stack( children: [ RefreshIndicator( onRefresh: () async { _bloc.add(const _ReloadCollections()); await _bloc.stream.first; }, child: CustomScrollView( slivers: [ _BlocBuilder( buildWhen: (previous, current) => previous.selectedItems.isEmpty != current.selectedItems.isEmpty, builder: (context, state) => state.selectedItems.isEmpty ? const _AppBar() : const _SelectionAppBar(), ), SliverPadding( padding: const EdgeInsets.symmetric(horizontal: 8), sliver: _BlocBuilder( buildWhen: (previous, current) => previous.selectedItems.isEmpty != current.selectedItems.isEmpty, builder: (context, state) => _ButtonGrid( account: _bloc.account, isEnabled: state.selectedItems.isEmpty, onSharingPressed: () { Navigator.of(context).pushNamed( SharingBrowser.routeName, arguments: SharingBrowserArguments(_bloc.account)); }, onEnhancedPhotosPressed: () { Navigator.of(context).pushNamed( EnhancedPhotoBrowser.routeName, arguments: const EnhancedPhotoBrowserArguments(null)); }, onArchivePressed: () { Navigator.of(context).pushNamed( ArchiveBrowser.routeName, arguments: ArchiveBrowserArguments(_bloc.account)); }, onTrashbinPressed: () { Navigator.of(context).pushNamed( TrashbinBrowser.routeName, arguments: TrashbinBrowserArguments(_bloc.account)); }, onNewCollectionPressed: () { _onNewCollectionPressed(context); }, ), ), ), const SliverToBoxAdapter( child: SizedBox(height: 8), ), _BlocBuilder( buildWhen: (previous, current) => previous.transformedItems != current.transformedItems || previous.selectedItems != current.selectedItems, builder: (context, state) => SliverPadding( padding: const EdgeInsets.symmetric(horizontal: 8), sliver: SelectableItemList( maxCrossAxisExtent: 256, childBorderRadius: BorderRadius.zero, indicatorAlignment: const Alignment(-.92, -.92), items: state.transformedItems, itemBuilder: (_, __, metadata) { final item = metadata as _Item; return _ItemView( account: _bloc.account, item: item, ); }, staggeredTileBuilder: (_, __) => const StaggeredTile.count(1, 1), selectedItems: state.selectedItems, onSelectionChange: (_, selected) { _bloc.add(_SetSelectedItems(items: selected.cast())); }, onItemTap: (context, _, metadata) { final item = metadata as _Item; Navigator.of(context).pushNamed( CollectionBrowser.routeName, arguments: CollectionBrowserArguments(item.collection), ); }, ), ), ), SliverToBoxAdapter( child: SizedBox( height: AppDimension.of(context).homeBottomAppBarHeight, ), ), ], ), ), Align( alignment: Alignment.bottomCenter, child: NavigationBarBlurFilter( height: AppDimension.of(context).homeBottomAppBarHeight, ), ), ], ), ); } Future _onNewCollectionPressed(BuildContext context) async { try { final collection = await showDialog( context: context, builder: (_) => NewCollectionDialog( account: _bloc.account, ), ); if (collection == null) { return; } // Right now we don't have a way to add photos inside the // CollectionBrowser, eventually we should add that and remove this // branching if (collection.isDynamicCollection) { // open the newly created collection unawaited(Navigator.of(context).pushNamed( CollectionBrowser.routeName, arguments: CollectionBrowserArguments(collection), )); } } catch (e, stacktrace) { _log.shout("[_onNewCollectionPressed] Failed", e, stacktrace); SnackBarManager().showSnackBar(SnackBar( content: Text(L10n.global().createCollectionFailureNotification), duration: k.snackBarDurationNormal, )); } } late final _Bloc _bloc = context.read(); } class _AppBar extends StatelessWidget { const _AppBar(); @override Widget build(BuildContext context) { return _BlocBuilder( buildWhen: (previous, current) => previous.isLoading != current.isLoading, builder: (context, state) => HomeSliverAppBar( account: context.read<_Bloc>().account, isShowProgressIcon: state.isLoading, menuActions: [ PopupMenuItem( value: _menuValueSort, child: Text(L10n.global().sortTooltip), ), PopupMenuItem( value: _menuValueImport, child: Text(L10n.global().importFoldersTooltip), ), ], onSelectedMenuActions: (option) { switch (option) { case _menuValueSort: _onSortPressed(context); break; case _menuValueImport: _onImportPressed(context); break; } }, ), ); } Future _onSortPressed(BuildContext context) async { final sort = context.read<_Bloc>().state.sort; final result = await showDialog( context: context, builder: (context) => FancyOptionPicker( title: Text(L10n.global().sortOptionDialogTitle), items: [ FancyOptionPickerItem( label: L10n.global().sortOptionTimeDescendingLabel, isSelected: sort == collection_util.CollectionSort.dateDescending, onSelect: () { Navigator.of(context) .pop(collection_util.CollectionSort.dateDescending); }, ), FancyOptionPickerItem( label: L10n.global().sortOptionTimeAscendingLabel, isSelected: sort == collection_util.CollectionSort.dateAscending, onSelect: () { Navigator.of(context) .pop(collection_util.CollectionSort.dateAscending); }, ), FancyOptionPickerItem( label: L10n.global().sortOptionAlbumNameLabel, isSelected: sort == collection_util.CollectionSort.nameAscending, onSelect: () { Navigator.of(context) .pop(collection_util.CollectionSort.nameAscending); }, ), FancyOptionPickerItem( label: L10n.global().sortOptionAlbumNameDescendingLabel, isSelected: sort == collection_util.CollectionSort.nameDescending, onSelect: () { Navigator.of(context) .pop(collection_util.CollectionSort.nameDescending); }, ), ], ), ); if (result == null) { return; } context.read<_Bloc>().add(_SetCollectionSort(result)); } void _onImportPressed(BuildContext context) { Navigator.of(context).pushNamed(AlbumImporter.routeName, arguments: AlbumImporterArguments(context.read<_Bloc>().account)); } static const _menuValueImport = 0; static const _menuValueSort = 1; } @npLog class _SelectionAppBar extends StatelessWidget { const _SelectionAppBar(); @override Widget build(BuildContext context) { return _BlocBuilder( buildWhen: (previous, current) => previous.selectedItems != current.selectedItems, builder: (context, state) => SelectionAppBar( count: state.selectedItems.length, onClosePressed: () { context.read<_Bloc>().add(const _SetSelectedItems(items: {})); }, actions: [ IconButton( icon: const Icon(Icons.delete), tooltip: L10n.global().deleteTooltip, onPressed: () { context.read<_Bloc>().add(const _RemoveSelectedItems()); }, ), ], ), ); } } class _ButtonGrid extends StatelessWidget { const _ButtonGrid({ required this.account, required this.isEnabled, this.onSharingPressed, this.onEnhancedPhotosPressed, this.onArchivePressed, this.onTrashbinPressed, this.onNewCollectionPressed, }); @override Widget build(BuildContext context) { // needed to workaround a scrolling bug when there are more than one // SliverStaggeredGrids in a CustomScrollView // see: https://github.com/letsar/flutter_staggered_grid_view/issues/98 and // https://github.com/letsar/flutter_staggered_grid_view/issues/265 return SliverToBoxAdapter( child: StaggeredGridView.extent( shrinkWrap: true, physics: const NeverScrollableScrollPhysics(), padding: const EdgeInsets.all(0), maxCrossAxisExtent: 256, staggeredTiles: List.filled(5, const StaggeredTile.fit(1)), children: [ _ButtonGridItemView( icon: Icons.share_outlined, label: L10n.global().collectionSharingLabel, isShowIndicator: AccountPref.of(account).hasNewSharedAlbumOr(), isEnabled: isEnabled, onTap: () { onSharingPressed?.call(); }, ), if (features.isSupportEnhancement) _ButtonGridItemView( icon: Icons.auto_fix_high_outlined, label: L10n.global().collectionEditedPhotosLabel, isEnabled: isEnabled, onTap: () { onEnhancedPhotosPressed?.call(); }, ), _ButtonGridItemView( icon: Icons.archive_outlined, label: L10n.global().albumArchiveLabel, isEnabled: isEnabled, onTap: () { onArchivePressed?.call(); }, ), _ButtonGridItemView( icon: Icons.delete_outlined, label: L10n.global().albumTrashLabel, isEnabled: isEnabled, onTap: () { onTrashbinPressed?.call(); }, ), _ButtonGridItemView( icon: Icons.add, label: L10n.global().createCollectionTooltip, isEnabled: isEnabled, onTap: () { onNewCollectionPressed?.call(); }, ), ], ), ); } final Account account; final bool isEnabled; final VoidCallback? onSharingPressed; final VoidCallback? onEnhancedPhotosPressed; final VoidCallback? onArchivePressed; final VoidCallback? onTrashbinPressed; final VoidCallback? onNewCollectionPressed; } class _ButtonGridItemView extends StatelessWidget { const _ButtonGridItemView({ required this.icon, required this.label, this.isShowIndicator = false, required this.isEnabled, this.onTap, }); @override Widget build(BuildContext context) { return Padding( padding: const EdgeInsets.all(4), child: ActionChip( materialTapTargetSize: MaterialTapTargetSize.shrinkWrap, labelPadding: const EdgeInsetsDirectional.fromSTEB(8, 0, 0, 0), // specify icon size explicitly to workaround size flickering during // theme transition avatar: Icon(icon, size: 18), label: Row( children: [ Expanded( child: Text(label), ), if (isShowIndicator) Icon( Icons.circle, color: Theme.of(context).colorScheme.tertiary, size: 8, ), ], ), onPressed: isEnabled ? onTap : null, ), ); } final IconData icon; final String label; final bool isShowIndicator; final bool isEnabled; final VoidCallback? onTap; } class _ItemView extends StatelessWidget { const _ItemView({ required this.account, required this.item, }); @override Widget build(BuildContext context) { Widget? icon; switch (item.itemType) { case _ItemType.ncAlbum: icon = const Icon(Icons.cloud); break; case _ItemType.album: icon = null; break; case _ItemType.tagAlbum: icon = const Icon(Icons.local_offer); break; case _ItemType.dirAlbum: icon = const Icon(Icons.folder); break; } String subtitle = ""; if (item.isShared) { subtitle = "${L10n.global().albumSharedLabel} | "; } subtitle += item.subtitle ?? ""; return CollectionGridItem( cover: _CollectionCover( account: account, url: item.coverUrl, ), title: item.name, subtitle: subtitle, icon: icon, ); } final Account account; final _Item item; } class _CollectionCover extends StatelessWidget { const _CollectionCover({ required this.account, required this.url, }); @override Widget build(BuildContext context) { return ClipRRect( borderRadius: BorderRadius.circular(8), child: Container( color: Theme.of(context).listPlaceholderBackgroundColor, constraints: const BoxConstraints.expand(), child: url != null ? FittedBox( clipBehavior: Clip.hardEdge, fit: BoxFit.cover, child: CachedNetworkImage( cacheManager: CoverCacheManager.inst, imageUrl: url!, httpHeaders: { "Authorization": AuthUtil.fromAccount(account).toHeaderValue(), }, fadeInDuration: const Duration(), filterQuality: FilterQuality.high, errorWidget: (context, url, error) { // just leave it empty return Container(); }, imageRenderMethodForWeb: ImageRenderMethodForWeb.HttpGet, ), ) : Icon( Icons.panorama, color: Theme.of(context).listPlaceholderForegroundColor, size: 88, ), ), ); } final Account account; final String? url; }