2024-08-26 23:21:22 +08:00
|
|
|
import 'package:get/get.dart';
|
|
|
|
import 'package:go_router/go_router.dart';
|
|
|
|
import 'package:html/dom.dart' hide Text;
|
|
|
|
import 'package:rhythm_box/services/sort.dart';
|
|
|
|
import 'package:spotify/spotify.dart';
|
|
|
|
|
|
|
|
import 'package:html/parser.dart' as parser;
|
|
|
|
|
|
|
|
import 'dart:async';
|
|
|
|
|
|
|
|
import 'package:flutter/material.dart' hide Element;
|
|
|
|
|
|
|
|
abstract class ServiceUtils {
|
|
|
|
static final _englishMatcherRegex = RegExp(
|
|
|
|
"^[a-zA-Z0-9\\s!\"#\$%&\\'()*+,-.\\/:;<=>?@\\[\\]^_`{|}~]*\$",
|
|
|
|
);
|
|
|
|
static bool onlyContainsEnglish(String text) {
|
|
|
|
return _englishMatcherRegex.hasMatch(text);
|
|
|
|
}
|
|
|
|
|
|
|
|
static String clearArtistsOfTitle(String title, List<String> artists) {
|
|
|
|
return title
|
2024-08-27 14:48:31 +08:00
|
|
|
.replaceAll(RegExp(artists.join('|'), caseSensitive: false), '')
|
2024-08-26 23:21:22 +08:00
|
|
|
.trim();
|
|
|
|
}
|
|
|
|
|
|
|
|
static String getTitle(
|
|
|
|
String title, {
|
|
|
|
List<String> artists = const [],
|
|
|
|
bool onlyCleanArtist = false,
|
|
|
|
}) {
|
2024-08-27 14:48:31 +08:00
|
|
|
final match = RegExp(r'(?<=\().+?(?=\))').firstMatch(title)?.group(0);
|
2024-08-26 23:21:22 +08:00
|
|
|
final artistInBracket =
|
|
|
|
artists.any((artist) => match?.contains(artist) ?? false);
|
|
|
|
|
|
|
|
if (artistInBracket) {
|
|
|
|
title = title.replaceAll(
|
2024-08-27 14:48:31 +08:00
|
|
|
RegExp(' *\\([^)]*\\) *'),
|
2024-08-26 23:21:22 +08:00
|
|
|
'',
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
title = clearArtistsOfTitle(title, artists);
|
|
|
|
if (onlyCleanArtist) {
|
|
|
|
artists = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
return "$title ${artists.map((e) => e.replaceAll(",", " ")).join(", ")}"
|
|
|
|
.toLowerCase()
|
2024-08-27 14:48:31 +08:00
|
|
|
.replaceAll(RegExp(r'\s*\[[^\]]*]'), ' ')
|
|
|
|
.replaceAll(RegExp(r'\sfeat\.|\sft\.'), ' ')
|
|
|
|
.replaceAll(RegExp(r'\s+'), ' ')
|
2024-08-26 23:21:22 +08:00
|
|
|
.trim();
|
|
|
|
}
|
|
|
|
|
|
|
|
static Future<String?> extractLyrics(Uri url) async {
|
|
|
|
final client = GetConnect();
|
|
|
|
final response = await client.get(url.toString());
|
|
|
|
|
|
|
|
Document document = parser.parse(response.body);
|
|
|
|
String? lyrics = document.querySelector('div.lyrics')?.text.trim();
|
|
|
|
if (lyrics == null) {
|
2024-08-27 14:48:31 +08:00
|
|
|
lyrics = '';
|
2024-08-26 23:21:22 +08:00
|
|
|
document
|
2024-08-27 14:48:31 +08:00
|
|
|
.querySelectorAll('div[class^="Lyrics__Container"]')
|
2024-08-26 23:21:22 +08:00
|
|
|
.forEach((element) {
|
|
|
|
if (element.text.trim().isNotEmpty) {
|
2024-08-27 14:48:31 +08:00
|
|
|
final snippet = element.innerHtml.replaceAll('<br>', '\n').replaceAll(
|
|
|
|
RegExp('<(?!\\s*br\\s*\\/?)[^>]+>', caseSensitive: false),
|
|
|
|
'',
|
2024-08-26 23:21:22 +08:00
|
|
|
);
|
2024-08-27 14:48:31 +08:00
|
|
|
final el = document.createElement('textarea');
|
2024-08-26 23:21:22 +08:00
|
|
|
el.innerHtml = snippet;
|
2024-08-27 14:48:31 +08:00
|
|
|
lyrics = '$lyrics${el.text.trim()}\n\n';
|
2024-08-26 23:21:22 +08:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return lyrics;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void navigate(BuildContext context, String location, {Object? extra}) {
|
|
|
|
if (GoRouterState.of(context).matchedLocation == location) return;
|
|
|
|
GoRouter.of(context).go(location, extra: extra);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void navigateNamed(
|
|
|
|
BuildContext context,
|
|
|
|
String name, {
|
|
|
|
Object? extra,
|
|
|
|
Map<String, String>? pathParameters,
|
|
|
|
Map<String, dynamic>? queryParameters,
|
|
|
|
}) {
|
|
|
|
if (GoRouterState.of(context).matchedLocation == name) return;
|
|
|
|
GoRouter.of(context).goNamed(
|
|
|
|
name,
|
|
|
|
pathParameters: pathParameters ?? const {},
|
|
|
|
queryParameters: queryParameters ?? const {},
|
|
|
|
extra: extra,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void push(BuildContext context, String location, {Object? extra}) {
|
|
|
|
final router = GoRouter.of(context);
|
|
|
|
final routerState = GoRouterState.of(context);
|
|
|
|
final routerStack = router.routerDelegate.currentConfiguration.matches
|
|
|
|
.map((e) => e.matchedLocation);
|
|
|
|
|
|
|
|
if (routerState.matchedLocation == location ||
|
|
|
|
routerStack.contains(location)) return;
|
|
|
|
router.push(location, extra: extra);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void pushNamed(
|
|
|
|
BuildContext context,
|
|
|
|
String name, {
|
|
|
|
Object? extra,
|
|
|
|
Map<String, String> pathParameters = const {},
|
|
|
|
Map<String, String> queryParameters = const {},
|
|
|
|
}) {
|
|
|
|
final router = GoRouter.of(context);
|
|
|
|
final routerState = GoRouterState.of(context);
|
|
|
|
final routerStack = router.routerDelegate.currentConfiguration.matches
|
|
|
|
.map((e) => e.matchedLocation);
|
|
|
|
|
|
|
|
final nameLocation = routerState.namedLocation(
|
|
|
|
name,
|
|
|
|
pathParameters: pathParameters,
|
|
|
|
queryParameters: queryParameters,
|
|
|
|
);
|
|
|
|
|
|
|
|
if (routerState.matchedLocation == nameLocation ||
|
|
|
|
routerStack.contains(nameLocation)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
router.pushNamed(
|
|
|
|
name,
|
|
|
|
pathParameters: pathParameters,
|
|
|
|
queryParameters: queryParameters,
|
|
|
|
extra: extra,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
static DateTime parseSpotifyAlbumDate(AlbumSimple? album) {
|
|
|
|
if (album == null || album.releaseDate == null) {
|
2024-08-27 14:48:31 +08:00
|
|
|
return DateTime.parse('1975-01-01');
|
2024-08-26 23:21:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
switch (album.releaseDatePrecision ?? DatePrecision.year) {
|
|
|
|
case DatePrecision.day:
|
|
|
|
return DateTime.parse(album.releaseDate!);
|
|
|
|
case DatePrecision.month:
|
2024-08-27 14:48:31 +08:00
|
|
|
return DateTime.parse('${album.releaseDate}-01');
|
2024-08-26 23:21:22 +08:00
|
|
|
case DatePrecision.year:
|
2024-08-27 14:48:31 +08:00
|
|
|
return DateTime.parse('${album.releaseDate}-01-01');
|
2024-08-26 23:21:22 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static List<T> sortTracks<T extends Track>(List<T> tracks, SortBy sortBy) {
|
|
|
|
if (sortBy == SortBy.none) return tracks;
|
|
|
|
return List<T>.from(tracks)
|
|
|
|
..sort((a, b) {
|
|
|
|
switch (sortBy) {
|
|
|
|
case SortBy.ascending:
|
2024-08-27 14:48:31 +08:00
|
|
|
return a.name?.compareTo(b.name ?? '') ?? 0;
|
2024-08-26 23:21:22 +08:00
|
|
|
case SortBy.descending:
|
2024-08-27 14:48:31 +08:00
|
|
|
return b.name?.compareTo(a.name ?? '') ?? 0;
|
2024-08-26 23:21:22 +08:00
|
|
|
case SortBy.newest:
|
|
|
|
final aDate = parseSpotifyAlbumDate(a.album);
|
|
|
|
final bDate = parseSpotifyAlbumDate(b.album);
|
|
|
|
return bDate.compareTo(aDate);
|
|
|
|
case SortBy.oldest:
|
|
|
|
final aDate = parseSpotifyAlbumDate(a.album);
|
|
|
|
final bDate = parseSpotifyAlbumDate(b.album);
|
|
|
|
return aDate.compareTo(bDate);
|
|
|
|
case SortBy.duration:
|
|
|
|
return a.durationMs?.compareTo(b.durationMs ?? 0) ?? 0;
|
|
|
|
case SortBy.artist:
|
|
|
|
return a.artists?.first.name
|
2024-08-27 14:48:31 +08:00
|
|
|
?.compareTo(b.artists?.first.name ?? '') ??
|
2024-08-26 23:21:22 +08:00
|
|
|
0;
|
|
|
|
case SortBy.album:
|
2024-08-27 14:48:31 +08:00
|
|
|
return a.album?.name?.compareTo(b.album?.name ?? '') ?? 0;
|
2024-08-26 23:21:22 +08:00
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|