feat(discord): album art, playing time and play pause support (#1765)

This commit is contained in:
nexpid 2024-08-10 17:30:11 +02:00 committed by GitHub
parent 6456b43d10
commit 64d25509b4
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

View File

@ -6,6 +6,7 @@ import 'package:spotify/spotify.dart';
import 'package:spotube/extensions/artist_simple.dart'; import 'package:spotube/extensions/artist_simple.dart';
import 'package:spotube/provider/audio_player/audio_player.dart'; import 'package:spotube/provider/audio_player/audio_player.dart';
import 'package:spotube/provider/user_preferences/user_preferences_provider.dart'; import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
import 'package:spotube/services/audio_player/audio_player.dart';
import 'package:spotube/utils/platform.dart'; import 'package:spotube/utils/platform.dart';
class DiscordNotifier extends AsyncNotifier<void> { class DiscordNotifier extends AsyncNotifier<void> {
@ -13,17 +14,39 @@ class DiscordNotifier extends AsyncNotifier<void> {
FutureOr<void> build() async { FutureOr<void> build() async {
final enabled = ref.watch( final enabled = ref.watch(
userPreferencesProvider.select((s) => s.discordPresence && kIsDesktop)); userPreferencesProvider.select((s) => s.discordPresence && kIsDesktop));
final playback = ref.read(audioPlayerProvider);
final subscription = var lastPosition = audioPlayer.position;
FlutterDiscordRPC.instance.isConnectedStream.listen((connected) async {
if (connected && playback.activeTrack != null) { final subscriptions =
await updatePresence(playback.activeTrack!); [
} FlutterDiscordRPC.instance.isConnectedStream.listen((connected) async {
}); final playback = ref.read(audioPlayerProvider);
if (connected && playback.activeTrack != null) {
await updatePresence(playback.activeTrack!);
}
}),
audioPlayer.playerStateStream.listen((state) async {
final playback = ref.read(audioPlayerProvider);
if (playback.activeTrack == null) return;
await updatePresence(ref.read(audioPlayerProvider).activeTrack!);
}),
audioPlayer.positionStream.listen((position) async {
final playback = ref.read(audioPlayerProvider);
if (playback.activeTrack != null) {
final diff = position.inMilliseconds - lastPosition.inMilliseconds;
if (diff > 500 || diff < -500) {
await updatePresence(ref.read(audioPlayerProvider).activeTrack!);
}
}
lastPosition = position;
})
];
ref.onDispose(() async { ref.onDispose(() async {
subscription.cancel(); for (final subscription in subscriptions) {
subscription.cancel();
}
await close(); await close();
await FlutterDiscordRPC.instance.dispose(); await FlutterDiscordRPC.instance.dispose();
}); });
@ -37,15 +60,18 @@ class DiscordNotifier extends AsyncNotifier<void> {
} }
Future<void> updatePresence(Track track) async { Future<void> updatePresence(Track track) async {
await clear(); final artistNames = track.artists?.asString();
final artistNames = track.artists?.asString() ?? ""; final isPlaying = audioPlayer.isPlaying;
final position = audioPlayer.position;
await FlutterDiscordRPC.instance.setActivity( await FlutterDiscordRPC.instance.setActivity(
activity: RPCActivity( activity: RPCActivity(
details: "${track.name} by $artistNames", details: track.name,
state: "Vibing in Music", state: artistNames != null ? "by $artistNames" : null,
assets: const RPCAssets( assets: RPCAssets(
largeImage: "spotube-logo-foreground", largeImage:
largeText: "Spotube", track.album?.images?.first.url ?? "spotube-logo-foreground",
largeText: track.album?.name ?? "Unknown album",
smallImage: "spotube-logo-foreground", smallImage: "spotube-logo-foreground",
smallText: "Spotube", smallText: "Spotube",
), ),
@ -57,7 +83,7 @@ class DiscordNotifier extends AsyncNotifier<void> {
), ),
], ],
timestamps: RPCTimestamps( timestamps: RPCTimestamps(
start: DateTime.now().millisecondsSinceEpoch, start: isPlaying ? DateTime.now().millisecondsSinceEpoch - position.inMilliseconds : null,
), ),
), ),
); );
@ -73,4 +99,4 @@ class DiscordNotifier extends AsyncNotifier<void> {
} }
final discordProvider = final discordProvider =
AsyncNotifierProvider<DiscordNotifier, void>(() => DiscordNotifier()); AsyncNotifierProvider<DiscordNotifier, void>(() => DiscordNotifier());