mirror of
				https://github.com/immich-app/immich.git
				synced 2025-11-03 19:17:11 -05:00 
			
		
		
		
	* fix: add unstack button * feat: allow unstacking inside of asset viewer * chore: update tests * chore: rework unstacking in asset viewer --------- Co-authored-by: shenlong-tanwen <139912620+shalong-tanwen@users.noreply.github.com> Co-authored-by: bwees <brandonwees@gmail.com>
		
			
				
	
	
		
			136 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
			
		
		
	
	
			136 lines
		
	
	
		
			6.6 KiB
		
	
	
	
		
			Dart
		
	
	
	
	
	
import 'package:easy_localization/easy_localization.dart';
 | 
						|
import 'package:flutter/material.dart';
 | 
						|
import 'package:hooks_riverpod/hooks_riverpod.dart';
 | 
						|
import 'package:immich_mobile/constants/enums.dart';
 | 
						|
import 'package:immich_mobile/domain/models/album/album.model.dart';
 | 
						|
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
 | 
						|
import 'package:immich_mobile/domain/models/setting.model.dart';
 | 
						|
import 'package:immich_mobile/extensions/translate_extensions.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/advanced_info_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/archive_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/delete_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/delete_local_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/delete_permanent_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/download_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/edit_date_time_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/edit_location_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/favorite_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/move_to_lock_folder_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/share_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/share_link_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/stack_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/trash_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/unstack_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/action_buttons/upload_action_button.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/album/album_selector.widget.dart';
 | 
						|
import 'package:immich_mobile/presentation/widgets/bottom_sheet/base_bottom_sheet.widget.dart';
 | 
						|
import 'package:immich_mobile/providers/infrastructure/album.provider.dart';
 | 
						|
import 'package:immich_mobile/providers/infrastructure/setting.provider.dart';
 | 
						|
import 'package:immich_mobile/providers/server_info.provider.dart';
 | 
						|
import 'package:immich_mobile/providers/timeline/multiselect.provider.dart';
 | 
						|
import 'package:immich_mobile/widgets/common/immich_toast.dart';
 | 
						|
 | 
						|
class GeneralBottomSheet extends ConsumerStatefulWidget {
 | 
						|
  final double? minChildSize;
 | 
						|
  const GeneralBottomSheet({super.key, this.minChildSize});
 | 
						|
 | 
						|
  @override
 | 
						|
  ConsumerState<GeneralBottomSheet> createState() => _GeneralBottomSheetState();
 | 
						|
}
 | 
						|
 | 
						|
class _GeneralBottomSheetState extends ConsumerState<GeneralBottomSheet> {
 | 
						|
  late DraggableScrollableController sheetController;
 | 
						|
  @override
 | 
						|
  void initState() {
 | 
						|
    super.initState();
 | 
						|
    sheetController = DraggableScrollableController();
 | 
						|
  }
 | 
						|
 | 
						|
  @override
 | 
						|
  void dispose() {
 | 
						|
    sheetController.dispose();
 | 
						|
    super.dispose();
 | 
						|
  }
 | 
						|
 | 
						|
  @override
 | 
						|
  Widget build(BuildContext context) {
 | 
						|
    final multiselect = ref.watch(multiSelectProvider);
 | 
						|
    final isTrashEnable = ref.watch(serverInfoProvider.select((state) => state.serverFeatures.trash));
 | 
						|
    final advancedTroubleshooting = ref.watch(settingsProvider.notifier).get(Setting.advancedTroubleshooting);
 | 
						|
 | 
						|
    Future<void> addAssetsToAlbum(RemoteAlbum album) async {
 | 
						|
      final selectedAssets = multiselect.selectedAssets;
 | 
						|
      if (selectedAssets.isEmpty) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
 | 
						|
      final remoteAssets = selectedAssets.whereType<RemoteAsset>();
 | 
						|
      final addedCount = await ref
 | 
						|
          .read(remoteAlbumProvider.notifier)
 | 
						|
          .addAssets(album.id, remoteAssets.map((e) => e.id).toList());
 | 
						|
 | 
						|
      if (selectedAssets.length != remoteAssets.length) {
 | 
						|
        ImmichToast.show(
 | 
						|
          context: context,
 | 
						|
          msg: 'add_to_album_bottom_sheet_some_local_assets'.t(context: context),
 | 
						|
        );
 | 
						|
      }
 | 
						|
 | 
						|
      if (addedCount != remoteAssets.length) {
 | 
						|
        ImmichToast.show(
 | 
						|
          context: context,
 | 
						|
          msg: 'add_to_album_bottom_sheet_already_exists'.tr(namedArgs: {"album": album.name}),
 | 
						|
        );
 | 
						|
      } else {
 | 
						|
        ImmichToast.show(
 | 
						|
          context: context,
 | 
						|
          msg: 'add_to_album_bottom_sheet_added'.tr(namedArgs: {"album": album.name}),
 | 
						|
        );
 | 
						|
      }
 | 
						|
 | 
						|
      ref.read(multiSelectProvider.notifier).reset();
 | 
						|
    }
 | 
						|
 | 
						|
    Future<void> onKeyboardExpand() {
 | 
						|
      return sheetController.animateTo(0.85, duration: const Duration(milliseconds: 200), curve: Curves.easeInOut);
 | 
						|
    }
 | 
						|
 | 
						|
    return BaseBottomSheet(
 | 
						|
      controller: sheetController,
 | 
						|
      initialChildSize: widget.minChildSize ?? 0.15,
 | 
						|
      minChildSize: widget.minChildSize,
 | 
						|
      maxChildSize: 0.85,
 | 
						|
      shouldCloseOnMinExtent: false,
 | 
						|
      actions: [
 | 
						|
        if (multiselect.selectedAssets.length == 1 && advancedTroubleshooting) ...[
 | 
						|
          const AdvancedInfoActionButton(source: ActionSource.timeline),
 | 
						|
        ],
 | 
						|
        const ShareActionButton(source: ActionSource.timeline),
 | 
						|
        if (multiselect.hasRemote) ...[
 | 
						|
          const ShareLinkActionButton(source: ActionSource.timeline),
 | 
						|
          const DownloadActionButton(source: ActionSource.timeline),
 | 
						|
          isTrashEnable
 | 
						|
              ? const TrashActionButton(source: ActionSource.timeline)
 | 
						|
              : const DeletePermanentActionButton(source: ActionSource.timeline),
 | 
						|
          const FavoriteActionButton(source: ActionSource.timeline),
 | 
						|
          const ArchiveActionButton(source: ActionSource.timeline),
 | 
						|
          const EditDateTimeActionButton(source: ActionSource.timeline),
 | 
						|
          const EditLocationActionButton(source: ActionSource.timeline),
 | 
						|
          const MoveToLockFolderActionButton(source: ActionSource.timeline),
 | 
						|
          if (multiselect.selectedAssets.length > 1) const StackActionButton(source: ActionSource.timeline),
 | 
						|
          if (multiselect.hasStacked) const UnStackActionButton(source: ActionSource.timeline),
 | 
						|
          const DeleteActionButton(source: ActionSource.timeline),
 | 
						|
        ],
 | 
						|
        if (multiselect.hasLocal || multiselect.hasMerged) const DeleteLocalActionButton(source: ActionSource.timeline),
 | 
						|
        if (multiselect.hasLocal) const UploadActionButton(source: ActionSource.timeline),
 | 
						|
      ],
 | 
						|
      slivers: multiselect.hasRemote
 | 
						|
          ? [
 | 
						|
              const AddToAlbumHeader(),
 | 
						|
              AlbumSelector(onAlbumSelected: addAssetsToAlbum, onKeyboardExpanded: onKeyboardExpand),
 | 
						|
            ]
 | 
						|
          : [],
 | 
						|
    );
 | 
						|
  }
 | 
						|
}
 |