spotube/lib/provider/proxy_playlist/skip_segments.dart
Kingkor Roy Tirtho 22a49e56a2
refactor: use tcp server based track matcher (#1386)
* refactor: remove SourcedTrack based audio player and utilize mediakit playback system

* feat: implement local (loopback) server to resolve stream source and leverage the media_kit playback API

* feat: add source change support and re-add prefetching tracks

* fix: assign lastId when track fetch completes regardless of error

* chore: remove print statements

* fix: remote queue not working

* fix: increase mpv network timeout to reduce auto-skipping

* fix: do not pre-fetch local tracks

* fix(proxy-playlist): reset collections on load

* chore: fix lint warnings

* fix(mobile): player overlay should not be visible when the player is not playing

* chore: fix typo in turkish translation

* cd: checkout PR branch

* cd: upgrade flutter version

* chore: fix lint errors
2024-04-11 17:56:41 +06:00

105 lines
2.8 KiB
Dart

import 'dart:convert';
import 'package:catcher_2/catcher_2.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:http/http.dart';
import 'package:spotube/models/skip_segment.dart';
import 'package:spotube/provider/server/active_sourced_track.dart';
import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
import 'package:spotube/provider/user_preferences/user_preferences_state.dart';
class SourcedSegments {
final String source;
final List<SkipSegment> segments;
SourcedSegments({required this.source, required this.segments});
}
Future<List<SkipSegment>> getAndCacheSkipSegments(String id) async {
try {
final cached = await SkipSegment.box.get(id) as List?;
if (cached != null && cached.isNotEmpty) {
return List.castFrom<dynamic, SkipSegment>(
cached
.map(
(json) => SkipSegment.fromJson(
Map.castFrom<dynamic, dynamic, String, dynamic>(json),
),
)
.toList(),
);
}
final res = await get(Uri(
scheme: "https",
host: "sponsor.ajay.app",
path: "/api/skipSegments",
queryParameters: {
"videoID": id,
"category": [
'sponsor',
'selfpromo',
'interaction',
'intro',
'outro',
'music_offtopic'
],
"actionType": 'skip'
},
));
if (res.body == "Not Found") {
return List.castFrom<dynamic, SkipSegment>([]);
}
final data = jsonDecode(res.body) as List;
final segments = data.map((obj) {
final start = obj["segment"].first.toInt();
final end = obj["segment"].last.toInt();
return SkipSegment(start, end);
}).toList();
await SkipSegment.box.put(
id,
segments.map((e) => e.toJson()).toList(),
);
return List.castFrom<dynamic, SkipSegment>(segments);
} catch (e, stack) {
await SkipSegment.box.put(id, []);
Catcher2.reportCheckedError(e, stack);
return List.castFrom<dynamic, SkipSegment>([]);
}
}
final segmentProvider = FutureProvider<SourcedSegments?>(
(ref) async {
final track = ref.watch(activeSourcedTrackProvider);
if (track == null) return null;
final skipNonMusic = ref.watch(
userPreferencesProvider.select(
(s) {
final isPipedYTMusicMode = s.audioSource == AudioSource.piped &&
s.searchMode == SearchMode.youtubeMusic;
return s.skipNonMusic && !isPipedYTMusicMode;
},
),
);
if (!skipNonMusic) {
return SourcedSegments(
segments: [],
source: track.sourceInfo.id,
);
}
final segments = await getAndCacheSkipSegments(track.sourceInfo.id);
return SourcedSegments(
source: track.sourceInfo.id,
segments: segments,
);
},
);