This commit is contained in:
Seungmin Kim 2025-03-25 16:42:25 +00:00 committed by GitHub
commit 0a6292c3de
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
5 changed files with 85 additions and 48 deletions

View File

@ -94,6 +94,7 @@ abstract class FakeData {
..trackNumber = 1 ..trackNumber = 1
..type = "type" ..type = "type"
..uri = "uri" ..uri = "uri"
..externalIds = externalIds
..isPlayable = true ..isPlayable = true
..explicit = false ..explicit = false
..linkedFrom = trackLink; ..linkedFrom = trackLink;

View File

@ -4,7 +4,9 @@ import 'dart:typed_data';
import 'package:metadata_god/metadata_god.dart'; import 'package:metadata_god/metadata_god.dart';
import 'package:path/path.dart'; import 'package:path/path.dart';
import 'package:spotify/spotify.dart'; import 'package:spotify/spotify.dart';
import 'package:spotube/provider/spotify/spotify.dart';
import 'package:spotube/services/audio_player/audio_player.dart'; import 'package:spotube/services/audio_player/audio_player.dart';
import 'package:spotube/services/logger/logger.dart';
extension TrackExtensions on Track { extension TrackExtensions on Track {
Track fromFile( Track fromFile(
@ -68,7 +70,16 @@ extension TrackExtensions on Track {
} }
extension TrackSimpleExtensions on TrackSimple { extension TrackSimpleExtensions on TrackSimple {
Track asTrack(AlbumSimple album) { Future<Track> asTrack(AlbumSimple album, ref) async {
try {
final spotify = ref.read(spotifyProvider);
return await spotify.invoke((api) => api.tracks.get(id!));
} catch (e, stack) {
if ((e is AuthorizationException && e.error != 'invalid_token') ||
e is SpotifyException) {
// Ignore this error and create the Track locally
AppLogger.reportError(e, stack);
Track track = Track(); Track track = Track();
track.name = name; track.name = name;
track.album = album; track.album = album;
@ -89,6 +100,9 @@ extension TrackSimpleExtensions on TrackSimple {
track.uri = uri; track.uri = uri;
return track; return track;
} }
rethrow;
}
}
} }
extension TracksToMediaExtension on Iterable<Track> { extension TracksToMediaExtension on Iterable<Track> {

View File

@ -54,7 +54,8 @@ class AlbumCard extends HookConsumerWidget {
Future<List<Track>> fetchAllTrack() async { Future<List<Track>> fetchAllTrack() async {
if (album.tracks != null && album.tracks!.isNotEmpty) { if (album.tracks != null && album.tracks!.isNotEmpty) {
return album.tracks!.map((track) => track.asTrack(album)).toList(); return Future.wait(
album.tracks!.map((track) => track.asTrack(album, ref)).toList());
} }
await ref.read(albumTracksProvider(album).future); await ref.read(albumTracksProvider(album).future);
return ref.read(albumTracksProvider(album).notifier).fetchAll(); return ref.read(albumTracksProvider(album).notifier).fetchAll();

View File

@ -33,7 +33,7 @@ class AlbumTracksNotifier extends AutoDisposeFamilyPaginatedAsyncNotifier<Track,
final tracks = await spotify.invoke( final tracks = await spotify.invoke(
(api) => api.albums.tracks(arg.id!).getPage(limit, offset), (api) => api.albums.tracks(arg.id!).getPage(limit, offset),
); );
final items = tracks.items?.map((e) => e.asTrack(arg)).toList() ?? []; final List<Track> items = await Future.wait(tracks.items?.map((e) => e.asTrack(arg, ref)).toList() ?? []);
return ( return (
items: items, items: items,

View File

@ -236,29 +236,42 @@ class YoutubeSourcedTrack extends SourcedTrack {
.toList(); .toList();
} }
static Future<List<YoutubeVideoInfo>> fetchFromIsrc({
required Track track,
required Provider provider,
required Ref ref,
}) async {
final isrcResults = <YoutubeVideoInfo>[];
final isrc = track.externalIds?.isrc;
if (isrc != null && isrc.isNotEmpty) {
final searchedVideos = await ref.read(provider)
.searchVideos(isrc.toString());
isrcResults.addAll(await searchedVideos
.map<YoutubeVideoInfo>(YoutubeVideoInfo.fromVideo)
.toList());
}
return isrcResults;
}
static Future<List<SiblingType>> fetchSiblings({ static Future<List<SiblingType>> fetchSiblings({
required Track track, required Track track,
required Ref ref, required Ref ref,
}) async { }) async {
final videoResults = <YoutubeVideoInfo>[];
videoResults.addAll(await fetchFromIsrc(
track: track, provider: youtubeEngineProvider, ref: ref));
final links = await SongLinkService.links(track.id!); final links = await SongLinkService.links(track.id!);
final ytLink = links.firstWhereOrNull((link) => link.platform == "youtube"); final ytLink = links.firstWhereOrNull((link) => link.platform == "youtube");
if (ytLink?.url != null if (ytLink?.url != null) {
// allows to fetch siblings more results for already sourced track
&&
track is! SourcedTrack) {
try { try {
return [ videoResults.add(YoutubeVideoInfo.fromVideo(
await toSiblingType( await ref
0, .read(youtubeEngineProvider)
YoutubeVideoInfo.fromVideo( .getVideo(Uri.parse(ytLink!.url!).queryParameters["v"]!),
await ref.read(youtubeEngineProvider).getVideo( ));
Uri.parse(ytLink!.url!).queryParameters["v"]!,
),
),
ref,
)
];
} on VideoUnplayableException catch (e, stack) { } on VideoUnplayableException catch (e, stack) {
// Ignore this error and continue with the search // Ignore this error and continue with the search
AppLogger.reportError(e, stack); AppLogger.reportError(e, stack);
@ -271,20 +284,28 @@ class YoutubeSourcedTrack extends SourcedTrack {
await ref.read(youtubeEngineProvider).searchVideos(query); await ref.read(youtubeEngineProvider).searchVideos(query);
if (ServiceUtils.onlyContainsEnglish(query)) { if (ServiceUtils.onlyContainsEnglish(query)) {
return await Future.wait(searchResults videoResults.addAll(
.map(YoutubeVideoInfo.fromVideo) searchResults.map(YoutubeVideoInfo.fromVideo).toList(),
.mapIndexed((index, info) => toSiblingType(index, info, ref))); );
} } else {
videoResults.addAll(rankResults(
final rankedSiblings = rankResults(
searchResults.map(YoutubeVideoInfo.fromVideo).toList(), searchResults.map(YoutubeVideoInfo.fromVideo).toList(),
track, track,
); ));
}
final seenIds = <String>{};
int index = 0;
return await Future.wait( return await Future.wait(
rankedSiblings videoResults.map((videoResult) async {
.mapIndexed((index, info) => toSiblingType(index, info, ref)), // Deduplicate results
); if (!seenIds.contains(videoResult.id)) {
seenIds.add(videoResult.id);
return await toSiblingType(index++, videoResult, ref);
}
return null;
}),
).then((s) => s.whereType<SiblingType>().toList());
} }
@override @override