2023-05-17 19:36:02 +02:00
|
|
|
import 'dart:async';
|
|
|
|
|
2022-08-03 07:04:34 +02:00
|
|
|
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
2024-12-16 17:11:48 +01:00
|
|
|
import 'package:immich_mobile/constants/enums.dart';
|
2024-10-10 10:44:14 +02:00
|
|
|
import 'package:immich_mobile/entities/user.entity.dart';
|
|
|
|
import 'package:immich_mobile/models/albums/album_search.model.dart';
|
2024-05-02 22:59:14 +02:00
|
|
|
import 'package:immich_mobile/services/album.service.dart';
|
2024-05-07 06:04:21 +02:00
|
|
|
import 'package:immich_mobile/widgets/asset_grid/asset_grid_data_structure.dart';
|
2024-05-01 04:36:40 +02:00
|
|
|
import 'package:immich_mobile/entities/asset.entity.dart';
|
|
|
|
import 'package:immich_mobile/entities/album.entity.dart';
|
2024-05-02 22:59:14 +02:00
|
|
|
import 'package:immich_mobile/providers/db.provider.dart';
|
2023-12-07 16:38:22 +01:00
|
|
|
import 'package:immich_mobile/utils/renderlist_generator.dart';
|
2023-03-03 23:38:30 +01:00
|
|
|
import 'package:isar/isar.dart';
|
2022-08-03 07:04:34 +02:00
|
|
|
|
2024-10-10 10:44:14 +02:00
|
|
|
final isRefreshingRemoteAlbumProvider = StateProvider<bool>((ref) => false);
|
|
|
|
|
2023-02-06 08:13:32 +01:00
|
|
|
class AlbumNotifier extends StateNotifier<List<Album>> {
|
2024-10-10 10:44:14 +02:00
|
|
|
AlbumNotifier(this._albumService, this.db, this.ref) : super([]) {
|
|
|
|
final query = db.albums.filter().remoteIdIsNotNull();
|
2024-01-10 16:31:54 +01:00
|
|
|
query.findAll().then((value) {
|
|
|
|
if (mounted) {
|
|
|
|
state = value;
|
|
|
|
}
|
|
|
|
});
|
2023-05-17 19:36:02 +02:00
|
|
|
_streamSub = query.watch().listen((data) => state = data);
|
2022-08-03 07:04:34 +02:00
|
|
|
}
|
2024-08-26 20:21:19 +02:00
|
|
|
|
2023-05-17 19:36:02 +02:00
|
|
|
final AlbumService _albumService;
|
2024-10-10 10:44:14 +02:00
|
|
|
final Isar db;
|
|
|
|
final Ref ref;
|
2023-05-17 19:36:02 +02:00
|
|
|
late final StreamSubscription<List<Album>> _streamSub;
|
2022-08-03 07:04:34 +02:00
|
|
|
|
2024-10-10 10:44:14 +02:00
|
|
|
Future<void> refreshRemoteAlbums() async {
|
|
|
|
ref.read(isRefreshingRemoteAlbumProvider.notifier).state = true;
|
|
|
|
await _albumService.refreshRemoteAlbums();
|
|
|
|
ref.read(isRefreshingRemoteAlbumProvider.notifier).state = false;
|
|
|
|
}
|
2023-05-17 19:36:02 +02:00
|
|
|
|
2024-10-10 10:44:14 +02:00
|
|
|
Future<void> refreshDeviceAlbums() => _albumService.refreshDeviceAlbums();
|
2023-11-14 21:30:27 +01:00
|
|
|
|
2023-05-17 19:36:02 +02:00
|
|
|
Future<bool> deleteAlbum(Album album) => _albumService.deleteAlbum(album);
|
2022-08-03 07:04:34 +02:00
|
|
|
|
2023-02-06 08:13:32 +01:00
|
|
|
Future<Album?> createAlbum(
|
2022-08-03 22:36:12 +02:00
|
|
|
String albumTitle,
|
2022-11-08 18:00:24 +01:00
|
|
|
Set<Asset> assets,
|
2023-05-17 19:36:02 +02:00
|
|
|
) =>
|
|
|
|
_albumService.createAlbum(albumTitle, assets, []);
|
|
|
|
|
2024-08-26 20:21:19 +02:00
|
|
|
Future<Album?> getAlbumByName(String albumName, {bool remoteOnly = false}) =>
|
|
|
|
_albumService.getAlbumByName(albumName, remoteOnly);
|
|
|
|
|
|
|
|
/// Create an album on the server with the same name as the selected album for backup
|
|
|
|
/// First this will check if the album already exists on the server with name
|
|
|
|
/// If it does not exist, it will create the album on the server
|
|
|
|
Future<void> createSyncAlbum(
|
|
|
|
String albumName,
|
|
|
|
) async {
|
|
|
|
final album = await getAlbumByName(albumName, remoteOnly: true);
|
|
|
|
if (album != null) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
await createAlbum(albumName, {});
|
|
|
|
}
|
|
|
|
|
2024-10-10 10:44:14 +02:00
|
|
|
Future<bool> leaveAlbum(Album album) async {
|
|
|
|
var res = await _albumService.leaveAlbum(album);
|
|
|
|
|
|
|
|
if (res) {
|
|
|
|
await deleteAlbum(album);
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void searchAlbums(String searchTerm, QuickFilterMode filterMode) async {
|
|
|
|
state = await _albumService.search(searchTerm, filterMode);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> addUsers(Album album, List<String> userIds) async {
|
|
|
|
await _albumService.addUsers(album, userIds);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<bool> removeUser(Album album, User user) async {
|
|
|
|
final isRemoved = await _albumService.removeUser(album, user);
|
|
|
|
|
|
|
|
if (isRemoved && album.sharedUsers.isEmpty) {
|
|
|
|
state = state.where((element) => element.id != album.id).toList();
|
|
|
|
}
|
|
|
|
|
|
|
|
return isRemoved;
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<void> addAssets(Album album, Iterable<Asset> assets) async {
|
|
|
|
await _albumService.addAssets(album, assets);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<bool> removeAsset(Album album, Iterable<Asset> assets) async {
|
|
|
|
return await _albumService.removeAsset(album, assets);
|
|
|
|
}
|
|
|
|
|
|
|
|
Future<bool> setActivitystatus(
|
|
|
|
Album album,
|
|
|
|
bool enabled,
|
|
|
|
) {
|
|
|
|
return _albumService.setActivityStatus(album, enabled);
|
|
|
|
}
|
|
|
|
|
2024-12-16 17:11:48 +01:00
|
|
|
Future<Album?> toggleSortOrder(Album album) {
|
|
|
|
final order =
|
|
|
|
album.sortOrder == SortOrder.asc ? SortOrder.desc : SortOrder.asc;
|
|
|
|
|
|
|
|
return _albumService.updateSortOrder(album, order);
|
|
|
|
}
|
|
|
|
|
2023-05-17 19:36:02 +02:00
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
_streamSub.cancel();
|
|
|
|
super.dispose();
|
2022-08-03 07:04:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-17 19:36:02 +02:00
|
|
|
final albumProvider =
|
|
|
|
StateNotifierProvider.autoDispose<AlbumNotifier, List<Album>>((ref) {
|
2022-10-17 14:53:27 +02:00
|
|
|
return AlbumNotifier(
|
|
|
|
ref.watch(albumServiceProvider),
|
2023-03-03 23:38:30 +01:00
|
|
|
ref.watch(dbProvider),
|
2024-10-10 10:44:14 +02:00
|
|
|
ref,
|
2022-10-17 14:53:27 +02:00
|
|
|
);
|
2022-08-03 07:04:34 +02:00
|
|
|
});
|
2023-12-07 16:38:22 +01:00
|
|
|
|
|
|
|
final albumWatcher =
|
|
|
|
StreamProvider.autoDispose.family<Album, int>((ref, albumId) async* {
|
|
|
|
final db = ref.watch(dbProvider);
|
|
|
|
final a = await db.albums.get(albumId);
|
|
|
|
if (a != null) yield a;
|
|
|
|
await for (final a in db.albums.watchObject(albumId, fireImmediately: true)) {
|
|
|
|
if (a != null) yield a;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
final albumRenderlistProvider =
|
|
|
|
StreamProvider.autoDispose.family<RenderList, int>((ref, albumId) {
|
|
|
|
final album = ref.watch(albumWatcher(albumId)).value;
|
2024-12-16 17:11:48 +01:00
|
|
|
|
2023-12-07 16:38:22 +01:00
|
|
|
if (album != null) {
|
2024-12-16 17:11:48 +01:00
|
|
|
final query = album.assets.filter().isTrashedEqualTo(false);
|
|
|
|
if (album.sortOrder == SortOrder.asc) {
|
|
|
|
return renderListGeneratorWithGroupBy(
|
|
|
|
query.sortByFileCreatedAt(),
|
|
|
|
GroupAssetsBy.none,
|
|
|
|
);
|
|
|
|
} else if (album.sortOrder == SortOrder.desc) {
|
|
|
|
return renderListGeneratorWithGroupBy(
|
|
|
|
query.sortByFileCreatedAtDesc(),
|
|
|
|
GroupAssetsBy.none,
|
|
|
|
);
|
|
|
|
}
|
2023-12-07 16:38:22 +01:00
|
|
|
}
|
2024-12-16 17:11:48 +01:00
|
|
|
|
2023-12-07 16:38:22 +01:00
|
|
|
return const Stream.empty();
|
|
|
|
});
|
2024-10-10 10:44:14 +02:00
|
|
|
|
|
|
|
class LocalAlbumsNotifier extends StateNotifier<List<Album>> {
|
|
|
|
LocalAlbumsNotifier(this.db) : super([]) {
|
|
|
|
final query = db.albums.where().remoteIdIsNull();
|
|
|
|
|
|
|
|
query.findAll().then((value) {
|
|
|
|
if (mounted) {
|
|
|
|
state = value;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
_streamSub = query.watch().listen((data) => state = data);
|
|
|
|
}
|
|
|
|
|
|
|
|
final Isar db;
|
|
|
|
late final StreamSubscription<List<Album>> _streamSub;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() {
|
|
|
|
_streamSub.cancel();
|
|
|
|
super.dispose();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final localAlbumsProvider =
|
|
|
|
StateNotifierProvider.autoDispose<LocalAlbumsNotifier, List<Album>>((ref) {
|
|
|
|
return LocalAlbumsNotifier(ref.watch(dbProvider));
|
|
|
|
});
|