import 'package:collection/collection.dart'; import 'package:rhythm_box/services/utils.dart'; import 'package:spotify/spotify.dart'; import 'package:rhythm_box/services/sourced_track/enums.dart'; import 'package:rhythm_box/services/sourced_track/exceptions.dart'; import 'package:rhythm_box/services/sourced_track/models/source_info.dart'; import 'package:rhythm_box/services/sourced_track/models/source_map.dart'; import 'package:rhythm_box/services/sourced_track/sources/piped.dart'; import 'package:rhythm_box/services/sourced_track/sources/youtube.dart'; import 'package:youtube_explode_dart/youtube_explode_dart.dart'; abstract class SourcedTrack extends Track { final SourceMap source; final List siblings; final SourceInfo sourceInfo; SourcedTrack({ required this.source, required this.siblings, required this.sourceInfo, required Track track, }) { id = track.id; name = track.name; artists = track.artists; album = track.album; durationMs = track.durationMs; discNumber = track.discNumber; explicit = track.explicit; externalIds = track.externalIds; href = track.href; isPlayable = track.isPlayable; linkedFrom = track.linkedFrom; popularity = track.popularity; previewUrl = track.previewUrl; trackNumber = track.trackNumber; type = track.type; uri = track.uri; } static SourcedTrack fromJson(Map json) { // TODO Follow user preferences const audioSource = 'youtube'; final sourceInfo = SourceInfo.fromJson(json); final source = SourceMap.fromJson(json); final track = Track.fromJson(json); final siblings = (json['siblings'] as List) .map((sibling) => SourceInfo.fromJson(sibling)) .toList() .cast(); return switch (audioSource) { 'piped' => PipedSourcedTrack( source: source, siblings: siblings, sourceInfo: sourceInfo, track: track, ), _ => YoutubeSourcedTrack( source: source, siblings: siblings, sourceInfo: sourceInfo, track: track, ), }; } static String getSearchTerm(Track track) { final artists = (track.artists ?? []) .map((ar) => ar.name) .toList() .whereNotNull() .toList(); final title = ServiceUtils.getTitle( track.name!, artists: artists, onlyCleanArtist: true, ).trim(); return "$title - ${artists.join(", ")}"; } static Future fetchFromTrack({ required Track track, }) async { // TODO Follow user preferences const audioSource = 'youtube'; try { return switch (audioSource) { 'piped' => await PipedSourcedTrack.fetchFromTrack(track: track), _ => await YoutubeSourcedTrack.fetchFromTrack(track: track), }; } on TrackNotFoundError catch (_) { // TODO Try to look it up in other source // But the youtube and piped.video are the same, and there is no extra sources, so i ignored this for temporary rethrow; } on HttpClientClosedException catch (_) { return await PipedSourcedTrack.fetchFromTrack(track: track); } on VideoUnplayableException catch (_) { return await PipedSourcedTrack.fetchFromTrack(track: track); } catch (e) { rethrow; } } static Future> fetchSiblings({ required Track track, }) { // TODO Follow user preferences const audioSource = 'youtube'; return switch (audioSource) { 'piped' => PipedSourcedTrack.fetchSiblings(track: track), _ => YoutubeSourcedTrack.fetchSiblings(track: track), }; } Future copyWithSibling(); Future swapWithSibling(SourceInfo sibling); Future swapWithSiblingOfIndex(int index) { return swapWithSibling(siblings[index]); } String get url { // TODO Follow user preferences const streamMusicCodec = SourceCodecs.weba; return getUrlOfCodec(streamMusicCodec); } String getUrlOfCodec(SourceCodecs codec) { // TODO Follow user preferences const audioQuality = SourceQualities.high; return source[codec]?[audioQuality] ?? // this will ensure playback doesn't break source[codec == SourceCodecs.m4a ? SourceCodecs.weba : SourceCodecs.m4a] [audioQuality]; } SourceCodecs get codec { // TODO Follow user preferences const streamMusicCodec = SourceCodecs.weba; return streamMusicCodec; } }