|
@@ -0,0 +1,205 @@
|
|
|
+import 'package:auto_route/auto_route.dart';
|
|
|
+import 'package:easy_localization/easy_localization.dart';
|
|
|
+import 'package:flutter/material.dart';
|
|
|
+import 'package:flutter_hooks/flutter_hooks.dart';
|
|
|
+import 'package:fluttertoast/fluttertoast.dart';
|
|
|
+import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|
|
+import 'package:immich_mobile/modules/album/providers/shared_album.provider.dart';
|
|
|
+import 'package:immich_mobile/modules/login/providers/authentication.provider.dart';
|
|
|
+import 'package:immich_mobile/routing/router.dart';
|
|
|
+import 'package:immich_mobile/shared/models/album.dart';
|
|
|
+import 'package:immich_mobile/shared/models/user.dart';
|
|
|
+import 'package:immich_mobile/shared/ui/immich_toast.dart';
|
|
|
+import 'package:immich_mobile/shared/ui/user_circle_avatar.dart';
|
|
|
+import 'package:immich_mobile/shared/views/immich_loading_overlay.dart';
|
|
|
+
|
|
|
+class AlbumOptionsPage extends HookConsumerWidget {
|
|
|
+ final Album album;
|
|
|
+
|
|
|
+ const AlbumOptionsPage({super.key, required this.album});
|
|
|
+
|
|
|
+ @override
|
|
|
+ Widget build(BuildContext context, WidgetRef ref) {
|
|
|
+ final sharedUsers = useState(album.sharedUsers.toList());
|
|
|
+ final owner = album.owner.value;
|
|
|
+ final userId = ref.watch(authenticationProvider).userId;
|
|
|
+ final isOwner = owner?.id == userId;
|
|
|
+
|
|
|
+ void showErrorMessage() {
|
|
|
+ Navigator.pop(context);
|
|
|
+ ImmichToast.show(
|
|
|
+ context: context,
|
|
|
+ msg: "Error leaving/removing from album",
|
|
|
+ toastType: ToastType.error,
|
|
|
+ gravity: ToastGravity.BOTTOM,
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ void leaveAlbum() async {
|
|
|
+ ImmichLoadingOverlayController.appLoader.show();
|
|
|
+
|
|
|
+ try {
|
|
|
+ final isSuccess =
|
|
|
+ await ref.read(sharedAlbumProvider.notifier).leaveAlbum(album);
|
|
|
+
|
|
|
+ if (isSuccess) {
|
|
|
+ AutoRouter.of(context)
|
|
|
+ .navigate(const TabControllerRoute(children: [SharingRoute()]));
|
|
|
+ } else {
|
|
|
+ showErrorMessage();
|
|
|
+ }
|
|
|
+ } catch (_) {
|
|
|
+ showErrorMessage();
|
|
|
+ }
|
|
|
+
|
|
|
+ ImmichLoadingOverlayController.appLoader.hide();
|
|
|
+ }
|
|
|
+
|
|
|
+ void removeUserFromAlbum(User user) async {
|
|
|
+ ImmichLoadingOverlayController.appLoader.show();
|
|
|
+
|
|
|
+ try {
|
|
|
+ await ref
|
|
|
+ .read(sharedAlbumProvider.notifier)
|
|
|
+ .removeUserFromAlbum(album, user);
|
|
|
+ album.sharedUsers.remove(user);
|
|
|
+ sharedUsers.value = album.sharedUsers.toList();
|
|
|
+ } catch (error) {
|
|
|
+ showErrorMessage();
|
|
|
+ }
|
|
|
+
|
|
|
+ Navigator.pop(context);
|
|
|
+ ImmichLoadingOverlayController.appLoader.hide();
|
|
|
+ }
|
|
|
+
|
|
|
+ void handleUserClick(User user) {
|
|
|
+ var actions = [];
|
|
|
+
|
|
|
+ if (user.id == userId) {
|
|
|
+ actions = [
|
|
|
+ ListTile(
|
|
|
+ leading: const Icon(Icons.exit_to_app_rounded),
|
|
|
+ title: const Text("Leave album"),
|
|
|
+ onTap: leaveAlbum,
|
|
|
+ ),
|
|
|
+ ];
|
|
|
+ }
|
|
|
+
|
|
|
+ if (isOwner) {
|
|
|
+ actions = [
|
|
|
+ ListTile(
|
|
|
+ leading: const Icon(Icons.person_remove_rounded),
|
|
|
+ title: const Text("Remove user from album"),
|
|
|
+ onTap: () => removeUserFromAlbum(user),
|
|
|
+ ),
|
|
|
+ ];
|
|
|
+ }
|
|
|
+
|
|
|
+ showModalBottomSheet(
|
|
|
+ backgroundColor: Theme.of(context).scaffoldBackgroundColor,
|
|
|
+ isScrollControlled: false,
|
|
|
+ context: context,
|
|
|
+ builder: (context) {
|
|
|
+ return SafeArea(
|
|
|
+ child: Padding(
|
|
|
+ padding: const EdgeInsets.only(top: 24.0),
|
|
|
+ child: Column(
|
|
|
+ mainAxisSize: MainAxisSize.min,
|
|
|
+ children: [...actions],
|
|
|
+ ),
|
|
|
+ ),
|
|
|
+ );
|
|
|
+ },
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ buildOwnerInfo() {
|
|
|
+ return ListTile(
|
|
|
+ leading: owner != null
|
|
|
+ ? UserCircleAvatar(
|
|
|
+ user: owner,
|
|
|
+ useRandomBackgroundColor: true,
|
|
|
+ )
|
|
|
+ : const SizedBox(),
|
|
|
+ title: Text(
|
|
|
+ album.owner.value?.firstName ?? "",
|
|
|
+ style: const TextStyle(
|
|
|
+ fontWeight: FontWeight.bold,
|
|
|
+ ),
|
|
|
+ ),
|
|
|
+ subtitle: Text(
|
|
|
+ album.owner.value?.email ?? "",
|
|
|
+ style: TextStyle(color: Colors.grey[500]),
|
|
|
+ ),
|
|
|
+ trailing: const Text(
|
|
|
+ "Owner",
|
|
|
+ style: TextStyle(
|
|
|
+ fontWeight: FontWeight.bold,
|
|
|
+ ),
|
|
|
+ ),
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ buildSharedUsersList() {
|
|
|
+ return ListView.builder(
|
|
|
+ shrinkWrap: true,
|
|
|
+ itemCount: sharedUsers.value.length,
|
|
|
+ itemBuilder: (context, index) {
|
|
|
+ final user = sharedUsers.value[index];
|
|
|
+ return ListTile(
|
|
|
+ leading: UserCircleAvatar(
|
|
|
+ user: user,
|
|
|
+ useRandomBackgroundColor: true,
|
|
|
+ radius: 22,
|
|
|
+ ),
|
|
|
+ title: Text(
|
|
|
+ user.firstName,
|
|
|
+ style: const TextStyle(
|
|
|
+ fontWeight: FontWeight.bold,
|
|
|
+ ),
|
|
|
+ ),
|
|
|
+ subtitle: Text(
|
|
|
+ user.email,
|
|
|
+ style: TextStyle(color: Colors.grey[500]),
|
|
|
+ ),
|
|
|
+ trailing: userId == user.id || isOwner
|
|
|
+ ? const Icon(Icons.more_horiz_rounded)
|
|
|
+ : const SizedBox(),
|
|
|
+ onTap: userId == user.id || isOwner
|
|
|
+ ? () => handleUserClick(user)
|
|
|
+ : null,
|
|
|
+ );
|
|
|
+ },
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ buildSectionTitle(String text) {
|
|
|
+ return Padding(
|
|
|
+ padding: const EdgeInsets.all(16.0),
|
|
|
+ child: Text(text, style: Theme.of(context).textTheme.bodySmall),
|
|
|
+ );
|
|
|
+ }
|
|
|
+
|
|
|
+ return Scaffold(
|
|
|
+ appBar: AppBar(
|
|
|
+ leading: IconButton(
|
|
|
+ icon: const Icon(Icons.arrow_back_ios_new_rounded),
|
|
|
+ onPressed: () {
|
|
|
+ AutoRouter.of(context).pop(null);
|
|
|
+ },
|
|
|
+ ),
|
|
|
+ centerTitle: true,
|
|
|
+ title: Text("translated_text_options".tr()),
|
|
|
+ ),
|
|
|
+ body: Column(
|
|
|
+ mainAxisAlignment: MainAxisAlignment.start,
|
|
|
+ crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
+ children: [
|
|
|
+ buildSectionTitle("PEOPLE"),
|
|
|
+ buildOwnerInfo(),
|
|
|
+ buildSharedUsersList(),
|
|
|
+ ],
|
|
|
+ ),
|
|
|
+ );
|
|
|
+ }
|
|
|
+}
|