mirror of
https://github.com/immich-app/immich.git
synced 2025-05-31 12:15:47 -04:00
refactor(mobile): trash provider (#16219)
* refactor(mobile): trash provider * refactor(mobile): trash provider * pr feedback
This commit is contained in:
parent
34b88bb47a
commit
17a2043e76
@ -79,7 +79,7 @@ custom_lint:
|
|||||||
- lib/widgets/asset_grid/asset_grid_data_structure.dart
|
- lib/widgets/asset_grid/asset_grid_data_structure.dart
|
||||||
- test/**.dart
|
- test/**.dart
|
||||||
# refactor the remaining providers
|
# refactor the remaining providers
|
||||||
- lib/providers/{archive,asset,authentication,db,favorite,partner,trash,user}.provider.dart
|
- lib/providers/{archive,asset,authentication,db,favorite,partner,user}.provider.dart
|
||||||
- lib/providers/{asset_viewer/render_list,backup/backup,search/all_motion_photos,search/recently_added_asset}.provider.dart
|
- lib/providers/{asset_viewer/render_list,backup/backup,search/all_motion_photos,search/recently_added_asset}.provider.dart
|
||||||
|
|
||||||
- import_rule_openapi:
|
- import_rule_openapi:
|
||||||
|
@ -2,6 +2,7 @@ import 'package:immich_mobile/entities/album.entity.dart';
|
|||||||
import 'package:immich_mobile/entities/asset.entity.dart';
|
import 'package:immich_mobile/entities/asset.entity.dart';
|
||||||
import 'package:immich_mobile/entities/device_asset.entity.dart';
|
import 'package:immich_mobile/entities/device_asset.entity.dart';
|
||||||
import 'package:immich_mobile/interfaces/database.interface.dart';
|
import 'package:immich_mobile/interfaces/database.interface.dart';
|
||||||
|
import 'package:immich_mobile/widgets/asset_grid/asset_grid_data_structure.dart';
|
||||||
|
|
||||||
abstract interface class IAssetRepository implements IDatabaseRepository {
|
abstract interface class IAssetRepository implements IDatabaseRepository {
|
||||||
Future<Asset?> getByRemoteId(String id);
|
Future<Asset?> getByRemoteId(String id);
|
||||||
@ -63,6 +64,10 @@ abstract interface class IAssetRepository implements IDatabaseRepository {
|
|||||||
Future<void> clearTable();
|
Future<void> clearTable();
|
||||||
|
|
||||||
Stream<Asset?> watchAsset(int id, {bool fireImmediately = false});
|
Stream<Asset?> watchAsset(int id, {bool fireImmediately = false});
|
||||||
|
|
||||||
|
Future<List<Asset>> getTrashAssets(int userId);
|
||||||
|
|
||||||
|
Stream<RenderList> getTrashRenderListStream(int userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
enum AssetSort { checksum, ownerIdChecksum }
|
enum AssetSort { checksum, ownerIdChecksum }
|
||||||
|
@ -6,6 +6,7 @@ import 'package:fluttertoast/fluttertoast.dart';
|
|||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
|
import 'package:immich_mobile/extensions/asyncvalue_extensions.dart';
|
||||||
import 'package:immich_mobile/extensions/build_context_extensions.dart';
|
import 'package:immich_mobile/extensions/build_context_extensions.dart';
|
||||||
|
import 'package:immich_mobile/providers/asset.provider.dart';
|
||||||
import 'package:immich_mobile/widgets/asset_grid/immich_asset_grid.dart';
|
import 'package:immich_mobile/widgets/asset_grid/immich_asset_grid.dart';
|
||||||
import 'package:immich_mobile/widgets/asset_grid/delete_dialog.dart';
|
import 'package:immich_mobile/widgets/asset_grid/delete_dialog.dart';
|
||||||
import 'package:immich_mobile/providers/trash.provider.dart';
|
import 'package:immich_mobile/providers/trash.provider.dart';
|
||||||
@ -67,8 +68,8 @@ class TrashPage extends HookConsumerWidget {
|
|||||||
try {
|
try {
|
||||||
if (selection.value.isNotEmpty) {
|
if (selection.value.isNotEmpty) {
|
||||||
final isRemoved = await ref
|
final isRemoved = await ref
|
||||||
.read(trashProvider.notifier)
|
.read(assetProvider.notifier)
|
||||||
.removeAssets(selection.value);
|
.deleteAssets(selection.value, force: true);
|
||||||
|
|
||||||
if (isRemoved) {
|
if (isRemoved) {
|
||||||
if (context.mounted) {
|
if (context.mounted) {
|
||||||
|
@ -2,150 +2,44 @@ import 'package:hooks_riverpod/hooks_riverpod.dart';
|
|||||||
import 'package:immich_mobile/widgets/asset_grid/asset_grid_data_structure.dart';
|
import 'package:immich_mobile/widgets/asset_grid/asset_grid_data_structure.dart';
|
||||||
import 'package:immich_mobile/services/trash.service.dart';
|
import 'package:immich_mobile/services/trash.service.dart';
|
||||||
import 'package:immich_mobile/entities/asset.entity.dart';
|
import 'package:immich_mobile/entities/asset.entity.dart';
|
||||||
import 'package:immich_mobile/providers/asset.provider.dart';
|
|
||||||
import 'package:immich_mobile/providers/db.provider.dart';
|
|
||||||
import 'package:immich_mobile/providers/user.provider.dart';
|
import 'package:immich_mobile/providers/user.provider.dart';
|
||||||
import 'package:immich_mobile/services/sync.service.dart';
|
|
||||||
import 'package:immich_mobile/utils/renderlist_generator.dart';
|
|
||||||
import 'package:isar/isar.dart';
|
|
||||||
import 'package:logging/logging.dart';
|
import 'package:logging/logging.dart';
|
||||||
|
|
||||||
class TrashNotifier extends StateNotifier<bool> {
|
class TrashNotifier extends StateNotifier<bool> {
|
||||||
final Isar _db;
|
|
||||||
final Ref _ref;
|
|
||||||
final TrashService _trashService;
|
final TrashService _trashService;
|
||||||
final _log = Logger('TrashNotifier');
|
final _log = Logger('TrashNotifier');
|
||||||
|
|
||||||
TrashNotifier(
|
TrashNotifier(
|
||||||
this._trashService,
|
this._trashService,
|
||||||
this._db,
|
|
||||||
this._ref,
|
|
||||||
) : super(false);
|
) : super(false);
|
||||||
|
|
||||||
Future<void> emptyTrash() async {
|
Future<void> emptyTrash() async {
|
||||||
try {
|
try {
|
||||||
final user = _ref.read(currentUserProvider);
|
|
||||||
if (user == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await _trashService.emptyTrash();
|
await _trashService.emptyTrash();
|
||||||
|
state = true;
|
||||||
final idsToRemove = await _db.assets
|
|
||||||
.where()
|
|
||||||
.remoteIdIsNotNull()
|
|
||||||
.filter()
|
|
||||||
.ownerIdEqualTo(user.isarId)
|
|
||||||
.isTrashedEqualTo(true)
|
|
||||||
.remoteIdProperty()
|
|
||||||
.findAll();
|
|
||||||
|
|
||||||
// TODO: handle local asset removal on emptyTrash
|
|
||||||
_ref
|
|
||||||
.read(syncServiceProvider)
|
|
||||||
.handleRemoteAssetRemoval(idsToRemove.cast<String>().toList());
|
|
||||||
} catch (error, stack) {
|
} catch (error, stack) {
|
||||||
_log.severe("Cannot empty trash", error, stack);
|
_log.severe("Cannot empty trash", error, stack);
|
||||||
|
state = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<bool> removeAssets(Iterable<Asset> assetList) async {
|
|
||||||
try {
|
|
||||||
final user = _ref.read(currentUserProvider);
|
|
||||||
if (user == null) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
final isRemoved = await _ref
|
|
||||||
.read(assetProvider.notifier)
|
|
||||||
.deleteRemoteAssets(assetList, shouldDeletePermanently: true);
|
|
||||||
|
|
||||||
if (isRemoved) {
|
|
||||||
final idsToRemove =
|
|
||||||
assetList.where((a) => a.isRemote).map((a) => a.remoteId!).toList();
|
|
||||||
|
|
||||||
_ref
|
|
||||||
.read(syncServiceProvider)
|
|
||||||
.handleRemoteAssetRemoval(idsToRemove.cast<String>().toList());
|
|
||||||
}
|
|
||||||
|
|
||||||
return isRemoved;
|
|
||||||
} catch (error, stack) {
|
|
||||||
_log.severe("Cannot remove assets", error, stack);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<bool> restoreAsset(Asset asset) async {
|
|
||||||
try {
|
|
||||||
final result = await _trashService.restoreAsset(asset);
|
|
||||||
|
|
||||||
if (result) {
|
|
||||||
final remoteAsset = asset.isRemote;
|
|
||||||
|
|
||||||
asset.isTrashed = false;
|
|
||||||
|
|
||||||
if (remoteAsset) {
|
|
||||||
await _db.writeTxn(() async {
|
|
||||||
await _db.assets.put(asset);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
} catch (error, stack) {
|
|
||||||
_log.severe("Cannot restore asset", error, stack);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<bool> restoreAssets(Iterable<Asset> assetList) async {
|
Future<bool> restoreAssets(Iterable<Asset> assetList) async {
|
||||||
try {
|
try {
|
||||||
final result = await _trashService.restoreAssets(assetList);
|
await _trashService.restoreAssets(assetList);
|
||||||
|
return true;
|
||||||
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);
|
|
||||||
});
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
} catch (error, stack) {
|
} catch (error, stack) {
|
||||||
_log.severe("Cannot restore assets", error, stack);
|
_log.severe("Cannot restore assets", error, stack);
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> restoreTrash() async {
|
Future<void> restoreTrash() async {
|
||||||
try {
|
try {
|
||||||
final user = _ref.read(currentUserProvider);
|
|
||||||
if (user == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await _trashService.restoreTrash();
|
await _trashService.restoreTrash();
|
||||||
|
state = true;
|
||||||
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);
|
|
||||||
});
|
|
||||||
} catch (error, stack) {
|
} catch (error, stack) {
|
||||||
_log.severe("Cannot restore trash", error, stack);
|
_log.severe("Cannot restore trash", error, stack);
|
||||||
|
state = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -153,20 +47,14 @@ class TrashNotifier extends StateNotifier<bool> {
|
|||||||
final trashProvider = StateNotifierProvider<TrashNotifier, bool>((ref) {
|
final trashProvider = StateNotifierProvider<TrashNotifier, bool>((ref) {
|
||||||
return TrashNotifier(
|
return TrashNotifier(
|
||||||
ref.watch(trashServiceProvider),
|
ref.watch(trashServiceProvider),
|
||||||
ref.watch(dbProvider),
|
|
||||||
ref,
|
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
final trashedAssetsProvider = StreamProvider<RenderList>((ref) {
|
final trashedAssetsProvider = StreamProvider<RenderList>((ref) {
|
||||||
final user = ref.read(currentUserProvider);
|
final user = ref.read(currentUserProvider);
|
||||||
if (user == null) return const Stream.empty();
|
if (user == null) {
|
||||||
final query = ref
|
return const Stream.empty();
|
||||||
.watch(dbProvider)
|
}
|
||||||
.assets
|
|
||||||
.filter()
|
return ref.watch(trashServiceProvider).getRenderListGenerator(user.isarId);
|
||||||
.ownerIdEqualTo(user.isarId)
|
|
||||||
.isTrashedEqualTo(true)
|
|
||||||
.sortByFileCreatedAtDesc();
|
|
||||||
return renderListGeneratorWithGroupBy(query, GroupAssetsBy.none);
|
|
||||||
});
|
});
|
||||||
|
@ -11,6 +11,7 @@ import 'package:immich_mobile/entities/ios_device_asset.entity.dart';
|
|||||||
import 'package:immich_mobile/interfaces/asset.interface.dart';
|
import 'package:immich_mobile/interfaces/asset.interface.dart';
|
||||||
import 'package:immich_mobile/providers/db.provider.dart';
|
import 'package:immich_mobile/providers/db.provider.dart';
|
||||||
import 'package:immich_mobile/repositories/database.repository.dart';
|
import 'package:immich_mobile/repositories/database.repository.dart';
|
||||||
|
import 'package:immich_mobile/widgets/asset_grid/asset_grid_data_structure.dart';
|
||||||
import 'package:isar/isar.dart';
|
import 'package:isar/isar.dart';
|
||||||
|
|
||||||
final assetRepositoryProvider =
|
final assetRepositoryProvider =
|
||||||
@ -222,6 +223,31 @@ class AssetRepository extends DatabaseRepository implements IAssetRepository {
|
|||||||
Stream<Asset?> watchAsset(int id, {bool fireImmediately = false}) {
|
Stream<Asset?> watchAsset(int id, {bool fireImmediately = false}) {
|
||||||
return db.assets.watchObject(id, fireImmediately: fireImmediately);
|
return db.assets.watchObject(id, fireImmediately: fireImmediately);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Future<List<Asset>> getTrashAssets(int userId) {
|
||||||
|
return db.assets
|
||||||
|
.where()
|
||||||
|
.remoteIdIsNotNull()
|
||||||
|
.filter()
|
||||||
|
.ownerIdEqualTo(userId)
|
||||||
|
.isTrashedEqualTo(true)
|
||||||
|
.findAll();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Stream<RenderList> getTrashRenderListStream(int userId) async* {
|
||||||
|
final query = db.assets
|
||||||
|
.filter()
|
||||||
|
.ownerIdEqualTo(userId)
|
||||||
|
.isTrashedEqualTo(true)
|
||||||
|
.sortByFileCreatedAtDesc();
|
||||||
|
|
||||||
|
yield await RenderList.fromQuery(query, GroupAssetsBy.none);
|
||||||
|
await for (final _ in query.watchLazy()) {
|
||||||
|
yield await RenderList.fromQuery(query, GroupAssetsBy.none);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<List<Asset>> _getMatchesImpl(
|
Future<List<Asset>> _getMatchesImpl(
|
||||||
|
@ -1,62 +1,89 @@
|
|||||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||||
import 'package:immich_mobile/entities/asset.entity.dart';
|
import 'package:immich_mobile/entities/asset.entity.dart';
|
||||||
|
import 'package:immich_mobile/interfaces/asset.interface.dart';
|
||||||
|
import 'package:immich_mobile/interfaces/user.interface.dart';
|
||||||
import 'package:immich_mobile/providers/api.provider.dart';
|
import 'package:immich_mobile/providers/api.provider.dart';
|
||||||
|
import 'package:immich_mobile/repositories/asset.repository.dart';
|
||||||
|
import 'package:immich_mobile/repositories/user.repository.dart';
|
||||||
import 'package:immich_mobile/services/api.service.dart';
|
import 'package:immich_mobile/services/api.service.dart';
|
||||||
import 'package:logging/logging.dart';
|
import 'package:immich_mobile/widgets/asset_grid/asset_grid_data_structure.dart';
|
||||||
import 'package:openapi/api.dart';
|
import 'package:openapi/api.dart';
|
||||||
|
|
||||||
final trashServiceProvider = Provider<TrashService>((ref) {
|
final trashServiceProvider = Provider<TrashService>((ref) {
|
||||||
return TrashService(
|
return TrashService(
|
||||||
ref.watch(apiServiceProvider),
|
ref.watch(apiServiceProvider),
|
||||||
|
ref.watch(assetRepositoryProvider),
|
||||||
|
ref.watch(userRepositoryProvider),
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
class TrashService {
|
class TrashService {
|
||||||
final _log = Logger("TrashService");
|
|
||||||
|
|
||||||
final ApiService _apiService;
|
final ApiService _apiService;
|
||||||
|
final IAssetRepository _assetRepository;
|
||||||
|
final IUserRepository _userRepository;
|
||||||
|
|
||||||
TrashService(this._apiService);
|
TrashService(this._apiService, this._assetRepository, this._userRepository);
|
||||||
|
|
||||||
Future<bool> restoreAssets(Iterable<Asset> assetList) async {
|
Future<void> restoreAssets(Iterable<Asset> assetList) async {
|
||||||
try {
|
final remoteAssets = assetList.where((a) => a.isRemote);
|
||||||
List<String> remoteIds =
|
await _apiService.trashApi.restoreAssets(
|
||||||
assetList.where((a) => a.isRemote).map((e) => e.remoteId!).toList();
|
BulkIdsDto(ids: remoteAssets.map((e) => e.remoteId!).toList()),
|
||||||
await _apiService.trashApi.restoreAssets(BulkIdsDto(ids: remoteIds));
|
);
|
||||||
return true;
|
|
||||||
} catch (error, stack) {
|
|
||||||
_log.severe("Cannot restore assets", error, stack);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Future<bool> restoreAsset(Asset asset) async {
|
final updatedAssets = remoteAssets.map((asset) {
|
||||||
try {
|
asset.isTrashed = false;
|
||||||
if (asset.isRemote) {
|
return asset;
|
||||||
List<String> remoteId = [asset.remoteId!];
|
}).toList();
|
||||||
|
|
||||||
await _apiService.trashApi.restoreAssets(BulkIdsDto(ids: remoteId));
|
await _assetRepository.updateAll(updatedAssets);
|
||||||
}
|
|
||||||
return true;
|
|
||||||
} catch (error, stack) {
|
|
||||||
_log.severe("Cannot restore assets", error, stack);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> emptyTrash() async {
|
Future<void> emptyTrash() async {
|
||||||
try {
|
final user = await _userRepository.me();
|
||||||
await _apiService.trashApi.emptyTrash();
|
|
||||||
} catch (error, stack) {
|
await _apiService.trashApi.emptyTrash();
|
||||||
_log.severe("Cannot empty trash", error, stack);
|
|
||||||
}
|
final trashedAssets = await _assetRepository.getTrashAssets(user.isarId);
|
||||||
|
final ids = trashedAssets.map((e) => e.remoteId!).toList();
|
||||||
|
|
||||||
|
await _assetRepository.transaction(() async {
|
||||||
|
await _assetRepository.deleteAllByRemoteId(
|
||||||
|
ids,
|
||||||
|
state: AssetState.remote,
|
||||||
|
);
|
||||||
|
|
||||||
|
final merged = await _assetRepository.getAllByRemoteId(
|
||||||
|
ids,
|
||||||
|
state: AssetState.merged,
|
||||||
|
);
|
||||||
|
if (merged.isEmpty) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (final Asset asset in merged) {
|
||||||
|
asset.remoteId = null;
|
||||||
|
asset.isTrashed = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
await _assetRepository.updateAll(merged);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> restoreTrash() async {
|
Future<void> restoreTrash() async {
|
||||||
try {
|
final user = await _userRepository.me();
|
||||||
await _apiService.trashApi.restoreTrash();
|
|
||||||
} catch (error, stack) {
|
await _apiService.trashApi.restoreTrash();
|
||||||
_log.severe("Cannot restore trash", error, stack);
|
|
||||||
}
|
final trashedAssets = await _assetRepository.getTrashAssets(user.isarId);
|
||||||
|
final updatedAssets = trashedAssets.map((asset) {
|
||||||
|
asset.isTrashed = false;
|
||||||
|
return asset;
|
||||||
|
}).toList();
|
||||||
|
|
||||||
|
await _assetRepository.updateAll(updatedAssets);
|
||||||
|
}
|
||||||
|
|
||||||
|
Stream<RenderList> getRenderListGenerator(int userId) {
|
||||||
|
return _assetRepository.getTrashRenderListStream(userId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -49,7 +49,8 @@ class GalleryAppBar extends ConsumerWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
handleRestore(Asset asset) async {
|
handleRestore(Asset asset) async {
|
||||||
final result = await ref.read(trashProvider.notifier).restoreAsset(asset);
|
final result =
|
||||||
|
await ref.read(trashProvider.notifier).restoreAssets([asset]);
|
||||||
|
|
||||||
if (result && context.mounted) {
|
if (result && context.mounted) {
|
||||||
ImmichToast.show(
|
ImmichToast.show(
|
||||||
|
3
mobile/openapi/devtools_options.yaml
Normal file
3
mobile/openapi/devtools_options.yaml
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
description: This file stores settings for Dart & Flutter DevTools.
|
||||||
|
documentation: https://docs.flutter.dev/tools/devtools/extensions#configure-extension-enablement-states
|
||||||
|
extensions:
|
Loading…
x
Reference in New Issue
Block a user