spotube/lib/provider/spotify/artist/following.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

105 lines
2.5 KiB
Dart

part of '../spotify.dart';
class FollowedArtistsState extends CursorPaginatedState<Artist> {
FollowedArtistsState({
required super.items,
required super.offset,
required super.limit,
required super.hasMore,
});
@override
FollowedArtistsState copyWith({
List<Artist>? items,
String? offset,
int? limit,
bool? hasMore,
}) {
return FollowedArtistsState(
items: items ?? this.items,
offset: offset ?? this.offset,
limit: limit ?? this.limit,
hasMore: hasMore ?? this.hasMore,
);
}
}
class FollowedArtistsNotifier
extends CursorPaginatedAsyncNotifier<Artist, FollowedArtistsState> {
FollowedArtistsNotifier() : super();
@override
fetch(offset, limit) async {
final artists = await spotify.me.following(FollowingType.artist).getPage(
limit,
offset ?? '',
);
return (artists.items?.toList() ?? [], artists.after);
}
@override
build() async {
ref.watch(spotifyProvider);
final (artists, nextCursor) = await fetch(null, 50);
return FollowedArtistsState(
items: artists,
offset: nextCursor,
limit: 50,
hasMore: artists.length == 50,
);
}
Future<void> saveArtists(List<String> artistIds) async {
if (state.value == null) return;
await spotify.me.follow(FollowingType.artist, artistIds);
state = await AsyncValue.guard(() async {
final artists = await spotify.artists.list(artistIds);
return state.value!.copyWith(
items: [
...state.value!.items,
...artists,
],
);
});
for (final id in artistIds) {
ref.invalidate(artistIsFollowingProvider(id));
}
}
Future<void> removeArtists(List<String> artistIds) async {
if (state.value == null) return;
await spotify.me.unfollow(FollowingType.artist, artistIds);
state = await AsyncValue.guard(() async {
final artists = state.value!.items.where((artist) {
return !artistIds.contains(artist.id);
}).toList();
return state.value!.copyWith(
items: artists,
);
});
for (final id in artistIds) {
ref.invalidate(artistIsFollowingProvider(id));
}
}
}
final followedArtistsProvider =
AsyncNotifierProvider<FollowedArtistsNotifier, FollowedArtistsState>(
() => FollowedArtistsNotifier(),
);
final allFollowedArtistsProvider = FutureProvider<List<Artist>>(
(ref) async {
final spotify = ref.watch(spotifyProvider);
final artists = await spotify.me.following(FollowingType.artist).all();
return artists.toList();
},
);