mirror of https://github.com/immich-app/immich.git
feat: selection mode timeline (#19734)
* feat: new page * force multi-selection state * fix: provider scoping * Return selected assets * lint * lint * simplify provider scope and drop drilling * selection stylingpull/19818/head
parent
dd94ad17aa
commit
87dd09d103
@ -0,0 +1,44 @@
|
|||||||
|
import 'package:auto_route/auto_route.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
|
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
||||||
|
import 'package:immich_mobile/presentation/widgets/timeline/timeline.widget.dart';
|
||||||
|
import 'package:immich_mobile/providers/infrastructure/timeline.provider.dart';
|
||||||
|
import 'package:immich_mobile/providers/timeline/multiselect.provider.dart';
|
||||||
|
|
||||||
|
@RoutePage()
|
||||||
|
class DriftAssetSelectionTimelinePage extends ConsumerWidget {
|
||||||
|
final Set<BaseAsset> lockedSelectionAssets;
|
||||||
|
const DriftAssetSelectionTimelinePage({
|
||||||
|
super.key,
|
||||||
|
this.lockedSelectionAssets = const {},
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context, WidgetRef ref) {
|
||||||
|
return ProviderScope(
|
||||||
|
overrides: [
|
||||||
|
multiSelectProvider.overrideWith(
|
||||||
|
() => MultiSelectNotifier(
|
||||||
|
MultiSelectState(
|
||||||
|
selectedAssets: {},
|
||||||
|
lockedSelectionAssets: lockedSelectionAssets,
|
||||||
|
forceEnable: true,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
timelineServiceProvider.overrideWith(
|
||||||
|
(ref) {
|
||||||
|
final timelineUsers =
|
||||||
|
ref.watch(timelineUsersProvider).valueOrNull ?? [];
|
||||||
|
final timelineService =
|
||||||
|
ref.watch(timelineFactoryProvider).remoteAssets(timelineUsers);
|
||||||
|
ref.onDispose(timelineService.dispose);
|
||||||
|
return timelineService;
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
child: const Timeline(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@ -0,0 +1,77 @@
|
|||||||
|
import 'package:auto_route/auto_route.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
|
import 'package:immich_mobile/domain/models/asset/base_asset.model.dart';
|
||||||
|
import 'package:immich_mobile/extensions/build_context_extensions.dart';
|
||||||
|
import 'package:immich_mobile/extensions/translate_extensions.dart';
|
||||||
|
import 'package:immich_mobile/providers/timeline/multiselect.provider.dart';
|
||||||
|
|
||||||
|
class SelectionSliverAppBar extends ConsumerStatefulWidget {
|
||||||
|
const SelectionSliverAppBar({
|
||||||
|
super.key,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
ConsumerState<SelectionSliverAppBar> createState() =>
|
||||||
|
_SelectionSliverAppBarState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _SelectionSliverAppBarState extends ConsumerState<SelectionSliverAppBar> {
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final selection = ref.watch(
|
||||||
|
multiSelectProvider.select((s) => s.selectedAssets),
|
||||||
|
);
|
||||||
|
|
||||||
|
final toExclude = ref.watch(
|
||||||
|
multiSelectProvider.select((s) => s.lockedSelectionAssets),
|
||||||
|
);
|
||||||
|
|
||||||
|
final filteredAssets = selection.where((asset) {
|
||||||
|
return !toExclude.contains(asset);
|
||||||
|
}).toSet();
|
||||||
|
|
||||||
|
onDone(Set<BaseAsset> selected) {
|
||||||
|
ref.read(multiSelectProvider.notifier).reset();
|
||||||
|
context.maybePop<Set<BaseAsset>>(selected);
|
||||||
|
}
|
||||||
|
|
||||||
|
return SliverAppBar(
|
||||||
|
floating: true,
|
||||||
|
pinned: true,
|
||||||
|
snap: false,
|
||||||
|
backgroundColor: context.colorScheme.surfaceContainer,
|
||||||
|
shape: const RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.all(Radius.circular(5)),
|
||||||
|
),
|
||||||
|
automaticallyImplyLeading: false,
|
||||||
|
leading: IconButton(
|
||||||
|
icon: const Icon(Icons.close_rounded),
|
||||||
|
onPressed: () {
|
||||||
|
ref.read(multiSelectProvider.notifier).reset();
|
||||||
|
context.pop<Set<BaseAsset>>(null);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
centerTitle: true,
|
||||||
|
title: Text(
|
||||||
|
"Select {count}".t(
|
||||||
|
context: context,
|
||||||
|
args: {
|
||||||
|
'count': filteredAssets.length.toString(),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => onDone(filteredAssets),
|
||||||
|
child: Text(
|
||||||
|
'done'.t(context: context),
|
||||||
|
style: context.textTheme.titleSmall?.copyWith(
|
||||||
|
color: context.colorScheme.primary,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
Loading…
Reference in New Issue