2024-01-05 05:20:55 +00:00
|
|
|
import 'package:auto_route/auto_route.dart';
|
2023-10-06 07:01:14 +00:00
|
|
|
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';
|
2023-11-29 04:17:29 +00:00
|
|
|
import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
|
2023-11-09 16:19:53 +00:00
|
|
|
import 'package:immich_mobile/extensions/build_context_extensions.dart';
|
2024-05-07 04:04:21 +00:00
|
|
|
import 'package:immich_mobile/widgets/asset_grid/immich_asset_grid.dart';
|
|
|
|
import 'package:immich_mobile/widgets/asset_grid/delete_dialog.dart';
|
2024-05-02 20:59:14 +00:00
|
|
|
import 'package:immich_mobile/providers/trash.provider.dart';
|
2024-05-01 02:36:40 +00:00
|
|
|
import 'package:immich_mobile/entities/asset.entity.dart';
|
2024-05-02 20:59:14 +00:00
|
|
|
import 'package:immich_mobile/providers/server_info.provider.dart';
|
2024-05-07 04:04:21 +00:00
|
|
|
import 'package:immich_mobile/widgets/common/confirm_dialog.dart';
|
|
|
|
import 'package:immich_mobile/widgets/common/immich_toast.dart';
|
2024-05-05 18:14:49 +00:00
|
|
|
import 'package:immich_mobile/utils/immich_loading_overlay.dart';
|
2023-10-06 07:01:14 +00:00
|
|
|
|
2024-01-15 16:50:33 +00:00
|
|
|
@RoutePage()
|
2023-10-06 07:01:14 +00:00
|
|
|
class TrashPage extends HookConsumerWidget {
|
|
|
|
const TrashPage({super.key});
|
|
|
|
|
|
|
|
@override
|
|
|
|
Widget build(BuildContext context, WidgetRef ref) {
|
|
|
|
final trashedAssets = ref.watch(trashedAssetsProvider);
|
|
|
|
final trashDays =
|
|
|
|
ref.watch(serverInfoProvider.select((v) => v.serverConfig.trashDays));
|
|
|
|
final selectionEnabledHook = useState(false);
|
|
|
|
final selection = useState(<Asset>{});
|
2023-11-29 04:20:00 +00:00
|
|
|
final processing = useProcessingOverlay();
|
2023-10-06 07:01:14 +00:00
|
|
|
|
|
|
|
void selectionListener(
|
|
|
|
bool multiselect,
|
|
|
|
Set<Asset> selectedAssets,
|
|
|
|
) {
|
|
|
|
selectionEnabledHook.value = multiselect;
|
|
|
|
selection.value = selectedAssets;
|
|
|
|
}
|
|
|
|
|
|
|
|
onEmptyTrash() async {
|
|
|
|
processing.value = true;
|
|
|
|
await ref.read(trashProvider.notifier).emptyTrash();
|
|
|
|
processing.value = false;
|
|
|
|
selectionEnabledHook.value = false;
|
|
|
|
if (context.mounted) {
|
|
|
|
ImmichToast.show(
|
|
|
|
context: context,
|
2024-08-14 13:21:59 +00:00
|
|
|
msg: 'trash_emptied'.tr(),
|
2023-10-06 07:01:14 +00:00
|
|
|
gravity: ToastGravity.BOTTOM,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
handleEmptyTrash() async {
|
|
|
|
await showDialog(
|
|
|
|
context: context,
|
|
|
|
builder: (context) => ConfirmDialog(
|
|
|
|
onOk: () => onEmptyTrash(),
|
|
|
|
title: "trash_page_empty_trash_btn".tr(),
|
|
|
|
ok: "trash_page_empty_trash_dialog_ok".tr(),
|
|
|
|
content: "trash_page_empty_trash_dialog_content".tr(),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> onPermanentlyDelete() async {
|
|
|
|
processing.value = true;
|
|
|
|
try {
|
|
|
|
if (selection.value.isNotEmpty) {
|
2024-01-18 20:55:19 +00:00
|
|
|
final isRemoved = await ref
|
|
|
|
.read(trashProvider.notifier)
|
|
|
|
.removeAssets(selection.value);
|
2023-10-06 07:01:14 +00:00
|
|
|
|
2024-01-18 20:55:19 +00:00
|
|
|
if (isRemoved) {
|
|
|
|
if (context.mounted) {
|
|
|
|
ImmichToast.show(
|
|
|
|
context: context,
|
2024-08-14 13:21:59 +00:00
|
|
|
msg: 'assets_deleted_permanently'
|
|
|
|
.tr(args: ["${selection.value.length}"]),
|
2024-01-18 20:55:19 +00:00
|
|
|
gravity: ToastGravity.BOTTOM,
|
|
|
|
);
|
|
|
|
}
|
2023-10-06 07:01:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
processing.value = false;
|
|
|
|
selectionEnabledHook.value = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
handlePermanentDelete() async {
|
|
|
|
await showDialog(
|
|
|
|
context: context,
|
|
|
|
builder: (context) => DeleteDialog(
|
2024-03-12 03:25:55 +00:00
|
|
|
alert: "delete_dialog_alert_remote",
|
2023-10-06 07:01:14 +00:00
|
|
|
onDelete: () => onPermanentlyDelete(),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> handleRestoreAll() async {
|
|
|
|
processing.value = true;
|
|
|
|
await ref.read(trashProvider.notifier).restoreTrash();
|
|
|
|
processing.value = false;
|
|
|
|
selectionEnabledHook.value = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> handleRestore() async {
|
|
|
|
processing.value = true;
|
|
|
|
try {
|
|
|
|
if (selection.value.isNotEmpty) {
|
|
|
|
final result = await ref
|
|
|
|
.read(trashProvider.notifier)
|
|
|
|
.restoreAssets(selection.value);
|
|
|
|
|
|
|
|
if (result && context.mounted) {
|
|
|
|
ImmichToast.show(
|
|
|
|
context: context,
|
2024-08-14 13:21:59 +00:00
|
|
|
msg: 'assets_restored_successfully'
|
|
|
|
.tr(args: ["${selection.value.length}"]),
|
2023-10-06 07:01:14 +00:00
|
|
|
gravity: ToastGravity.BOTTOM,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
processing.value = false;
|
|
|
|
selectionEnabledHook.value = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
String getAppBarTitle(String count) {
|
|
|
|
if (selectionEnabledHook.value) {
|
|
|
|
return selection.value.isNotEmpty
|
|
|
|
? "${selection.value.length}"
|
|
|
|
: "trash_page_select_assets_btn".tr();
|
|
|
|
}
|
|
|
|
return 'trash_page_title'.tr(args: [count]);
|
|
|
|
}
|
|
|
|
|
|
|
|
AppBar buildAppBar(String count) {
|
|
|
|
return AppBar(
|
|
|
|
leading: IconButton(
|
|
|
|
onPressed: !selectionEnabledHook.value
|
2024-05-14 19:07:31 +00:00
|
|
|
? () => context.maybePop()
|
2023-10-06 07:01:14 +00:00
|
|
|
: () {
|
|
|
|
selectionEnabledHook.value = false;
|
|
|
|
selection.value = {};
|
|
|
|
},
|
|
|
|
icon: !selectionEnabledHook.value
|
|
|
|
? const Icon(Icons.arrow_back_ios_rounded)
|
|
|
|
: const Icon(Icons.close_rounded),
|
|
|
|
),
|
|
|
|
centerTitle: !selectionEnabledHook.value,
|
|
|
|
automaticallyImplyLeading: false,
|
|
|
|
title: Text(getAppBarTitle(count)),
|
|
|
|
actions: <Widget>[
|
|
|
|
if (!selectionEnabledHook.value)
|
|
|
|
PopupMenuButton<void Function()>(
|
|
|
|
itemBuilder: (context) {
|
|
|
|
return [
|
|
|
|
PopupMenuItem(
|
|
|
|
value: () => selectionEnabledHook.value = true,
|
|
|
|
child: const Text('trash_page_select_btn').tr(),
|
|
|
|
),
|
|
|
|
PopupMenuItem(
|
|
|
|
value: handleEmptyTrash,
|
|
|
|
child: const Text('trash_page_empty_trash_btn').tr(),
|
|
|
|
),
|
|
|
|
];
|
|
|
|
},
|
|
|
|
onSelected: (fn) => fn(),
|
|
|
|
),
|
|
|
|
],
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
Widget buildBottomBar() {
|
|
|
|
return SafeArea(
|
|
|
|
child: Align(
|
|
|
|
alignment: Alignment.bottomCenter,
|
|
|
|
child: SizedBox(
|
|
|
|
height: 64,
|
|
|
|
child: Container(
|
2023-11-09 16:19:53 +00:00
|
|
|
color: context.themeData.canvasColor,
|
2023-10-06 07:01:14 +00:00
|
|
|
child: Row(
|
|
|
|
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
|
|
|
|
children: [
|
|
|
|
TextButton.icon(
|
|
|
|
icon: Icon(
|
|
|
|
Icons.delete_forever,
|
|
|
|
color: Colors.red[400],
|
|
|
|
),
|
|
|
|
label: Text(
|
|
|
|
selection.value.isEmpty
|
|
|
|
? 'trash_page_delete_all'.tr()
|
|
|
|
: 'trash_page_delete'.tr(),
|
|
|
|
style: TextStyle(
|
|
|
|
fontSize: 14,
|
|
|
|
color: Colors.red[400],
|
|
|
|
fontWeight: FontWeight.bold,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
onPressed: processing.value
|
|
|
|
? null
|
|
|
|
: selection.value.isEmpty
|
|
|
|
? handleEmptyTrash
|
|
|
|
: handlePermanentDelete,
|
|
|
|
),
|
|
|
|
TextButton.icon(
|
|
|
|
icon: const Icon(
|
|
|
|
Icons.history_rounded,
|
|
|
|
),
|
|
|
|
label: Text(
|
|
|
|
selection.value.isEmpty
|
|
|
|
? 'trash_page_restore_all'.tr()
|
|
|
|
: 'trash_page_restore'.tr(),
|
|
|
|
style: const TextStyle(
|
|
|
|
fontSize: 14,
|
|
|
|
fontWeight: FontWeight.bold,
|
|
|
|
),
|
|
|
|
),
|
|
|
|
onPressed: processing.value
|
|
|
|
? null
|
|
|
|
: selection.value.isEmpty
|
|
|
|
? handleRestoreAll
|
|
|
|
: handleRestore,
|
|
|
|
),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-11-29 04:17:29 +00:00
|
|
|
return Scaffold(
|
|
|
|
appBar: trashedAssets.maybeWhen(
|
|
|
|
orElse: () => buildAppBar("?"),
|
|
|
|
data: (data) => buildAppBar(data.totalAssets.toString()),
|
2023-10-06 07:01:14 +00:00
|
|
|
),
|
2023-11-29 04:17:29 +00:00
|
|
|
body: trashedAssets.widgetWhen(
|
|
|
|
onData: (data) => data.isEmpty
|
2023-10-06 07:01:14 +00:00
|
|
|
? Center(
|
|
|
|
child: Text('trash_page_no_assets'.tr()),
|
|
|
|
)
|
|
|
|
: Stack(
|
|
|
|
children: [
|
|
|
|
SafeArea(
|
|
|
|
child: ImmichAssetGrid(
|
|
|
|
renderList: data,
|
|
|
|
listener: selectionListener,
|
|
|
|
selectionActive: selectionEnabledHook.value,
|
|
|
|
showMultiSelectIndicator: false,
|
2023-10-22 02:38:07 +00:00
|
|
|
showStack: true,
|
2023-10-06 07:01:14 +00:00
|
|
|
topWidget: Padding(
|
2023-11-29 04:17:29 +00:00
|
|
|
padding: const EdgeInsets.symmetric(
|
|
|
|
horizontal: 12,
|
|
|
|
vertical: 24,
|
2023-10-06 07:01:14 +00:00
|
|
|
),
|
|
|
|
child: const Text(
|
|
|
|
"trash_page_info",
|
|
|
|
).tr(args: ["$trashDays"]),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
),
|
|
|
|
if (selectionEnabledHook.value) buildBottomBar(),
|
|
|
|
],
|
|
|
|
),
|
|
|
|
),
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|