import 'package:hooks_riverpod/hooks_riverpod.dart'; import 'package:immich_mobile/modules/home/ui/asset_grid/asset_grid_data_structure.dart'; import 'package:immich_mobile/modules/trash/services/trash.service.dart'; import 'package:immich_mobile/shared/models/asset.dart'; import 'package:immich_mobile/shared/models/exif_info.dart'; import 'package:immich_mobile/shared/providers/asset.provider.dart'; import 'package:immich_mobile/shared/providers/db.provider.dart'; import 'package:immich_mobile/shared/providers/user.provider.dart'; import 'package:isar/isar.dart'; import 'package:logging/logging.dart'; class TrashNotifier extends StateNotifier { final Isar _db; final Ref _ref; final TrashService _trashService; final _log = Logger('TrashNotifier'); TrashNotifier( this._trashService, this._db, this._ref, ) : super(false); Future emptyTrash() async { try { final user = _ref.read(currentUserProvider); if (user == null) { return; } await _trashService.emptyTrash(); final dbIds = await _db.assets .where() .remoteIdIsNotNull() .filter() .ownerIdEqualTo(user.isarId) .isTrashedEqualTo(true) .idProperty() .findAll(); await _db.writeTxn(() async { await _db.exifInfos.deleteAll(dbIds); await _db.assets.deleteAll(dbIds); }); // Refresh assets in background Future.delayed( const Duration(seconds: 4), () async => await _ref.read(assetProvider.notifier).getAllAsset(), ); } catch (error, stack) { _log.severe("Cannot empty trash ${error.toString()}", error, stack); } } Future restoreAssets(Iterable assetList) async { try { final result = await _trashService.restoreAssets(assetList); if (result) { final remoteAssets = assetList.where((a) => a.isRemote).toList(); final updatedAssets = remoteAssets.map((e) { e.isTrashed = false; return e; }).toList(); await _db.writeTxn(() async { await _db.assets.putAll(updatedAssets); }); // Refresh assets in background Future.delayed( const Duration(seconds: 4), () async => await _ref.read(assetProvider.notifier).getAllAsset(), ); return true; } } catch (error, stack) { _log.severe("Cannot restore trash ${error.toString()}", error, stack); } return false; } Future restoreTrash() async { try { final user = _ref.read(currentUserProvider); if (user == null) { return; } await _trashService.restoreTrash(); final assets = await _db.assets .where() .remoteIdIsNotNull() .filter() .ownerIdEqualTo(user.isarId) .isTrashedEqualTo(true) .findAll(); final updatedAssets = assets.map((e) { e.isTrashed = false; return e; }).toList(); await _db.writeTxn(() async { await _db.assets.putAll(updatedAssets); }); // Refresh assets in background Future.delayed( const Duration(seconds: 4), () async => await _ref.read(assetProvider.notifier).getAllAsset(), ); } catch (error, stack) { _log.severe("Cannot restore trash ${error.toString()}", error, stack); } } } final trashProvider = StateNotifierProvider((ref) { return TrashNotifier( ref.watch(trashServiceProvider), ref.watch(dbProvider), ref, ); }); final trashedAssetsProvider = StreamProvider((ref) async* { final user = ref.read(currentUserProvider); if (user == null) return; final query = ref .watch(dbProvider) .assets .filter() .ownerIdEqualTo(user.isarId) .isTrashedEqualTo(true) .sortByFileCreatedAt(); const groupBy = GroupAssetsBy.none; yield await RenderList.fromQuery(query, groupBy); await for (final _ in query.watchLazy()) { yield await RenderList.fromQuery(query, groupBy); } });