spotube/lib/provider/spotify/playlist/favorite.dart
Kingkor Roy Tirtho 6673e5a8a8
feat: improved caching based on riverpod (#1343)
* feat: add riverpod based favorite album provider

* feat: add album is saved, new releases and tracks providers

* feat: add artist related providers

* feat: add all categories providers

* feat: add lyrics provider

* feat: add playlist related providers

* feat: add search provider

* feat: add view and spotify friends provider

* feat: add playlist create and update and favorite handlers

* feat: use providers in home screen

* chore: fix dart lint issues

* feat: use new providers for playlist and albums screen

* feat: use providers in artist page

* feat: use providers on library page

* feat: use provider for playlist and album card and heart button

* feat: use provider in search page

* feat: use providers in generate playlist

* feat: use provider in lyrics screen

* feat: use provider for create playlist

* feat: use provider in add track dialog

* feat: use providers in remaining pages and remove fl_query

* fix: remove direct access to provider.value

* fix: glitching when loading

* fix: user album loading next page indicator

* feat: make many provider autoDispose after 5 minutes of no usage

* fix: ignore episodes in tracks
2024-03-20 23:38:39 +06:00

123 lines
3.0 KiB
Dart

part of '../spotify.dart';
class FavoritePlaylistsState extends PaginatedState<PlaylistSimple> {
FavoritePlaylistsState({
required super.items,
required super.offset,
required super.limit,
required super.hasMore,
});
@override
FavoritePlaylistsState copyWith({
List<PlaylistSimple>? items,
int? offset,
int? limit,
bool? hasMore,
}) {
return FavoritePlaylistsState(
items: items ?? this.items,
offset: offset ?? this.offset,
limit: limit ?? this.limit,
hasMore: hasMore ?? this.hasMore,
);
}
}
class FavoritePlaylistsNotifier
extends PaginatedAsyncNotifier<PlaylistSimple, FavoritePlaylistsState> {
FavoritePlaylistsNotifier() : super();
@override
fetch(int offset, int limit) async {
final playlists = await spotify.playlists.me.getPage(
limit,
offset,
);
return playlists.items?.toList() ?? [];
}
@override
build() async {
ref.watch(spotifyProvider);
final playlists = await fetch(0, 20);
return FavoritePlaylistsState(
items: playlists,
offset: 0,
limit: 20,
hasMore: playlists.length == 20,
);
}
Future<void> addFavorite(PlaylistSimple playlist) async {
await update((state) async {
await spotify.playlists.followPlaylist(playlist.id!);
return state.copyWith(
items: [...state.items, playlist],
);
});
ref.invalidate(isFavoritePlaylistProvider(playlist.id!));
}
Future<void> removeFavorite(PlaylistSimple playlist) async {
await update((state) async {
await spotify.playlists.unfollowPlaylist(playlist.id!);
return state.copyWith(
items: state.items.where((e) => e.id != playlist.id).toList(),
);
});
ref.invalidate(isFavoritePlaylistProvider(playlist.id!));
}
Future<void> addTracks(String playlistId, List<String> trackIds) async {
if (state.value == null) return;
final spotify = ref.read(spotifyProvider);
await spotify.playlists.addTracks(
trackIds.map((id) => 'spotify:track:$id').toList(),
playlistId,
);
ref.invalidate(playlistTracksProvider(playlistId));
}
Future<void> removeTracks(String playlistId, List<String> trackIds) async {
if (state.value == null) return;
final spotify = ref.read(spotifyProvider);
await spotify.playlists.removeTracks(
trackIds.map((id) => 'spotify:track:$id').toList(),
playlistId,
);
ref.invalidate(playlistTracksProvider(playlistId));
}
}
final favoritePlaylistsProvider =
AsyncNotifierProvider<FavoritePlaylistsNotifier, FavoritePlaylistsState>(
() => FavoritePlaylistsNotifier(),
);
final isFavoritePlaylistProvider = FutureProvider.family<bool, String>(
(ref, id) async {
final spotify = ref.watch(spotifyProvider);
final me = ref.watch(meProvider);
if (me.value == null) {
return false;
}
final follows =
await spotify.playlists.followedByUsers(id, [me.value!.id!]);
return follows[me.value!.id!] ?? false;
},
);