Merge branch 'dev' into feat/stats

This commit is contained in:
Kingkor Roy Tirtho 2024-05-23 21:43:56 +06:00
commit b828255aeb
39 changed files with 2453 additions and 567 deletions

View File

@ -41,4 +41,4 @@ abstract class Env {
kIsFlatpak || _enableUpdateChecker == "1";
static String discordAppId = "1176718791388975124";
}
}

View File

@ -81,10 +81,10 @@ abstract class LanguageLocals {
// name: "Bashkir",
// nativeName: "башҡорт теле",
// ),
// "eu": const ISOLanguageName(
// name: "Basque",
// nativeName: "euskara,",
// ),
"eu": const ISOLanguageName(
name: "Basque",
nativeName: "euskara",
),
// "be": const ISOLanguageName(
// name: "Belarusian",
// nativeName: "Беларуская",
@ -197,10 +197,10 @@ abstract class LanguageLocals {
// name: "Fijian",
// nativeName: "vosa Vakaviti",
// ),
// "fi": const ISOLanguageName(
// name: "Finnish",
// nativeName: "suomi",
// ),
"fi": const ISOLanguageName(
name: "Finnish",
nativeName: "suomi",
),
"fr": const ISOLanguageName(
name: "French",
nativeName: "français",
@ -213,10 +213,10 @@ abstract class LanguageLocals {
// name: "Galician",
// nativeName: "Galego",
// ),
// "ka": const ISOLanguageName(
// name: "Georgian",
// nativeName: "ქართული",
// ),
"ka": const ISOLanguageName(
name: "Georgian",
nativeName: "ქართული",
),
"de": const ISOLanguageName(
name: "German",
nativeName: "Deutsch",
@ -265,10 +265,10 @@ abstract class LanguageLocals {
// name: "Interlingua",
// nativeName: "Interlingua",
// ),
// "id": const ISOLanguageName(
// name: "Indonesian",
// nativeName: "Bahasa Indonesia",
// ),
"id": const ISOLanguageName(
name: "Indonesian",
nativeName: "Bahasa Indonesia",
),
// "ie": const ISOLanguageName(
// name: "Interlingue",
// nativeName: "Occidental",

View File

@ -14,6 +14,7 @@ import 'package:spotube/pages/home/genres/genre_playlists.dart';
import 'package:spotube/pages/home/genres/genres.dart';
import 'package:spotube/pages/home/home.dart';
import 'package:spotube/pages/lastfm_login/lastfm_login.dart';
import 'package:spotube/pages/library/local_folder.dart';
import 'package:spotube/pages/library/playlist_generate/playlist_generate.dart';
import 'package:spotube/pages/library/playlist_generate/playlist_generate_result.dart';
import 'package:spotube/pages/lyrics/mini_lyrics.dart';
@ -105,12 +106,12 @@ final routerProvider = Provider((ref) {
const SpotubePage(child: SearchPage()),
),
GoRoute(
path: "/library",
name: LibraryPage.name,
pageBuilder: (context, state) =>
const SpotubePage(child: LibraryPage()),
routes: [
GoRoute(
path: "/library",
name: LibraryPage.name,
pageBuilder: (context, state) =>
const SpotubePage(child: LibraryPage()),
routes: [
GoRoute(
path: "generate",
name: PlaylistGeneratorPage.name,
pageBuilder: (context, state) =>
@ -124,10 +125,22 @@ final routerProvider = Provider((ref) {
state: state.extra as GeneratePlaylistProviderInput,
),
),
),
]),
],
),
)
],
),
GoRoute(
path: "local",
name: LocalLibraryPage.name,
pageBuilder: (context, state) {
assert(state.extra is String);
return SpotubePage(
child: LocalLibraryPage(state.extra as String,
isDownloads:
state.uri.queryParameters["downloads"] != null),
);
},
),
]),
GoRoute(
path: "/lyrics",
name: LyricsPage.name,

View File

@ -122,4 +122,6 @@ abstract class SpotubeIcons {
static const power = FeatherIcons.power;
static const bluetooth = FeatherIcons.bluetooth;
static const chart = FeatherIcons.barChart2;
static const folderAdd = FeatherIcons.folderPlus;
static const folderRemove = FeatherIcons.folderMinus;
}

View File

@ -56,7 +56,7 @@ class HomeGenresSection extends HookConsumerWidget {
},
icon: const Icon(SpotubeIcons.angleRight),
label: Text(
"Browse All",
context.l10n.browse_all,
style: textTheme.bodyMedium?.copyWith(
color: colorScheme.secondary,
),

View File

@ -0,0 +1,199 @@
import 'dart:math';
import 'package:flutter/material.dart';
import 'package:gap/gap.dart';
import 'package:go_router/go_router.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:path/path.dart';
import 'package:spotube/collections/spotube_icons.dart';
import 'package:spotube/components/shared/image/universal_image.dart';
import 'package:spotube/extensions/constrains.dart';
import 'package:spotube/extensions/context.dart';
import 'package:spotube/extensions/image.dart';
import 'package:spotube/hooks/utils/use_brightness_value.dart';
import 'package:spotube/provider/local_tracks/local_tracks_provider.dart';
import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
class LocalFolderItem extends HookConsumerWidget {
final String folder;
const LocalFolderItem({super.key, required this.folder});
@override
Widget build(BuildContext context, ref) {
final ThemeData(:colorScheme) = Theme.of(context);
final mediaQuery = MediaQuery.of(context);
final lerpValue = useBrightnessValue(.9, .7);
final downloadFolder =
ref.watch(userPreferencesProvider.select((s) => s.downloadLocation));
final isDownloadFolder = folder == downloadFolder;
final Uri(:pathSegments) = Uri.parse(
folder
.replaceFirst(RegExp(r'^/Volumes/[^/]+/Users/'), "")
.replaceFirst(r'C:\Users\', "")
.replaceFirst(r'/home/', ""),
);
// if length > 5, we ... all the middle segments after 2 and the last 2
final segments = pathSegments.length > 5
? [
...pathSegments.take(2),
"...",
...pathSegments.skip(pathSegments.length - 3).toList()
..removeLast(),
]
: pathSegments.take(pathSegments.length - 1).toList();
final trackSnapshot = ref.watch(
localTracksProvider.select(
(s) => s.whenData((tracks) => tracks[folder]?.take(4).toList()),
),
);
final tracks = trackSnapshot.value ?? [];
return InkWell(
onTap: () {
if (isDownloadFolder) {
context.go("/library/local?downloads=1", extra: folder);
} else {
context.go(
"/library/local",
extra: folder,
);
}
},
borderRadius: BorderRadius.circular(8),
child: Ink(
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(8),
color: Color.lerp(
colorScheme.surfaceVariant,
colorScheme.surface,
lerpValue,
),
),
child: Padding(
padding: const EdgeInsets.all(8.0),
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
if (tracks.isEmpty)
Card(
child: Padding(
padding: const EdgeInsets.all(8.0),
child: Icon(
SpotubeIcons.folder,
size: mediaQuery.smAndDown
? 95
: mediaQuery.mdAndDown
? 100
: 142,
),
),
)
else
ClipRRect(
borderRadius: BorderRadius.circular(8),
child: GridView.builder(
shrinkWrap: true,
physics: const NeverScrollableScrollPhysics(),
gridDelegate: SliverGridDelegateWithFixedCrossAxisCount(
crossAxisCount: max((tracks.length / 2).ceil(), 2),
),
itemCount: tracks.length,
itemBuilder: (context, index) {
final track = tracks[index];
return UniversalImage(
path: (track.album?.images).asUrlString(
placeholder: ImagePlaceholder.albumArt,
),
fit: BoxFit.cover,
);
},
),
),
const Gap(8),
Stack(
children: [
Center(
child: Text(
isDownloadFolder
? context.l10n.downloads
: basename(folder),
style: const TextStyle(fontWeight: FontWeight.bold),
textAlign: TextAlign.center,
),
),
if (!isDownloadFolder)
Align(
alignment: Alignment.topRight,
child: PopupMenuButton(
child: const Padding(
padding: EdgeInsets.all(3),
child: Icon(Icons.more_vert),
),
itemBuilder: (context) {
return [
PopupMenuItem(
child: ListTile(
leading: const Icon(SpotubeIcons.folderRemove),
iconColor: colorScheme.error,
title:
Text(context.l10n.remove_library_location),
onTap: () {
final libraryLocations = ref
.read(userPreferencesProvider)
.localLibraryLocation;
ref
.read(userPreferencesProvider.notifier)
.setLocalLibraryLocation(
libraryLocations
.where((e) => e != folder)
.toList(),
);
},
),
)
];
},
),
),
],
),
const Spacer(),
Wrap(
spacing: 2,
runSpacing: 2,
children: [
for (final MapEntry(key: index, value: segment)
in segments.asMap().entries)
Text.rich(
TextSpan(
children: [
if (index != 0)
TextSpan(
text: "/ ",
style: TextStyle(color: colorScheme.primary),
),
TextSpan(text: segment),
],
),
style: TextStyle(
fontSize: 10,
color: colorScheme.tertiary,
),
),
],
),
const Spacer(),
],
),
),
),
);
}
}

View File

@ -1,52 +1,18 @@
import 'dart:io';
import 'package:catcher_2/catcher_2.dart';
import 'package:flutter/foundation.dart';
import 'package:file_picker/file_picker.dart';
import 'package:file_selector/file_selector.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:collection/collection.dart';
import 'package:fuzzywuzzy/fuzzywuzzy.dart';
import 'package:gap/gap.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:metadata_god/metadata_god.dart';
import 'package:mime/mime.dart';
import 'package:path/path.dart';
import 'package:path_provider/path_provider.dart';
import 'package:skeletonizer/skeletonizer.dart';
import 'package:spotify/spotify.dart';
import 'package:spotube/collections/fake.dart';
import 'package:spotube/collections/spotube_icons.dart';
import 'package:spotube/components/shared/expandable_search/expandable_search.dart';
import 'package:spotube/components/shared/fallbacks/not_found.dart';
import 'package:spotube/components/shared/inter_scrollbar/inter_scrollbar.dart';
import 'package:spotube/components/shared/sort_tracks_dropdown.dart';
import 'package:spotube/components/shared/track_tile/track_tile.dart';
import 'package:spotube/extensions/artist_simple.dart';
import 'package:spotube/components/library/local_folder/local_folder_item.dart';
import 'package:spotube/extensions/constrains.dart';
import 'package:spotube/extensions/context.dart';
import 'package:spotube/extensions/track.dart';
import 'package:spotube/models/local_track.dart';
import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.dart';
import 'package:spotube/provider/local_tracks/local_tracks_provider.dart';
import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
import 'package:spotube/utils/service_utils.dart';
import 'package:spotube/utils/platform.dart';
// ignore: depend_on_referenced_packages
import 'package:flutter_rust_bridge/flutter_rust_bridge.dart' show FfiException;
const supportedAudioTypes = [
"audio/webm",
"audio/ogg",
"audio/mpeg",
"audio/mp4",
"audio/opus",
"audio/wav",
"audio/aac",
];
const imgMimeToExt = {
"image/png": ".png",
"image/jpeg": ".jpg",
"image/webp": ".webp",
"image/gif": ".gif",
};
enum SortBy {
none,
@ -59,273 +25,77 @@ enum SortBy {
album,
}
final localTracksProvider = FutureProvider<List<LocalTrack>>((ref) async {
try {
if (kIsWeb) return [];
final downloadLocation = ref.watch(
userPreferencesProvider.select((s) => s.downloadLocation),
);
if (downloadLocation.isEmpty) return [];
final downloadDir = Directory(downloadLocation);
if (!await downloadDir.exists()) {
await downloadDir.create(recursive: true);
return [];
}
final entities = downloadDir.listSync(recursive: true);
final filesWithMetadata = (await Future.wait(
entities.map((e) => File(e.path)).where((file) {
final mimetype = lookupMimeType(file.path);
return mimetype != null && supportedAudioTypes.contains(mimetype);
}).map(
(file) async {
try {
final metadata = await MetadataGod.readMetadata(file: file.path);
final imageFile = File(join(
(await getTemporaryDirectory()).path,
"spotube",
basenameWithoutExtension(file.path) +
imgMimeToExt[metadata.picture?.mimeType ?? "image/jpeg"]!,
));
if (!await imageFile.exists() && metadata.picture != null) {
await imageFile.create(recursive: true);
await imageFile.writeAsBytes(
metadata.picture?.data ?? [],
mode: FileMode.writeOnly,
);
}
return {"metadata": metadata, "file": file, "art": imageFile.path};
} catch (e, stack) {
if (e is FfiException) {
return {"file": file};
}
Catcher2.reportCheckedError(e, stack);
return {};
}
},
),
))
.where((e) => e.isNotEmpty)
.toList();
final tracks = filesWithMetadata
.map(
(fileWithMetadata) => LocalTrack.fromTrack(
track: Track().fromFile(
fileWithMetadata["file"],
metadata: fileWithMetadata["metadata"],
art: fileWithMetadata["art"],
),
path: fileWithMetadata["file"].path,
),
)
.toList();
return tracks;
} catch (e, stack) {
Catcher2.reportCheckedError(e, stack);
return [];
}
});
class UserLocalTracks extends HookConsumerWidget {
const UserLocalTracks({super.key});
Future<void> playLocalTracks(
WidgetRef ref,
List<LocalTrack> tracks, {
LocalTrack? currentTrack,
}) async {
final playlist = ref.read(proxyPlaylistProvider);
final playback = ref.read(proxyPlaylistProvider.notifier);
currentTrack ??= tracks.first;
final isPlaylistPlaying = playlist.containsTracks(tracks);
if (!isPlaylistPlaying) {
await playback.load(
tracks,
initialIndex: tracks.indexWhere((s) => s.id == currentTrack?.id),
autoPlay: true,
);
} else if (isPlaylistPlaying &&
currentTrack.id != null &&
currentTrack.id != playlist.activeTrack?.id) {
await playback.jumpToTrack(currentTrack);
}
}
@override
Widget build(BuildContext context, ref) {
final sortBy = useState<SortBy>(SortBy.none);
final playlist = ref.watch(proxyPlaylistProvider);
final trackSnapshot = ref.watch(localTracksProvider);
final isPlaylistPlaying =
playlist.containsTracks(trackSnapshot.asData?.value ?? []);
final preferencesNotifier = ref.watch(userPreferencesProvider.notifier);
final preferences = ref.watch(userPreferencesProvider);
final searchController = useTextEditingController();
useValueListenable(searchController);
final searchFocus = useFocusNode();
final isFiltering = useState(false);
final addLocalLibraryLocation = useCallback(() async {
if (kIsMobile || kIsMacOS) {
final dirStr = await FilePicker.platform.getDirectoryPath(
initialDirectory: preferences.downloadLocation,
);
if (dirStr == null) return;
if (preferences.localLibraryLocation.contains(dirStr)) return;
preferencesNotifier.setLocalLibraryLocation(
[...preferences.localLibraryLocation, dirStr]);
} else {
String? dirStr = await getDirectoryPath(
initialDirectory: preferences.downloadLocation,
);
if (dirStr == null) return;
if (preferences.localLibraryLocation.contains(dirStr)) return;
preferencesNotifier.setLocalLibraryLocation(
[...preferences.localLibraryLocation, dirStr]);
}
}, [preferences.localLibraryLocation]);
final controller = useScrollController();
// This is just to pre-load the tracks.
// For now, this gets all of them.
ref.watch(localTracksProvider);
return Column(
children: [
Padding(
padding: const EdgeInsets.all(8.0),
child: Row(
children: [
const SizedBox(width: 5),
FilledButton(
onPressed: trackSnapshot.asData?.value != null
? () async {
if (trackSnapshot.asData?.value.isNotEmpty == true) {
if (!isPlaylistPlaying) {
await playLocalTracks(
ref,
trackSnapshot.asData!.value,
);
}
}
}
: null,
child: Row(
children: [
Text(context.l10n.play),
Icon(
isPlaylistPlaying ? SpotubeIcons.stop : SpotubeIcons.play,
)
],
),
),
const Spacer(),
ExpandableSearchButton(
isFiltering: isFiltering.value,
onPressed: (value) => isFiltering.value = value,
searchFocus: searchFocus,
),
const SizedBox(width: 10),
SortTracksDropdown(
value: sortBy.value,
onChanged: (value) {
sortBy.value = value;
},
),
const SizedBox(width: 5),
FilledButton(
child: const Icon(SpotubeIcons.refresh),
onPressed: () {
ref.invalidate(localTracksProvider);
},
)
],
),
),
ExpandableSearchField(
searchController: searchController,
searchFocus: searchFocus,
isFiltering: isFiltering.value,
onChangeFiltering: (value) => isFiltering.value = value,
),
trackSnapshot.when(
data: (tracks) {
final sortedTracks = useMemoized(() {
return ServiceUtils.sortTracks(tracks, sortBy.value);
}, [sortBy.value, tracks]);
final filteredTracks = useMemoized(() {
if (searchController.text.isEmpty) {
return sortedTracks;
}
return sortedTracks
.map((e) => (
weightedRatio(
"${e.name} - ${e.artists?.asString() ?? ""}",
searchController.text,
),
e,
))
.toList()
.sorted(
(a, b) => b.$1.compareTo(a.$1),
)
.where((e) => e.$1 > 50)
.map((e) => e.$2)
.toList()
.toList();
}, [searchController.text, sortedTracks]);
if (!trackSnapshot.isLoading && filteredTracks.isEmpty) {
return const Expanded(
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [NotFound()],
),
);
}
return Expanded(
child: RefreshIndicator(
onRefresh: () async {
ref.invalidate(localTracksProvider);
},
child: InterScrollbar(
controller: controller,
child: Skeletonizer(
enabled: trackSnapshot.isLoading,
child: ListView.builder(
controller: controller,
physics: const AlwaysScrollableScrollPhysics(),
itemCount:
trackSnapshot.isLoading ? 5 : filteredTracks.length,
itemBuilder: (context, index) {
if (trackSnapshot.isLoading) {
return TrackTile(
playlist: playlist,
track: FakeData.track,
index: index,
);
}
final track = filteredTracks[index];
return TrackTile(
index: index,
playlist: playlist,
track: track,
userPlaylist: false,
onTap: () async {
await playLocalTracks(
ref,
sortedTracks,
currentTrack: track,
);
},
);
},
),
),
),
),
);
},
loading: () => Expanded(
child: Skeletonizer(
enabled: true,
child: ListView.builder(
itemCount: 5,
itemBuilder: (context, index) => TrackTile(
track: FakeData.track,
index: index,
playlist: playlist,
),
return LayoutBuilder(builder: (context, constrains) {
return Padding(
padding: const EdgeInsets.symmetric(horizontal: 12.0),
child: Column(
children: [
Align(
alignment: Alignment.centerRight,
child: TextButton.icon(
icon: const Icon(SpotubeIcons.folderAdd),
label: Text(context.l10n.add_library_location),
onPressed: addLocalLibraryLocation,
),
),
),
error: (error, stackTrace) =>
Text(error.toString() + stackTrace.toString()),
)
],
);
const Gap(8),
Expanded(
child: GridView.builder(
gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent(
maxCrossAxisExtent: 200,
mainAxisExtent: constrains.isXs
? 210
: constrains.mdAndDown
? 280
: 250,
crossAxisSpacing: 10,
mainAxisSpacing: 10,
),
itemCount: preferences.localLibraryLocation.length + 1,
itemBuilder: (context, index) {
return LocalFolderItem(
folder: index == 0
? preferences.downloadLocation
: preferences.localLibraryLocation[index - 1],
);
},
),
),
],
),
);
});
}
}

View File

@ -1,6 +1,5 @@
import 'package:flutter/gestures.dart';
import 'package:flutter/material.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:spotube/collections/spotube_icons.dart';
@ -31,11 +30,17 @@ class VolumeSlider extends HookConsumerWidget {
}
}
},
child: Slider(
min: 0,
max: 1,
value: value,
onChanged: onChanged,
child: SliderTheme(
data: const SliderThemeData(
showValueIndicator: ShowValueIndicator.always,
),
child: Slider(
min: 0,
max: 1,
label: (value * 100).toStringAsFixed(0),
value: value,
onChanged: onChanged,
),
),
);
return Row(

View File

@ -8,7 +8,6 @@ import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:spotify/spotify.dart';
import 'package:spotube/collections/assets.gen.dart';
import 'package:spotube/collections/spotube_icons.dart';
import 'package:spotube/components/library/user_local_tracks.dart';
import 'package:spotube/components/shared/adaptive/adaptive_pop_sheet_list.dart';
import 'package:spotube/components/shared/dialogs/playlist_add_track_dialog.dart';
import 'package:spotube/components/shared/dialogs/prompt_dialog.dart';
@ -23,6 +22,7 @@ import 'package:spotube/models/local_track.dart';
import 'package:spotube/provider/authentication_provider.dart';
import 'package:spotube/provider/blacklist_provider.dart';
import 'package:spotube/provider/download_manager_provider.dart';
import 'package:spotube/provider/local_tracks/local_tracks_provider.dart';
import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.dart';
import 'package:spotube/provider/spotify/spotify.dart';
import 'package:spotube/provider/spotify_provider.dart';
@ -197,6 +197,8 @@ class TrackOptions extends HookConsumerWidget {
return downloadManager.getProgressNotifier(spotubeTrack);
});
final isLocalTrack = track is LocalTrack;
final adaptivePopSheetList = AdaptivePopSheetList<TrackOptionValue>(
onSelected: (value) async {
switch (value) {
@ -314,118 +316,120 @@ class TrackOptions extends HookConsumerWidget {
),
),
],
children: switch (track.runtimeType) {
LocalTrack() => [
PopSheetEntry(
value: TrackOptionValue.delete,
leading: const Icon(SpotubeIcons.trash),
title: Text(context.l10n.delete),
)
],
_ => [
if (mediaQuery.smAndDown)
PopSheetEntry(
value: TrackOptionValue.album,
leading: const Icon(SpotubeIcons.album),
title: Text(context.l10n.go_to_album),
subtitle: Text(track.album!.name!),
),
if (!playlist.containsTrack(track)) ...[
PopSheetEntry(
value: TrackOptionValue.addToQueue,
leading: const Icon(SpotubeIcons.queueAdd),
title: Text(context.l10n.add_to_queue),
),
PopSheetEntry(
value: TrackOptionValue.playNext,
leading: const Icon(SpotubeIcons.lightning),
title: Text(context.l10n.play_next),
),
] else
PopSheetEntry(
value: TrackOptionValue.removeFromQueue,
enabled: playlist.activeTrack?.id != track.id,
leading: const Icon(SpotubeIcons.queueRemove),
title: Text(context.l10n.remove_from_queue),
),
if (me.asData?.value != null)
PopSheetEntry(
value: TrackOptionValue.favorite,
leading: favorites.isLiked
? const Icon(
SpotubeIcons.heartFilled,
color: Colors.pink,
)
: const Icon(SpotubeIcons.heart),
title: Text(
favorites.isLiked
? context.l10n.remove_from_favorites
: context.l10n.save_as_favorite,
),
),
if (auth != null) ...[
PopSheetEntry(
value: TrackOptionValue.startRadio,
leading: const Icon(SpotubeIcons.radio),
title: Text(context.l10n.start_a_radio),
),
PopSheetEntry(
value: TrackOptionValue.addToPlaylist,
leading: const Icon(SpotubeIcons.playlistAdd),
title: Text(context.l10n.add_to_playlist),
),
],
if (userPlaylist && auth != null)
PopSheetEntry(
value: TrackOptionValue.removeFromPlaylist,
leading: const Icon(SpotubeIcons.removeFilled),
title: Text(context.l10n.remove_from_playlist),
),
PopSheetEntry(
value: TrackOptionValue.download,
enabled: !isInQueue,
leading: isInQueue
? HookBuilder(builder: (context) {
final progress = useListenable(progressNotifier!);
return CircularProgressIndicator(
value: progress.value,
);
})
: const Icon(SpotubeIcons.download),
title: Text(context.l10n.download_track),
children: [
if (isLocalTrack)
PopSheetEntry(
value: TrackOptionValue.delete,
leading: const Icon(SpotubeIcons.trash),
title: Text(context.l10n.delete),
),
if (mediaQuery.smAndDown)
PopSheetEntry(
value: TrackOptionValue.album,
leading: const Icon(SpotubeIcons.album),
title: Text(context.l10n.go_to_album),
subtitle: Text(track.album!.name!),
),
if (!playlist.containsTrack(track)) ...[
PopSheetEntry(
value: TrackOptionValue.addToQueue,
leading: const Icon(SpotubeIcons.queueAdd),
title: Text(context.l10n.add_to_queue),
),
PopSheetEntry(
value: TrackOptionValue.playNext,
leading: const Icon(SpotubeIcons.lightning),
title: Text(context.l10n.play_next),
),
] else
PopSheetEntry(
value: TrackOptionValue.removeFromQueue,
enabled: playlist.activeTrack?.id != track.id,
leading: const Icon(SpotubeIcons.queueRemove),
title: Text(context.l10n.remove_from_queue),
),
if (me.asData?.value != null && !isLocalTrack)
PopSheetEntry(
value: TrackOptionValue.favorite,
leading: favorites.isLiked
? const Icon(
SpotubeIcons.heartFilled,
color: Colors.pink,
)
: const Icon(SpotubeIcons.heart),
title: Text(
favorites.isLiked
? context.l10n.remove_from_favorites
: context.l10n.save_as_favorite,
),
PopSheetEntry(
value: TrackOptionValue.blacklist,
leading: const Icon(SpotubeIcons.playlistRemove),
iconColor: !isBlackListed ? Colors.red[400] : null,
textColor: !isBlackListed ? Colors.red[400] : null,
title: Text(
isBlackListed
? context.l10n.remove_from_blacklist
: context.l10n.add_to_blacklist,
),
),
if (auth != null && !isLocalTrack) ...[
PopSheetEntry(
value: TrackOptionValue.startRadio,
leading: const Icon(SpotubeIcons.radio),
title: Text(context.l10n.start_a_radio),
),
PopSheetEntry(
value: TrackOptionValue.addToPlaylist,
leading: const Icon(SpotubeIcons.playlistAdd),
title: Text(context.l10n.add_to_playlist),
),
],
if (userPlaylist && auth != null && !isLocalTrack)
PopSheetEntry(
value: TrackOptionValue.removeFromPlaylist,
leading: const Icon(SpotubeIcons.removeFilled),
title: Text(context.l10n.remove_from_playlist),
),
if (!isLocalTrack)
PopSheetEntry(
value: TrackOptionValue.download,
enabled: !isInQueue,
leading: isInQueue
? HookBuilder(builder: (context) {
final progress = useListenable(progressNotifier!);
return CircularProgressIndicator(
value: progress.value,
);
})
: const Icon(SpotubeIcons.download),
title: Text(context.l10n.download_track),
),
if (!isLocalTrack)
PopSheetEntry(
value: TrackOptionValue.blacklist,
leading: const Icon(SpotubeIcons.playlistRemove),
iconColor: !isBlackListed ? Colors.red[400] : null,
textColor: !isBlackListed ? Colors.red[400] : null,
title: Text(
isBlackListed
? context.l10n.remove_from_blacklist
: context.l10n.add_to_blacklist,
),
PopSheetEntry(
value: TrackOptionValue.share,
leading: const Icon(SpotubeIcons.share),
title: Text(context.l10n.share),
),
if (!isLocalTrack)
PopSheetEntry(
value: TrackOptionValue.share,
leading: const Icon(SpotubeIcons.share),
title: Text(context.l10n.share),
),
if (!isLocalTrack)
PopSheetEntry(
value: TrackOptionValue.songlink,
leading: Assets.logos.songlinkTransparent.image(
width: 22,
height: 22,
color: colorScheme.onSurface.withOpacity(0.5),
),
PopSheetEntry(
value: TrackOptionValue.songlink,
leading: Assets.logos.songlinkTransparent.image(
width: 22,
height: 22,
color: colorScheme.onSurface.withOpacity(0.5),
),
title: Text(context.l10n.song_link),
),
PopSheetEntry(
value: TrackOptionValue.details,
leading: const Icon(SpotubeIcons.info),
title: Text(context.l10n.details),
),
]
},
title: Text(context.l10n.song_link),
),
if (!isLocalTrack)
PopSheetEntry(
value: TrackOptionValue.details,
leading: const Icon(SpotubeIcons.info),
title: Text(context.l10n.details),
),
],
);
//! This is the most ANTI pattern I've ever done, but it works

View File

@ -195,19 +195,26 @@ class TrackTile extends HookConsumerWidget {
children: [
Expanded(
flex: 6,
child: LinkText(
track.name!,
"/track/${track.id}",
push: true,
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
child: switch (track) {
LocalTrack() => Text(
track.name!,
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
_ => LinkText(
track.name!,
"/track/${track.id}",
push: true,
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
},
),
if (constrains.mdAndUp) ...[
const SizedBox(width: 8),
Expanded(
flex: 4,
child: switch (track.runtimeType) {
child: switch (track) {
LocalTrack() => Text(
track.album!.name!,
maxLines: 1,

View File

@ -3,8 +3,8 @@ import 'package:device_info_plus/device_info_plus.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:permission_handler/permission_handler.dart';
import 'package:spotube/components/library/user_local_tracks.dart';
import 'package:spotube/hooks/utils/use_async_effect.dart';
import 'package:spotube/provider/local_tracks/local_tracks_provider.dart';
import 'package:spotube/utils/platform.dart';
void useGetStoragePermissions(WidgetRef ref) {

View File

@ -107,6 +107,9 @@
"always_on_top": "Always on top",
"exit_mini_player": "Exit Mini player",
"download_location": "Download location",
"local_library": "Local library",
"add_library_location": "Add to library",
"remove_library_location": "Remove from library",
"account": "Account",
"login_with_spotify": "Login with your Spotify account",
"connect_with_spotify": "Connect with Spotify",
@ -295,6 +298,7 @@
"delete_playlist": "Delete Playlist",
"delete_playlist_confirmation": "Are you sure you want to delete this playlist?",
"local_tracks": "Local Tracks",
"local_tab": "Local",
"song_link": "Song Link",
"skip_this_nonsense": "Skip this nonsense",
"freedom_of_music": "“Freedom of Music”",

324
lib/l10n/app_eu.arb Normal file
View File

@ -0,0 +1,324 @@
{
"guest": "Gonbidatua",
"browse": "Arakatu",
"search": "Bilatu",
"library": "Liburutegia",
"lyrics": "Hitzak",
"settings": "Ezarpenak",
"genre_categories_filter": "Kategoria edo generoak filtratu...",
"genre": "Generoa",
"personalized": "Pertsonalizatua",
"featured": "Nabarmenduak",
"new_releases": "Argitaratze berriak",
"songs": "Abestiak",
"playing_track": "{track} erreproduzitzen",
"queue_clear_alert": "Uneko zerrenda ezabatuko da. {track_length} abesti ezabatuko dira.\nJarraitu nahi duzu?",
"load_more": "Gehiago kargatu",
"playlists": "Zerrendak",
"artists": "Artistak",
"albums": "Albumak",
"tracks": "Kantak",
"downloads": "Deskargak",
"filter_playlists": "Zure zerrendak filtratu...",
"liked_tracks": "Gustuko Kantak",
"liked_tracks_description": "Zure gustuko kanta guztiak",
"create_playlist": "Sortu zerrenda",
"create_a_playlist": "Sortu zerrenda bat",
"update_playlist": "Eguneratu zerrenda",
"create": "Sortu",
"cancel": "Ezeztatu",
"update": "Eguneratu",
"playlist_name": "Zerrenda Izena",
"name_of_playlist": "Zerrendaren izena",
"description": "Deskribapena",
"public": "Publikoa",
"collaborative": "Kolaboratiboa",
"search_local_tracks": "Bilatu kanta lokalak...",
"play": "Erreproduzitu",
"delete": "Ezabatu",
"none": "Batere ez",
"sort_a_z": "Ordenatu A-Z",
"sort_z_a": "Ordenatu Z-A",
"sort_artist": "Ordenatu Artistaren arabera",
"sort_album": "Ordenatu Albumaren arabera",
"sort_duration": "Ordenar Iraupenaren arabera",
"sort_tracks": "Ordenatu Kantak",
"currently_downloading": "Oraintxe ({tracks_length}) deskargatzen",
"cancel_all": "Ezeztatu dena",
"filter_artist": "Filtratu artistak...",
"followers": "{followers} Jarraitzaile",
"add_artist_to_blacklist": "Gehitu artista zerrenda beltzera",
"top_tracks": "Top Kantak",
"fans_also_like": "Fan-ek hau ere gustuko dute",
"loading": "Kargatzen...",
"artist": "Artista",
"blacklisted": "Zerrenda beltzean",
"following": "Jarraitzen",
"follow": "Jarraitu",
"artist_url_copied": "Artistaren URL-a arbelera kopiatua",
"added_to_queue": "{tracks} kanta zerrendara gehituak",
"filter_albums": "Albumak filtratu...",
"synced": "Sinkronizatuta",
"plain": "Arrunta",
"shuffle": "Ausaz",
"search_tracks": "Bilatu kantak...",
"released": "Argitaratua",
"error": "Errorea: {error}",
"title": "Izenburua",
"time": "Iraupena",
"more_actions": "Ekintza gehiago",
"download_count": "({count}) deskarga",
"add_count_to_playlist": "Gehitu ({count}) zerrendara",
"add_count_to_queue": "Gehitu ({count}) ilarara",
"play_count_next": "Erreproduzitu hurrengo ({count})-ak",
"album": "Albuma",
"copied_to_clipboard": "{data} arbelean kopiatua",
"add_to_following_playlists": "Gehitu {track} hurrengo erreprodukzio-zerrendetara",
"add": "Gehitu",
"added_track_to_queue": "{track} zerrendan gehitua",
"add_to_queue": "Gehitu zerrendan",
"track_will_play_next": "{track} erreproduzituko da ondoren",
"play_next": "Hurrengo erreprodukzioa",
"removed_track_from_queue": "{track} zerrendatik ezabatua",
"remove_from_queue": "Ezabatu ilaratik",
"remove_from_favorites": "Ezabatu gogokoetatik",
"save_as_favorite": "Gorde gogokoetan",
"add_to_playlist": "Gehitu zerrendara",
"remove_from_playlist": "Ezabatu zerrendatik",
"add_to_blacklist": "Gehitu zerrenda beltzera",
"remove_from_blacklist": "Ezabatu zerrenda beltzetik",
"share": "Elkarbanatu",
"mini_player": "Mini Erreproduzitzailea",
"slide_to_seek": "Arrastatu aurrerantz edo atzearantz bilatzeko",
"shuffle_playlist": "Erreproduzitu zerrenda ausazko ordenean",
"unshuffle_playlist": "Desgaitu ausazko erreprodukzioa",
"previous_track": "Aurreko pista",
"next_track": "Hurrengo pista",
"pause_playback": "Pausatu erreprodukzioa",
"resume_playback": "Berrabiarazi erreprodukzioa",
"loop_track": "Kanta begiztan",
"repeat_playlist": "Errepikatu lista",
"queue": "Ilara",
"alternative_track_sources": "Kanten iturri alternatiboak",
"download_track": "Deskargatu kanta",
"tracks_in_queue": "{tracks} kanta zerrendan",
"clear_all": "Garbitu dena",
"show_hide_ui_on_hover": "Erakutsi/Ezkutatu interfazea kurtsorea pasatzean",
"always_on_top": "Beti ikusgai",
"exit_mini_player": "Irten mini erreproduzitzailetik",
"download_location": "Deskargen kokapena",
"account": "Kontua",
"login_with_spotify": "Hasi saioa zure Spotify kontuarekin",
"connect_with_spotify": "Spotify-rekin konektatu",
"logout": "Itxi saioa",
"logout_of_this_account": "Itxi kontu honen saioa",
"language_region": "Hizkuntza eta Herrialdea",
"language": "Hizkuntza",
"system_default": "Sisteman lehenetsia",
"market_place_region": "Dendaren herrialdea",
"recommendation_country": "Gomendio herrialdea",
"appearance": "Itxura",
"layout_mode": "Diseinu modua",
"override_layout_settings": "Responsive diseinu moduaren ezarpenak ezeztatu",
"adaptive": "Moldagarria",
"compact": "Trinkoa",
"extended": "Hedatua",
"theme": "Gaia",
"dark": "Iluna",
"light": "Argia",
"system": "Sistema",
"accent_color": "Azentu kolorea",
"sync_album_color": "Sinkronizatu albumaren kolorea",
"sync_album_color_description": "Albumaren artearen kolore nagusia erabili azentu kolore bezala",
"playback": "Erreprodukzioa",
"audio_quality": "Audioaren kalitatea",
"high": "Altua",
"low": "Baxua",
"pre_download_play": "Aurre-deskargatu eta erreproduzitu",
"pre_download_play_description": "Streaming egin beharrean, byte-ak deskargatu eta erreproduzitu (banda-zabalera handia duten erabiltzaileentzat gomendagarria)",
"skip_non_music": "Musika ez diren segmentuak baztertu (SponsorBlock)",
"blacklist_description": "Zerrenda beltzeko abesti eta artistak",
"wait_for_download_to_finish": "Mesedez, itxaron uneko deskarga bukatu arte",
"desktop": "Mahaigaina",
"close_behavior": "Ixterako Portaera",
"close": "Itxi",
"minimize_to_tray": "Sistemako erretilura minimizatu",
"show_tray_icon": "Erakutsi ikonoa sistemaren erretiluan",
"about": "Honi buruz",
"u_love_spotube": "Badakigu Spotube maite duzula",
"check_for_updates": "Bilatu eguneraketak",
"about_spotube": "Spotube-ri buruz",
"blacklist": "Zerrenda beltza",
"please_sponsor": "Mesedez, babestu/diruz lagundu",
"spotube_description": "Spotube, arina, plataforma-anitza eta doakoa den Spotify-ren bezeroa",
"version": "Bertsioa",
"build_number": "Konpilazio zenbakia",
"founder": "Sortzailea",
"repository": "Errepositorioa",
"bug_issues": "Erroreak eta arazoak",
"made_with": "Bangladesh🇧🇩-en ❤️-z egina",
"kingkor_roy_tirtho": "Kingkor Roy Tirtho",
"copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
"license": "Lizentzia",
"add_spotify_credentials": "Gehitu zure Spotify kredentzialak hasi ahal izateko",
"credentials_will_not_be_shared_disclaimer": "Ez arduratu, zure kredentzialak ez ditugu bilduko edo inorekin elkarbanatuko",
"know_how_to_login": "Ez dakizu nola egin?",
"follow_step_by_step_guide": "Jarraitu pausoz-pausoko gida",
"spotify_cookie": "Spotify-ren {name} cookiea",
"cookie_name_cookie": "{name} cookiea",
"fill_in_all_fields": "Mesedez, osatu eremu guztiak",
"submit": "Bidali",
"exit": "Irten",
"previous": "Aurrekoa",
"next": "Hurrengoa",
"done": "Eginda",
"step_1": "1. pausua",
"first_go_to": "Hasteko, joan hona",
"login_if_not_logged_in": "eta hasi saioa/sortu kontua lehendik ez baduzu eginda",
"step_2": "2. pausua",
"step_2_steps": "1. Saioa hasita duzularik, sakatu F12 edo saguaren eskuineko botoia klikatu > Ikuskatu nabigatzaileko garapen tresnak irekitzeko.\n2. Joan \"Aplikazio\" (Chrome, Edge, Brave, etab.) edo \"Biltegiratzea\" (Firefox, Palemoon, etab.)\n3. Joan \"Cookieak\" atalera eta gero \"https://accounts.spotify.com\" azpiatalera",
"step_3": "3. pausua",
"step_3_steps": "Kopiatu \"sp_dc\" cookiearen balioa",
"success_emoji": "Eginda! 🥳",
"success_message": "Ongi hasi duzu zure Spotify kontua. Lan bikaina, lagun!",
"step_4": "4. pausua",
"step_4_steps": "Itsatsi \"sp_dc\"-tik kopiatutako balioa",
"something_went_wrong": "Zerbaitek huts egin du",
"piped_instance": "Piped zerbitzariaren instantzia",
"piped_description": "Kanten koizidentzietan erabiltzeko Piped zerbitzariaren instantzia",
"piped_warning": "Batzuk agian ez dute ongi funtzionatuko, zure ardurapean erabili",
"generate_playlist": "Sortu Zerrenda",
"track_exists": "{track} kanta dagoeneko badago",
"replace_downloaded_tracks": "Ordezkatu deskargatutako kanta guztiak",
"skip_download_tracks": "Deskargatutako kanta guztien deskarga baztertu",
"do_you_want_to_replace": "Dagoen kanta ordezkatu nahi duzu??",
"replace": "Ordezkatu",
"skip": "Baztertu",
"select_up_to_count_type": "Aukertu {count} {type}",
"select_genres": "Aukeratu Generoak",
"add_genres": "Gehitu Generoak",
"country": "Herrialdea",
"number_of_tracks_generate": "Sortzeko kanta kopurua",
"acousticness": "Akustikotasuna",
"danceability": "Dantzagarritasuna",
"energy": "Energia",
"instrumentalness": "Instrumentaltasuna",
"liveness": "Zuzenean",
"loudness": "Ozentasuna",
"speechiness": "Hitzaldia",
"valence": "Balentzia",
"popularity": "Populartasuna",
"key": "Tonua",
"duration": "Iraupena (s)",
"tempo": "Tenpoa (BPM)",
"mode": "Modua",
"time_signature": "Konpasa",
"short": "Motza",
"medium": "Ertaina",
"long": "Luzea",
"min": "Min.",
"max": "Max.",
"target": "Helburua",
"moderate": "Moderatua",
"deselect_all": "Desaukeratu dena",
"select_all": "Aukeratu dena",
"are_you_sure": "Ziur zaude?",
"generating_playlist": "Zure pertsonalizatutako zerrenda sortzen...",
"selected_count_tracks": "{count} kanta aukeratuta",
"download_warning": "Abesti guztiak aldi berean deskargatuz gero, argi dago musika pirateatzen ari zarela eta musikaren gizarte sortzaileari kalte egiten diozula. Honen jakitun izan eta artisten lan gogorra errespetatu eta babestea espero dut",
"download_ip_ban_warning": "Bidenabar, baliteke zure IPa YouTuben blokeatzea deskarga eskera gehiegi egiten badituzu. IPa blokeatzeak esan nahi du ezin izango duzula YouTube erabili (nahiz eta saioa hasia izan) gutxienez 2-3 hilabetez IP helbide horretatik. Eta Spotube ez da erantzule izango hori gertatzen bazaizu",
"by_clicking_accept_terms": "'Onartu' klikatzean, ondorengo baldintzak onartzen dituzu:",
"download_agreement_1": "Badakit musika pirateatzen ari naizela. Gaiztoa naiz",
"download_agreement_2": "Ahal dudanean lagunduko diot artistari baina oraingoz ez dut bere artea erosteko dirurik",
"download_agreement_3": "Erabat jakitun naiz YouTubek nire IPa blokea dezakeela eta ez diot Spotube-ri edo bere jabe/laguntzaileei erantzukizunik eskatuko nire oraingo jokaerak ekar ditzakeen arazoengatik",
"decline": "Baztertu",
"accept": "Onartu",
"details": "Xehetasunak",
"youtube": "YouTube",
"channel": "Kanala",
"likes": "Gustukoak",
"dislikes": "Ez gustukoak",
"views": "Ikuspenak",
"streamUrl": "Streaming-aren URLa",
"stop": "Gelditu",
"sort_newest": "Ordenatu gehitu berrienetik",
"sort_oldest": "Ordenatu gehitu zaharrenetik",
"sleep_timer": "Itzaltzeko tenporizadorea",
"mins": "{minutes} minutu",
"hours": "{hours} ordu",
"hour": "{hours} ordu",
"custom_hours": "Ordu pertsonalizatuak",
"logs": "Log-ak",
"developers": "Garatzaileak",
"not_logged_in": "Ez duzu saioa hasi",
"search_mode": "Bilaketa modua",
"audio_source": "Audio Iturria",
"ok": "OK",
"failed_to_encrypt": "Errorea zifratzean",
"encryption_failed_warning": "Spotube-ek zifratzea darabil datuak modu seguruan biltegiratzeko. Baina huts egin du. Hori dela eta, biltegiratzea ez da segurua izango\nLinux erabiltzen ari bazara, ziurtatu edozein sekretu-zerbitzu (gnome-keyring, kde-wallet, keepassxc etab.) instalatuta duzula",
"querying_info": "Informazioa egiaztatzen...",
"piped_api_down": "Piped-en APIa ez dago eskuragarri",
"piped_down_error_instructions": "Piped-en {pipedInstance} instantzia ez dago martxan une honetan\n\nAldatu instantzia edo aldatu 'API mota' YouTuberen API ofizialera\n\nZiurtatu aplikazioa berrabiarazten duzula aldaketa eta gero",
"you_are_offline": "Une honetan konexiorik gabe zaude",
"connection_restored": "Internet konexioa berrezarri egin da",
"use_system_title_bar": "Erabili sistemako izenburu barra",
"crunching_results": "Emaitzak prozesatzen...",
"search_to_get_results": "Bilatu emaitzak lortzeko",
"use_amoled_mode": "Erabili AMOLED modua",
"pitch_dark_theme": "Dart-en gai iluna",
"normalize_audio": "Normalizatu audioa",
"change_cover": "Aldatu azala",
"add_cover": "Gehitu azala",
"restore_defaults": "Berrezarri berezko balioak",
"download_music_codec": "Deskargatutako musikaren codec-a",
"streaming_music_codec": "Streaming musikaren codec-a",
"login_with_lastfm": "Hasi saioa Last.fm-n",
"connect": "Konektatu",
"disconnect_lastfm": "Deskonektatu Last.fm-tik",
"disconnect": "Deskonektatu",
"username": "Erabiltzaile izena",
"password": "Pasahitza",
"login": "Hasi saioa",
"login_with_your_lastfm": "Hasi saioa Last.fm-ko zure kontuarekin",
"scrobble_to_lastfm": "Scrobble Last.fm-ra",
"go_to_album": "Albumera joan",
"discord_rich_presence": "Discord-en presentzia aberatsa",
"browse_all": "Esploratu dena",
"genres": "Generoak",
"explore_genres": "Esploratu generoak",
"friends": "Lagunak",
"no_lyrics_available": "Sentitzen dut, ezin dira kanta honen hitzak aurkitu",
"start_a_radio": "Hasi Irrati bat",
"how_to_start_radio": "Nola hasi nahi duzu irratia?",
"replace_queue_question": "Uneko zerrenda ordezkatu nahi duzu edo bertan gehitu?",
"endless_playback": "Amaigabeko erreprodukzioa",
"delete_playlist": "Ezabatu zerrenda",
"delete_playlist_confirmation": "Ziur zaude zerrenda ezabatu nahi duzula?",
"local_tracks": "Kanta lokalak",
"song_link": "Kantaren lotura",
"skip_this_nonsense": "Utzi txorakeria hau",
"freedom_of_music": "“Musika Askatasuna”",
"freedom_of_music_palm": "“Musika Askatasuna zure eskuetan”",
"get_started": "Has gaitezen",
"youtube_source_description": "Gomendatua eta hobekien dabilena.",
"piped_source_description": "Aske zara? YouTube bezala, baino askeago.",
"jiosaavn_source_description": "Asia hegoaldeko herrialdeetarako hoberena.",
"highest_quality": "Kalitate Onena: {quality}",
"select_audio_source": "Aukeratu Audio Iturria",
"endless_playback_description": "Gehitu automatikoki kanta berriak\n ilararen bukaeran",
"choose_your_region": "Aukeratu zure herrialdea",
"choose_your_region_description": "Honekin Spotube-k zure kokalerakuari dagokion edukia\neskeiniko dizu.",
"choose_your_language": "Aukeratu zure hizkuntza",
"help_project_grow": "Lagundu proiektu honi hazten",
"help_project_grow_description": "Spotube kode irekiko proiektu bat da. Proiektu hau hazten lagundu dezakezu, erroreak jakinaraziz edo ezaugarri berriak proposatuz.",
"contribute_on_github": "GitHub-en lagundu",
"donate_on_open_collective": "Open Collective-en diruz lagundu",
"browse_anonymously": "Nabigatu Anonimoki",
"enable_connect": "Gaitu konexioa",
"enable_connect_description": "Kontrolatu Spotube beste gailu batzuetatik",
"devices": "Gailuak",
"select": "Aukeratu",
"connect_client_alert": "{client} gailuak kontrolatzen zaitu",
"this_device": "Gailu hau",
"remote": "Urrunekoa"
}

324
lib/l10n/app_fi.arb Normal file
View File

@ -0,0 +1,324 @@
{
"guest": "Vieras",
"browse": "Selaa",
"search": "Hae",
"library": "Kirjasto",
"lyrics": "Lyriikat",
"settings": "Asetukset",
"genre_categories_filter": "Suodata kategorioita tai genrejä",
"genre": "Genre",
"personalized": "Personoidut",
"featured": "Esittelyssä",
"new_releases": "Uusi julkaisu",
"songs": "Laulut",
"playing_track": "Soitetaan {track}",
"queue_clear_alert": "Tämä tulee tyhjentämään jonon. {track_length} Kappaleita poistetaan\nHaluatko jatkaa?",
"load_more": "Lataa lisää",
"playlists": "Soittolistat",
"artists": "Artistit",
"albums": "Albumit",
"tracks": "Kappaleet",
"downloads": "Lataukset",
"filter_playlists": "Suodata soittolistasi...",
"liked_tracks": "Tykätyt kappaleet",
"liked_tracks_description": "Kaikki tykättysi kappaleet",
"create_playlist": "Luo soittolista",
"create_a_playlist": "Luo soittolista",
"update_playlist": "Päivitä soittolista",
"create": "Luo",
"cancel": "Peruuta",
"update": "Päivitä",
"playlist_name": "Soittolistan nimi",
"name_of_playlist": "Soittolistan nimi",
"description": "Kuvaus",
"public": "Julkinen",
"collaborative": "Collaborative",
"search_local_tracks": "Hae paikallisia lauluja...",
"play": "Soita",
"delete": "Poista",
"none": "Ei mitään",
"sort_a_z": "Suodata A-Z",
"sort_z_a": "Suodata Z-A",
"sort_artist": "Suodata Artistilta",
"sort_album": "Suodata Albumilta",
"sort_duration": "Suodata Pituudelta",
"sort_tracks": "Suodata Kappaleet",
"currently_downloading": "Ladataan ({tracks_length})",
"cancel_all": "Peru kaikki",
"filter_artist": "Suodata artistit...",
"followers": "{followers} Seuraajaa",
"add_artist_to_blacklist": "Lisää artisti mustalle listalle",
"top_tracks": "Suosituimmat kappaleet",
"fans_also_like": "Fanit myös tykkäsivät",
"loading": "Ladataan...",
"artist": "Artisti",
"blacklisted": "Mustalistattu",
"following": "Seurataan",
"follow": "Seuraa",
"artist_url_copied": "Aristin URL kopioitiin leikepöytään",
"added_to_queue": "Lisättiin {tracks} kappaletta jonoon",
"filter_albums": "Suodata albumit...",
"synced": "Synkronoitu",
"plain": "Tavallinen",
"shuffle": "Sekoita",
"search_tracks": "Hae kappaleita...",
"released": "Julkaistu",
"error": "Virhe {error}",
"title": "Otsikko",
"time": "Aika",
"more_actions": "Lisää toimintoja",
"download_count": "Lataa ({count})",
"add_count_to_playlist": "Lisää ({count}) Soittolistaasi",
"add_count_to_queue": "Lisää ({count}) Jonoon",
"play_count_next": "Soita ({count}) seuraavaksi",
"album": "Albumi",
"copied_to_clipboard": "Kopioitiin {data} leikepöytään",
"add_to_following_playlists": "Lisää {track} seuraaviin soittolistoihin",
"add": "Lisää",
"added_track_to_queue": "Lisättiin {track} jonoon",
"add_to_queue": "Lisää jonoon",
"track_will_play_next": "{track} Soitetaan seuraavaksi",
"play_next": "Soita seuraavaksi",
"removed_track_from_queue": "Poistettiin {track} jonosta",
"remove_from_queue": "Poista jonosta",
"remove_from_favorites": "Poista suosikeista",
"save_as_favorite": "Tallenna soittolistana",
"add_to_playlist": "Lisää soittolistaan",
"remove_from_playlist": "Poista soittolistasta",
"add_to_blacklist": "Lisää mustalle listalle",
"remove_from_blacklist": "Poista mustalistalta",
"share": "Jaa",
"mini_player": "Minisoitin",
"slide_to_seek": "Liu'uta mennäkseen eteenpäin tai taaksepäin",
"shuffle_playlist": "Sekoita soittolista",
"unshuffle_playlist": "Poista sekoitus soittolistasta",
"previous_track": "Äskeinen kappale",
"next_track": "Seuraava kappale",
"pause_playback": "Pysäytä soittolistan toisto",
"resume_playback": "Jatka soittolistan toistoa",
"loop_track": "Uudelleentoista kappale",
"repeat_playlist": "Toista soittolista uudelleen",
"queue": "Jono",
"alternative_track_sources": "Toinen kappale lähde",
"download_track": "Lataa kappale",
"tracks_in_queue": "{tracks} kappaletta jonossa",
"clear_all": "Tyhjennä kaikki",
"show_hide_ui_on_hover": "Näytä/Piilota UI leijumalla",
"always_on_top": "Aina päällimmäisenä",
"exit_mini_player": "Lähde minisoittimesta",
"download_location": "Lataus sijainti",
"account": "Käyttäjä",
"login_with_spotify": "Kirjaudu Spotify-käyttäjällä",
"connect_with_spotify": "Yhdistä Spotify:lla",
"logout": "Kirjaudu ulos",
"logout_of_this_account": "Kirjaudu ulos tältä käyttäjältä",
"language_region": "Kieli ja Maa",
"language": "Kieli",
"system_default": "Järjestelmän oletus",
"market_place_region": "Markkina-alue",
"recommendation_country": "Suositeltu maa",
"appearance": "Ulkomuto",
"layout_mode": "Asettelutila",
"override_layout_settings": "Jätä reagoiva asettelutila huomioimatta",
"adaptive": "Mukautuva",
"compact": "Kompakti",
"extended": "Laajennettu",
"theme": "Teema",
"dark": "Tumma",
"light": "Vaalea",
"system": "Järjestelmä",
"accent_color": "Korostusväri",
"sync_album_color": "Synkronoi albumin väri",
"sync_album_color_description": "Käyttää albumin kansitaiteen vallitsevaa väirä korostuvärinä",
"playback": "Toisto",
"audio_quality": "Äänenlaatu",
"high": "Korkea",
"low": "Matala",
"pre_download_play": "Esilataa ja soita",
"pre_download_play_description": "Audion suoratoiston sijaan, lataa tavut ja soita ne (Suositeltu korkeamman kaistanleveyden käyttäjille)",
"skip_non_music": "Ohita ei-musiikki kohdat (SponsorBlock)",
"blacklist_description": "Mustalistat kappaleet aja artistit",
"wait_for_download_to_finish": "Odota nykyisen latauksen lopetteluun",
"desktop": "Työpöytä",
"close_behavior": "Sulkemisen käyttäytyminen",
"close": "Sulje",
"minimize_to_tray": "Minimisoi tehtäväpalkkiin",
"show_tray_icon": "Näytä järjestelmäkuvake",
"about": "Tietoa",
"u_love_spotube": "Tiedämme että rakastat Spotubea",
"check_for_updates": "Tarkista päivitykset",
"about_spotube": "Tietoa Spotube:sta",
"blacklist": "Mustalista",
"please_sponsor": "Sponsoroi/Lahjoita, kiitos",
"spotube_description": "Spotube, kevyt, cross-platform, vapaa-kaikille spotify clientti",
"version": "Versio",
"build_number": "Rakennusnumero",
"founder": "Perustaja",
"repository": "Arkisto",
"bug_issues": "Bugit+Ongelmat",
"made_with": "Tehty ❤️ Bangladeshista 🇧🇩",
"kingkor_roy_tirtho": "Kingkor Roy Tirtho",
"copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
"license": "Lisenssi",
"add_spotify_credentials": "Lisää Spotify-tunnuksesi aloittaaksesi",
"credentials_will_not_be_shared_disclaimer": "Älä huoli, tunnuksiasi ei talleteta tai jaeta kenenkään kanssa",
"know_how_to_login": "Etkö tiedä miten tehdä tämä?",
"follow_step_by_step_guide": "Seuraa askel askeleelta opasta",
"spotify_cookie": "Spotify {name} Keksi",
"cookie_name_cookie": "{name} Keksi",
"fill_in_all_fields": "Täytä kaikki kentät",
"submit": "Lähetä",
"exit": "Poistu",
"previous": "Edellinen",
"next": "Seuraava",
"done": "Tehty",
"step_1": "Vaihe 1",
"first_go_to": "Ensiksi, mene",
"login_if_not_logged_in": "ja Kirjaudu/Tee tili jos et ole kirjautunut sisään",
"step_2": "Vaihe 2",
"step_2_steps": "1. Kun olet kirjautunut, paina F12 tai oikeaa hiiren näppäintä > Tarkista ja avaa selaimen kehittäjä työkalut.\n2. Mene sitten \"Application\"-välilehteen (Chrome, Edge, Brave jne..) tai \"Storage\"-välilehteen (Firefox, Palemoon jne..)\n3. Mene \"Cookies\"-osastoon, sitten \"https://accounts.spotify.com\" alakohtaan.",
"step_3": "Vaihe 3",
"step_3_steps": "Kopioi Keksin \"sp_dc\" arvo",
"success_emoji": "Onnistuit🥳",
"success_message": "Olet nyt kirjautunut sisään Spotify-käyttäjällesi. Hyvää työtä toveri!",
"step_4": "Vaihe 4",
"step_4_steps": "Liitä kopioitu \"sp_dc\" arvo",
"something_went_wrong": "Jotain meni pieleen",
"piped_instance": "Johdettu palvelinesiintymä",
"piped_description": "Johdettu palvelinesiintymä Kappale täsmäyksiin",
"piped_warning": "Jotkut niistä eivät toimi hyvin, käytä siis omalla vastuullasi",
"generate_playlist": "Tuota soittolista",
"track_exists": "Kappale {track} on jo olemassa!",
"replace_downloaded_tracks": "Korvaa kaikki ladatut kappaleet",
"skip_download_tracks": "Ohita ladattujen laulujen lataaminen",
"do_you_want_to_replace": "Haluatko korvata olemassa olevan kappaleen??",
"replace": "Korvaa",
"skip": "Ohita",
"select_up_to_count_type": "Valitse enintään {count} {type}",
"select_genres": "Valitse Genret",
"add_genres": "Lisää Genrejä",
"country": "Maa",
"number_of_tracks_generate": "Numero tuotettavia kappaleita",
"acousticness": "Akustisuus",
"danceability": "Tanssittavuus",
"energy": "Energia",
"instrumentalness": "Instrumentaalisuus",
"liveness": "Elävyyttä",
"loudness": "Äänekkyys",
"speechiness": "Puheisuus",
"valence": "Valenssi",
"popularity": "Suosio",
"key": "Sävellaji",
"duration": "Pituus (s)",
"tempo": "Tempo (BPM)",
"mode": "Tila",
"time_signature": "Aikamerkki",
"short": "Lyhyt",
"medium": "Keskikokoinen",
"long": "Pitkä",
"min": "Minimi",
"max": "Maximi",
"target": "Kohde",
"moderate": "Kohtalainen",
"deselect_all": "Poista kaikki valinnat",
"select_all": "Valitse kaikki",
"are_you_sure": "Oletko varma?",
"generating_playlist": "Luodaan mukautettua soittolistoa...",
"selected_count_tracks": "Valittu {count} kappaletta",
"download_warning": "Jos lataat kaikki laulut kerrällä olet selkeästi Piratoimassa ja aiheuttamassa vahinkoa musiikin luovaan yhteiskuntaan. Toivottavasti olet tietoinen tästä. Yritä aina kunnioittaa ja tukea Artistin kovaa työtä.",
"download_ip_ban_warning": "BTW, YouTube voi estää IP-Osoitteesi tavallista liiallisten latauspyyntöjen takia. IP-Osoitteen esto tarkoittaa sitä, ettet voi käyttää YouTubea (vaikka olisit kirjautunut) vähintään 2-3kk aikana kyseiseltä laitteelta. Spotube ei kanna yhtään vastuuta jos se tapahtuu.",
"by_clicking_accept_terms": "Painamalla 'hyväksy' hyväksyt seuraaviin ehtoihin:",
"download_agreement_1": "Tiedän että Piratoin musiikkia. Olen paha.",
"download_agreement_2": "Tuen Artisteja silloin kun pystyn, ja teen tämän vain koska minulla ei ole rahaa ostaa heidän taidetta",
"download_agreement_3": "Ymmärrän että minun YouTube voi estää IP-Osoitteeni ja en pidä Spotubea tai omistajiinsa/avustajia vastuullisena mistään omista teoistsani",
"decline": "Hylkää",
"accept": "Hyväksy",
"details": "Yksityiskohdat",
"youtube": "YouTube",
"channel": "Kanava",
"likes": "Tykkäykset",
"dislikes": "Epä-tykkäykset",
"views": "Näyttökerrat",
"streamUrl": "Suoratoiston URL",
"stop": "Lopeta",
"sort_newest": "Suodata uusimmista",
"sort_oldest": "Suodata vanhimmista",
"sleep_timer": "Uniajastin",
"mins": "{minutes} Minuuttia",
"hours": "{hours} Tuntia",
"hour": "{hours} Tunti",
"custom_hours": "Mukautetut tunnit",
"logs": "Lokit",
"developers": "Kehittäjät",
"not_logged_in": "Et ole kirjautunut sisään.",
"search_mode": "Hakutila",
"audio_source": "Äänilähde",
"ok": "Ok",
"failed_to_encrypt": "Salaaminen epäonnistui",
"encryption_failed_warning": "Spotube käyttää salausta tallentaakseen tietosi, mutta epäonnistui, joten se palaa epäturvalliseen tallennukseen\nJos käytät Linuxia, varmista että sinulla on turvallisuuspalvelu (gnome-keyring, kde-wallet, keepassxc jne) asennettu",
"querying_info": "Hankitaan tietoa...",
"piped_api_down": "Johdettu palvelinesiintymä on alhaalla",
"piped_down_error_instructions": "Johdettu palvelinesiintymä {pipedInstance} on alhaalla.\n\nVaihda joko ilmeytymä tia vahda 'API tyyppi' YouTuben viralliseen API\n\nKäynnistä sovellus uudestaan vaihdon jälkeen",
"you_are_offline": "Et ole yhdistetty verkkoon",
"connection_restored": "Verkkoyhteys palautettu",
"use_system_title_bar": "Käytä järjestelmäpalkkia",
"crunching_results": "Paloitellaan tuloksia...",
"search_to_get_results": "Hae saadakseen tuloksia",
"use_amoled_mode": "Pilkkopimeä tumma teema",
"pitch_dark_theme": "AMOLED Tila",
"normalize_audio": "Normalisoi audio",
"change_cover": "Vaihda koveri",
"add_cover": "Lisää koveri",
"restore_defaults": "Palauta oletukset",
"download_music_codec": "Ladatun musiikin codefc",
"streaming_music_codec": "Suoratoistetun musiikin codec",
"login_with_lastfm": "Kirjaudu sisään Last.fm:llä",
"connect": "Yhdistä",
"disconnect_lastfm": "Katkaise Last.fm",
"disconnect": "Katkaise",
"username": "Käyttäjänimi",
"password": "Salasana",
"login": "Kirjaudu",
"login_with_your_lastfm": "Kirjaudu Last.fm käyttäjälläsi",
"scrobble_to_lastfm": "Scrobble Last.fm:ään",
"go_to_album": "Mene albumiin",
"discord_rich_presence": "Discord Rich Presence",
"browse_all": "Selaa kaikki",
"genres": "Genret",
"explore_genres": "Seikkaile genrejä",
"friends": "Kaverit",
"no_lyrics_available": "Anteeksi, emme löytäneet lyriikoita tälle laululle",
"start_a_radio": "Aloita Radio",
"how_to_start_radio": "Kuinka haluat aloittaa radion?",
"replace_queue_question": "Haluatko korvata nykyisen jonon vai lisätä siihen?",
"endless_playback": "Loputon toisto",
"delete_playlist": "Poista soittolista",
"delete_playlist_confirmation": "Oletko varma että haluat poistaa tämän soittolistan?",
"local_tracks": "Paikalliset kappaleet",
"song_link": "Laulun linkki",
"skip_this_nonsense": "Ohita tämä hölynpöly",
"freedom_of_music": "“Musiikin vapaus”",
"freedom_of_music_palm": "“Musiikin vapaus käsissäsi”",
"get_started": "Aloitetaan",
"youtube_source_description": "Suositeltu ja toimii parhaiten.",
"piped_source_description": "Tuntuuko vapaalta? Sama kuin YouTube mutta paljon vapautta",
"jiosaavn_source_description": "Paras Etelä-Aasian alueelle.",
"highest_quality": "Korkein laatu: {quality}",
"select_audio_source": "Valitse äänilähde",
"endless_playback_description": "Lisää automaattisesti uusia lauluja\njonon perään",
"choose_your_region": "Valitse alueesi",
"choose_your_region_description": "Tämä auttaa Spotube näyttämään sinulle oikeaa sisältöä\nsijaintiasi varten.",
"choose_your_language": "Valitse kielesi",
"help_project_grow": "Auta tätä projektia kasvamaan",
"help_project_grow_description": "Spotube projekti minkä lähdekoodi on julkisesti saatavilla. Voit autta tätä projektia kasvamaan muutoksilla, ilmoittamalla bugeista, tai ehdottamalla uusia ominaisuuksia.",
"contribute_on_github": "Auta GitHub:ssa",
"donate_on_open_collective": "Lahjoita avoimessa kollektiivissa",
"browse_anonymously": "Selaa anonyyminä",
"enable_connect": "Ota käyttöön yhdistäminen",
"enable_connect_description": "Ohjaa Spotubea toiselta laitteelta",
"devices": "Laitteet",
"select": "Valitse",
"connect_client_alert": "{client} ohjaa sinua",
"this_device": "Tämä laite",
"remote": "Etä"
}

324
lib/l10n/app_id.arb Normal file
View File

@ -0,0 +1,324 @@
{
"guest": "Tamu",
"browse": "Jelajahi",
"search": "Cari",
"library": "Pustaka",
"lyrics": "Lirik",
"settings": "Pengaturan",
"genre_categories_filter": "Urutkan kategori atau genre...",
"genre": "Genre",
"personalized": "Dipersonalisasi",
"featured": "Unggulan",
"new_releases": "Rilis Terbaru",
"songs": "Lagu",
"playing_track": "Memutar {track}",
"queue_clear_alert": "Ini akan menghapus antrian saat ini This will clear the current queue. {track_length} trek akan dihapus\nAnda ingin melanjutkan?",
"load_more": "Lebih Banyak",
"playlists": "Daftar Putar",
"artists": "Artis",
"albums": "Album",
"tracks": "Trek",
"downloads": "Unduhan",
"filter_playlists": "Urutkan daftar putar Anda...",
"liked_tracks": "Lagu Yang Disukai",
"liked_tracks_description": "Semua lagu yang Anda sukai",
"create_playlist": "Buat Daftar Putar",
"create_a_playlist": "Buat daftar putar",
"update_playlist": "Ubah daftar putar",
"create": "Buat",
"cancel": "Batal",
"update": "Ubah",
"playlist_name": "Nama Daftar Putar",
"name_of_playlist": "Nama daftar putar",
"description": "Deskripsi",
"public": "Publik",
"collaborative": "Kolaboratif",
"search_local_tracks": "Cari trek lokal...",
"play": "Putar",
"delete": "Hapus",
"none": "Tidak Ada",
"sort_a_z": "Urutkan berdasarkan A-Z",
"sort_z_a": "Urutkan berdasarkan Z-A",
"sort_artist": "Urutkan berdasarkan Artis",
"sort_album": "Urutkan berdasarkan Album",
"sort_duration": "Urutkan berdasarkan Durasi",
"sort_tracks": "Urutkan trek",
"currently_downloading": "Sedang Mengunduh ({tracks_length})",
"cancel_all": "Batalkan Semua",
"filter_artist": "Urutkan artis...",
"followers": "{followers} Pengikut",
"add_artist_to_blacklist": "Tambah artis ke daftar hitam",
"top_tracks": "Lagu Teratas",
"fans_also_like": "Penggemar juga menyukainya",
"loading": "Memuat...",
"artist": "Artis",
"blacklisted": "Masuk Daftar Hitam",
"following": "Mengikuti",
"follow": "Ikuti",
"artist_url_copied": "URL artis telah disalin",
"added_to_queue": "Menambah trek {tracks} ke antrean",
"filter_albums": "Urutkan album...",
"synced": "Disinkronkan",
"plain": "Normal",
"shuffle": "Acak",
"search_tracks": "Cari trek...",
"released": "Dirilis",
"error": "Kesalahan {error}",
"title": "Judul",
"time": "Waktu",
"more_actions": "Tindakan Lainnya",
"download_count": "Unduhan ({count})",
"add_count_to_playlist": "Menambah ({count}) ke Daftar Putar",
"add_count_to_queue": "Menambah ({count}) ke Antrian",
"play_count_next": "Mainkan ({count}) selanjutnya",
"album": "Album",
"copied_to_clipboard": "{data} telah disalin",
"add_to_following_playlists": "Menambah {track} ke Daftar Putar berikut",
"add": "Tambah",
"added_track_to_queue": "Menambah {track} ke antrian",
"add_to_queue": "Tambah ke antrian",
"track_will_play_next": "{track} akan diputar berikutnya",
"play_next": "Mainkan selanjutnya",
"removed_track_from_queue": "Menghapus {track} dari antrian",
"remove_from_queue": "Hapus dari antrian",
"remove_from_favorites": "Hapus dari favorit",
"save_as_favorite": "Simpan sebagai favorit",
"add_to_playlist": "Tambah ke daftar putar",
"remove_from_playlist": "Hapus dari daftar putar",
"add_to_blacklist": "Tambah ke daftar hitam",
"remove_from_blacklist": "Hapus dari daftar hitam",
"share": "Bagikan",
"mini_player": "Pemutar Mini",
"slide_to_seek": "Geser untuk maju atau mundur",
"shuffle_playlist": "Acak daftar putar",
"unshuffle_playlist": "Batalkan pengacakan daftar putar",
"previous_track": "Lagu sebelumnya",
"next_track": "Lagu berikutnya",
"pause_playback": "Jeda Pemutaran",
"resume_playback": "Lanjutkan Pemutaran",
"loop_track": "Ulangi Pemutaran",
"repeat_playlist": "Ulangi daftar putar",
"queue": "Antrian",
"alternative_track_sources": "Sumber trek alternatif",
"download_track": "Unduh lagu",
"tracks_in_queue": "{tracks} trek dalam antrian",
"clear_all": "Bersihkan semua",
"show_hide_ui_on_hover": "Tampil/Sembunyikan UI saat mengarahkan kursor",
"always_on_top": "Selalu di atas",
"exit_mini_player": "Keluar Pemutar Mini",
"download_location": "Lokasi unduhan",
"account": "Akun",
"login_with_spotify": "Masuk dengan Spotify",
"connect_with_spotify": "Hubungkan dengan Spotify",
"logout": "Keluar",
"logout_of_this_account": "Keluar dari akun",
"language_region": "Bahasa & Wilayah",
"language": "Bahasa",
"system_default": "Bawaan Sistem",
"market_place_region": "Wilayah Pasar",
"recommendation_country": "Negara Rekomendasi",
"appearance": "Tampilan",
"layout_mode": "Mode Tata Letak",
"override_layout_settings": "Ganti pengaturan mode tata letak responsif",
"adaptive": "Adaptif",
"compact": "Ringkas",
"extended": "Diperluas",
"theme": "Tema",
"dark": "Gelap",
"light": "Terang",
"system": "Sistem",
"accent_color": "Warna Aksen",
"sync_album_color": "Sinkronkan warna album",
"sync_album_color_description": "Menggunakan warna dominan sampul album sebagai warna aksen",
"playback": "Pemutaran",
"audio_quality": "Kualitas Suara",
"high": "Tinggi",
"low": "Rendah",
"pre_download_play": "Unduh dan putar",
"pre_download_play_description": "Daripada streaming audio, unduh byte dan mainkan (Direkomendasikan untuk pengguna bandwidth yang lebih tinggi)",
"skip_non_music": "Lewati segmen non-musik (SponsorBlock)",
"blacklist_description": "Lagu dan artis di daftar hitam",
"wait_for_download_to_finish": "Tunggu hingga unduhan saat ini selesai",
"desktop": "Desktop",
"close_behavior": "Tutup Perilaku",
"close": "Tutup",
"minimize_to_tray": "Perkecil ke tray",
"show_tray_icon": "Tampilkan tray ikon sistem",
"about": "Tentang",
"u_love_spotube": "Kami tahu Anda menyukai Spotube",
"check_for_updates": "Periksa pembaruan",
"about_spotube": "Tentang Spotube",
"blacklist": "Daftar Hitam",
"please_sponsor": "Silakan Sponsor/Menyumbang",
"spotube_description": "Spotube, klien Spotify yang ringan, lintas platform, dan gratis untuk semua",
"version": "Versi",
"build_number": "Nomor Pembuatan",
"founder": "Pendiri",
"repository": "Repositori",
"bug_issues": "Bug+Masalah",
"made_with": "Dibuat dengan ❤️ di Bangladesh🇧🇩",
"kingkor_roy_tirtho": "Kingkor Roy Tirtho",
"copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
"license": "Lisensi",
"add_spotify_credentials": "Tambahkan kredensial Spotify Anda untuk memulai",
"credentials_will_not_be_shared_disclaimer": "Jangan khawatir, kredensial Anda tidak akan dikumpulkan atau dibagikan kepada siapa pun",
"know_how_to_login": "Tidak tahu bagaimana melakukan ini?",
"follow_step_by_step_guide": "Ikuti panduan Langkah demi Langkah",
"spotify_cookie": "Spotify {name} Cookie",
"cookie_name_cookie": "{name} Cookie",
"fill_in_all_fields": "Silakan isi semua kolom",
"submit": "Kirim",
"exit": "Keluar",
"previous": "Sebelumnya",
"next": "Berikutnya",
"done": "Selesai",
"step_1": "Langkah 1",
"first_go_to": "Pertama, Pergi ke",
"login_if_not_logged_in": "dan Masuk/Daftar jika Anda belum masuk",
"step_2": "Langkah 2",
"step_2_steps": "1. Setelah Anda masuk, tekan F12 atau Klik Kanan Mouse > Buka Browser Devtools.\n2. Lalu buka Tab \"Aplikasi\" (Chrome, Edge, Brave, dll.) atau Tab \"Penyimpanan\" (Firefox, Palemoon, dll.)\n3. Buka bagian \"Cookie\" lalu subbagian \"https://accounts.spotify.com\"",
"step_3": "Langkah 3",
"step_3_steps": "Salin nilai Cookie \"sp_dc\" ",
"success_emoji": "Berhasil🥳",
"success_message": "Sekarang Anda telah berhasil Masuk dengan akun Spotify Anda. Kerja bagus, sobat!",
"step_4": "Langkah 4",
"step_4_steps": "Tempel nilai \"sp_dc\" yang disalin",
"something_went_wrong": "Terjadi kesalahan",
"piped_instance": "Piped Server Instance",
"piped_description": "The Piped server instance untuk digunakan sebagai pencocokan trek",
"piped_warning": "Beberapa di antaranya mungkin tidak berfungsi dengan baik. Jadi gunakan dengan risiko Anda sendiri",
"generate_playlist": "Hasilkan Daftar Putar",
"track_exists": "Lagu {track} sudah ada",
"replace_downloaded_tracks": "Ganti semua trek yang diunduh",
"skip_download_tracks": "Lewati pengunduhan semua trek yang diunduh",
"do_you_want_to_replace": "Apakah Anda ingin mengganti track yang ada?",
"replace": "Ganti",
"skip": "Lewati",
"select_up_to_count_type": "Pilih hingga {count} {type}",
"select_genres": "Pilih Genre",
"add_genres": "Tambah Genre",
"country": "Negara",
"number_of_tracks_generate": "Jumlah trek yang akan dihasilkan",
"acousticness": "Akustik",
"danceability": "Menari",
"energy": "Energi",
"instrumentalness": "Instrumentalitas",
"liveness": "Kehidupan",
"loudness": "Kekerasan",
"speechiness": "Berbicara",
"valence": "Valensi",
"popularity": "Popularitas",
"key": "Kunci",
"duration": "Durasi (s)",
"tempo": "Tempo (BPM)",
"mode": "Mode",
"time_signature": "Tanda Tangan Waktu",
"short": "Pendek",
"medium": "Sedang",
"long": "Panjang",
"min": "Minimal",
"max": "Maksimal",
"target": "Target",
"moderate": "Sedang",
"deselect_all": "Batalkan Semua",
"select_all": "Pilih Semua",
"are_you_sure": "Anda yakin?",
"generating_playlist": "Menghasilkan daftar putar khusus Anda...",
"selected_count_tracks": "{count} lagu yang dipilih",
"download_warning": "Jika Anda mengunduh semua Lagu secara massal, Anda jelas membajak Musik & menyebabkan kerusakan pada masyarakat kreatif Musik. Saya harap Anda menyadari hal ini. Selalu berusaha menghormati & mendukung kerja keras Artis",
"download_ip_ban_warning": "BTW, IP Anda bisa diblokir di YouTube karena permintaan unduhan yang berlebihan dari biasanya. Blokir IP berarti Anda tidak dapat menggunakan YouTube (meskipun Anda masuk) setidaknya selama 2-3 bulan dari perangkat IP tersebut. Dan Spotube tidak bertanggung jawab jika hal ini terjadi",
"by_clicking_accept_terms": "Dengan mengklik 'terima' Anda menyetujui ketentuan berikut:",
"download_agreement_1": "Saya tahu saya membajak Musik. Saya buruk",
"download_agreement_2": "Saya akan mendukung Artis di mana pun saya bisa dan saya melakukan ini hanya karena saya tidak punya uang untuk membeli karya seni mereka",
"download_agreement_3": "Saya sepenuhnya menyadari bahwa IP saya dapat diblokir di YouTube & saya tidak menganggap Spotube atau pemilik/kontributornya bertanggung jawab atas kecelakaan apa pun yang disebabkan oleh tindakan saya saat ini",
"decline": "Menolak",
"accept": "Setuju",
"details": "Detail",
"youtube": "YouTube",
"channel": "Channel",
"likes": "Suka",
"dislikes": "Tidak Suka",
"views": "Dilihat",
"streamUrl": "URL Stream",
"stop": "Berhenti",
"sort_newest": "Urutkan yang baru ditambah",
"sort_oldest": "Urutkan yang paling lama ditambah",
"sleep_timer": "Pengatur Waktu Tidur",
"mins": "{minutes} Menit",
"hours": "{hours} Jam",
"hour": "{hours} Jam",
"custom_hours": "Jam Kostum",
"logs": "Log",
"developers": "Pengembang",
"not_logged_in": "Anda belum masuk",
"search_mode": "Mode Pencarian",
"audio_source": "Sumber Suara",
"ok": "OK",
"failed_to_encrypt": "Gagal mengenkripsi",
"encryption_failed_warning": "Spotube menggunakan enkripsi untuk menyimpan data Anda dengan aman. Namun gagal melakukannya. Jadi itu akan kembali ke penyimpanan yang tidak aman\nJika Anda menggunakan linux, pastikan Anda telah menginstal layanan rahasia (gnome-keyring, kde-wallet, keepassxc, dll)",
"querying_info": "Mencari informasi...",
"piped_api_down": "Piped API tidak aktif",
"piped_down_error_instructions": "Piped Instance {pipedInstance} saat ini tidak aktif\n\nUbah instance atau ubah 'jenis API' menjadi API YouTube resmi\n\nPastikan untuk memulai ulang aplikasi setelah perubahan",
"you_are_offline": "Anda sedang offline",
"connection_restored": "Koneksi internet Anda telah pulih",
"use_system_title_bar": "Gunakan bilah judul sistem",
"crunching_results": "Mengolah hasil...",
"search_to_get_results": "Cari untuk mendapatkan hasil",
"use_amoled_mode": "Tema gelap gulita",
"pitch_dark_theme": "Mode AMOLED",
"normalize_audio": "Normalisasi audio",
"change_cover": "Ganti sampul",
"add_cover": "Tambah sampul",
"restore_defaults": "Kembalikan semula",
"download_music_codec": "Unduh codec musik",
"streaming_music_codec": "Streaming codec musik",
"login_with_lastfm": "Masuk dengan Last.fm",
"connect": "Hubungkan",
"disconnect_lastfm": "Memutuskan Last.fm",
"disconnect": "Memutuskan",
"username": "Username",
"password": "Password",
"login": "Masuk",
"login_with_your_lastfm": "Masuk dengan Last.fm Anda",
"scrobble_to_lastfm": "Scrobble ke Last.fm",
"go_to_album": "Pergi ke Album",
"discord_rich_presence": "Discord Rich Presence",
"browse_all": "Lihat Semua",
"genres": "Genre",
"explore_genres": "Jelajahi Genre",
"friends": "Daftar Teman",
"no_lyrics_available": "Maaf, tidak dapat menemukan lirik untuk lagu ini",
"start_a_radio": "Putar Radio",
"how_to_start_radio": "Bagaimana Anda ingin memutar radio?",
"replace_queue_question": "Apakah Anda ingin mengganti antrean saat ini atau menambahkannya?",
"endless_playback": "Pemutaran Tanpa Akhir",
"delete_playlist": "Hapus Daftar Putar",
"delete_playlist_confirmation": "Anda yakin ingin menghapus daftar putar ini?",
"local_tracks": "Trek Lokal",
"song_link": "Tautan Lagu",
"skip_this_nonsense": "Lewati omong kosong ini",
"freedom_of_music": "“Kebebasan Musik”",
"freedom_of_music_palm": "“Kebebasan Musik di telapak tangan Anda”",
"get_started": "Mari kita mulai",
"youtube_source_description": "Direkomendasikan dan berfungsi paling baik.",
"piped_source_description": "Merasa bebas? Sama seperti YouTube tetapi banyak yang gratis.",
"jiosaavn_source_description": "Terbaik untuk wilayah Asia Selatan.",
"highest_quality": "Kualitas Terbaik: {quality}",
"select_audio_source": "Pilih Sumber Suara",
"endless_playback_description": "Tambahkan lagu baru secara otomatis\nke akhir antrean",
"choose_your_region": "Pilih wilayah Anda",
"choose_your_region_description": "Ini akan membantu Spotube menampilkan konten yang tepat\nuntuk lokasi Anda.",
"choose_your_language": "Pilih bahasa Anda",
"help_project_grow": "Bantu proyek ini berkembang",
"help_project_grow_description": "Spotube adalah proyek sumber terbuka. Anda dapat membantu proyek ini berkembang dengan berkontribusi pada proyek, melaporkan bug, atau menyarankan fitur baru.",
"contribute_on_github": "Berkontribusi di GitHub",
"donate_on_open_collective": "Donasi di Open Collective",
"browse_anonymously": "Jelajahi Secara Anonim",
"enable_connect": "Aktifkan Hubungkan",
"enable_connect_description": "Kontrol Spotube dari perangkat lain",
"devices": "Perangkat",
"select": "Pilih",
"connect_client_alert": "Anda dikendalikan oleh {client}",
"this_device": "Perangkat Ini",
"remote": "Remot"
}

324
lib/l10n/app_ka.arb Normal file
View File

@ -0,0 +1,324 @@
{
"guest": "სტუმარი",
"browse": "ნახვა",
"search": "ძებნა",
"library": "ბიბლიოთეკა",
"lyrics": "ტექსტები",
"settings": "კონფიგურაციები",
"genre_categories_filter": "კატეგორიების ან ჟანრების ფილტრი...",
"genre": "ჟანრი",
"personalized": "პეერსონალიზებული",
"featured": "გამორჩეული",
"new_releases": "ახალი გამოცემები",
"songs": "სიმღერები",
"playing_track": "უკრავს {track}",
"queue_clear_alert": "ეს გაასუფთავებს მიმდინარე რიგს. {track_length} ტრეკი წაიშლება\nᲒინდა გააგრძელო?",
"load_more": "მეტის ჩატვირთვა",
"playlists": "ფლეილისტები",
"artists": "არტისტები",
"albums": "ალბომები",
"tracks": "ტრეკები",
"downloads": "ჩამოტვირთვები",
"filter_playlists": "ფლეილისტების გაფილტვრა...",
"liked_tracks": "მოწონებული ტრეკები",
"liked_tracks_description": "ყველა შენი მოწონებული ტრეკი",
"create_playlist": "ფლეილისტის შექმნა",
"create_a_playlist": "ფლეილისტის შექმნა",
"update_playlist": "ფლეილისტის განახლება",
"create": "შექმნა",
"cancel": "გაუქმება",
"update": "განახლება",
"playlist_name": "ფლეილისტის სახელი",
"name_of_playlist": "ფლეილისტის სახელი",
"description": "აღწერა",
"public": "საჯარო",
"collaborative": "კოლაბორაციული",
"search_local_tracks": "ლოცალური ტრეკების ძებნა...",
"play": "დაკვრა",
"delete": "წაშლა",
"none": "არცერთი",
"sort_a_z": "დალაგება A-Z-ს მიხედვით",
"sort_z_a": "დალაგება Z-A-ს მიხედვით",
"sort_artist": "დალაგება არტისტის მიხედვით",
"sort_album": "დალაგება ალბომის მიხედვით",
"sort_duration": "დალაგება ხანგრძლივობის მიხედვით",
"sort_tracks": "ტრეკების დალაგება",
"currently_downloading": "მიმდინარეობს ჩამოტვირთვა ({tracks_length})",
"cancel_all": "ყველას გაუქმება",
"filter_artist": "არტისტების ფილტრი...",
"followers": "{followers} ფოლოვერები",
"add_artist_to_blacklist": "არტისტის შავ სიაში დამატება",
"top_tracks": "ტოპ ტრეკები",
"fans_also_like": "ფანებს ასევე მოსწონთ",
"loading": "იტვირთება...",
"artist": "არტისტი",
"blacklisted": "შავ სიაში მყოფი",
"following": "ფოლოვინგი",
"follow": "დაფოლოვება",
"artist_url_copied": "არტისტის ლინკი დაკოპირებულია",
"added_to_queue": "{tracks} ტრეკი დაემატა რიგში",
"filter_albums": "ალბომების გაფილტვრა...",
"synced": "სინქრონიზებული",
"plain": "Plain",
"shuffle": "რიგის არევა",
"search_tracks": "ტრეკების ძებნა...",
"released": "გამოშვებული",
"error": "შეცდომა {error}",
"title": "სათაური",
"time": "დრო",
"more_actions": "მეტი მოქმედებები",
"download_count": "გადმოწერა ({count})",
"add_count_to_playlist": "ფლეილისტში ({count})-ის დამატება",
"add_count_to_queue": "რიგში ({count})-ის დამატება",
"play_count_next": "შემდეგი ({count})-ის დაკვრა",
"album": "ალბომი",
"copied_to_clipboard": "{data} დაკოპირებულია",
"add_to_following_playlists": "დაამატე {track} ამ ფლეილისტებში",
"add": "დამატება",
"added_track_to_queue": "რიგში დაემატა {track}",
"add_to_queue": "რიგში დამატება",
"track_will_play_next": "{track} დაუკრავს შემდეგს",
"play_next": "შემდეგის დაკვრა",
"removed_track_from_queue": "რიგიდან წაიშალა {track}",
"remove_from_queue": "რიგიდან წაშლა",
"remove_from_favorites": "ფავორიტებიდან წაშლა",
"save_as_favorite": "ფავორიტებში დამატება",
"add_to_playlist": "ფლეილისტში დამატება",
"remove_from_playlist": "ფლეილისტიდან წაშლა",
"add_to_blacklist": "შავ სიაში დამატება",
"remove_from_blacklist": "შავი სიიდან წაშლა",
"share": "გაზიარება",
"mini_player": "მინი დამკვრელი",
"slide_to_seek": "გადახვევისთვის გაასრიალეთ წინ ან უკან",
"shuffle_playlist": "ფლეილისტის არევა",
"unshuffle_playlist": "ფლეილისტის დალაგება",
"previous_track": "წინა ტრეკი",
"next_track": "შემდეგი ტრეკი",
"pause_playback": "დაკვრის გაჩერება",
"resume_playback": "დაკვრის გაგრძელება",
"loop_track": "ტრეკის ლუპზე დაკვრა",
"repeat_playlist": "ფლეილისტის გამეორება",
"queue": "რიგი",
"alternative_track_sources": "ალტერნატიული ტრეკების წყაროები",
"download_track": "გადმოწერე ტრეკი",
"tracks_in_queue": "{tracks} ტრეკი რიგში",
"clear_all": "ყველას წაშლა",
"show_hide_ui_on_hover": "UI-ის ჩვენება/დამალვა ჰოვერზე",
"always_on_top": "ტოველთვის ზემოდან",
"exit_mini_player": "მინი დამკვრელიდან გამოსვლა",
"download_location": "ჩამოტვირთვის მდებარეობა",
"account": "ანგარიში",
"login_with_spotify": "შედით თქვენი Spotify ანგარიშით",
"connect_with_spotify": "დაუკავშირდით Spotify-ს",
"logout": "გასვლა",
"logout_of_this_account": "ანგარიშიდან გასვლა",
"language_region": "ენა და რეგიონი",
"language": "ენა",
"system_default": "სისტემის ნაგულისხმევი",
"market_place_region": "მარკეტფლეისის რეგიონი",
"recommendation_country": "რეკომენდირებული ქვეყანა",
"appearance": "გარეგნობა",
"layout_mode": "განლაგების რეჟიმი",
"override_layout_settings": "რესფონსივ განლაგების რეჟიმის კონფიგურაციაზე გადაწერა",
"adaptive": "ადაპტირებული",
"compact": "კომპაქტური",
"extended": "გაფართოებული",
"theme": "თემა",
"dark": "ბნელი",
"light": "ღია",
"system": "სისტემის",
"accent_color": "აქცენტის ფერი",
"sync_album_color": "ალბომის ფერის სინქრონიზაცია",
"sync_album_color_description": "დომინანტური ალბომის ფერის აქცენტის ფერად გამოყენება",
"playback": "დაკვრა",
"audio_quality": "აუდიოს ხარისხი",
"high": "მაღალი",
"low": "დაბალი",
"pre_download_play": "წინასწარ ჩამოტვირთვა და დაკვრა",
"pre_download_play_description": "აუდიოს სტრიმინგის ნაცვლად, ბაიტების ჩამოტვირთვა და დაკვრა (რეკომენდებულია უფრო მაღალი გამტარუნარიანობის მომხმარებლებისთვის)",
"skip_non_music": "არა მუსიკალური ნაწილის გამოტოვება (სპონსორის ბლოკი)",
"blacklist_description": "შავ სიაში მყოფი არტისტები და ტრეკები",
"wait_for_download_to_finish": "გთხოვთ, დაელოდოთ მიმდინარე ჩამოტვირთვის დასრულებას",
"desktop": "დესკტოპი",
"close_behavior": "დახურვის ქცევა",
"close": "დახურვა",
"minimize_to_tray": "მინიმიზაცია",
"show_tray_icon": "სისტემის აიკონის ჩვენება",
"about": "ჩვენს შესახებ",
"u_love_spotube": "We know you love Spotube",
"check_for_updates": "განახლებების შემოწმება",
"about_spotube": "Spotube-ს შესახებ",
"blacklist": "შავი სია",
"please_sponsor": "გთხოვთ დაგვასპონსოროთ",
"spotube_description": "Spotube, a lightweight, cross-platform, free-for-all spotify client",
"version": "ვერსია",
"build_number": "Build Number",
"founder": "დამფუძნებელი",
"repository": "რეპოზიტორია",
"bug_issues": "Bug+Issues",
"made_with": "Made with ❤️ in Bangladesh🇧🇩",
"kingkor_roy_tirtho": "Kingkor Roy Tirtho",
"copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
"license": "ლიცენზია",
"add_spotify_credentials": "დასაწყებად დაამატეთ თქვენი Spotify მონაცემები",
"credentials_will_not_be_shared_disclaimer": "არ ინერვიულოთ, თქვენი მონაცემები არ იქნება შეგროვებული ან გაზიარებული ვინმესთან",
"know_how_to_login": "არ იცით როგორ გააკეთოთ ეს?",
"follow_step_by_step_guide": "მიჰყევით ნაბიჯ-ნაბიჯ სახელმძღვანელოს",
"spotify_cookie": "Spotify {name} ქუქი",
"cookie_name_cookie": "{name} ქუქი",
"fill_in_all_fields": "გთხოვთ შეავსოთ ყველა ველი",
"submit": "გაგზავნა",
"exit": "გამოსვლა",
"previous": "წინა",
"next": "შემდეგი",
"done": "მზადაა",
"step_1": "ნაბიჯი 1",
"first_go_to": "პირველი, გადადით",
"login_if_not_logged_in": "და შესვლა/რეგისტრაცია, თუ არ ხართ შესული",
"step_2": "ნაბიჯი 2",
"step_2_steps": "1. როცა შეხვალთ, დააჭირეთ F12-ს ან მაუსის მარჯვენა ღილაკს > Inspect to Open the Browser devtools.\n2. შემდეგ გახსენით \"Application\" განყოფილება (Chrome, Edge, Brave etc..) ან \"Storage\" განყოფილება (Firefox, Palemoon etc..)\n3. შედით \"Cookies\" სექციაში და შემდეგ \"https://accounts.spotify.com\" სუბსექციაში",
"step_3": "ნაბიჯი 3",
"step_3_steps": "დააკოპირეთ \"sp_dc\" ქუქი-ფაილის მნიშვნელობა",
"success_emoji": "წარმატება🥳",
"success_message": "თქვენ წარმატებით შეხვედით თქვენი Spotify ანგარიშით.",
"step_4": "ნაბიჯი 4",
"step_4_steps": "ჩასვით კოპირებული \"sp_dc\" მნიშვნელობა",
"something_went_wrong": "Რაღაც არასწორად წავიდა",
"piped_instance": "Piped Server Instance",
"piped_description": "The Piped server instance to use for track matching",
"piped_warning": "ზოგიერთი მათგანმა შეიძლება კარგად არ იმუშაოს. ",
"generate_playlist": "ფლეილისტის დაგენერირება",
"track_exists": "ტრეკი {track} უკვე არსებობს",
"replace_downloaded_tracks": "ყველა ჩამოტვირთული ტრეკის შეცვლა",
"skip_download_tracks": "ყველა ჩამოტვირთული ტრეკის გამოტოვება",
"do_you_want_to_replace": "გსურთ შეცვალოთ არსებული ტრეკი??",
"replace": "შეცვლა",
"skip": "გამოტოვება",
"select_up_to_count_type": "აირჩიე {count}-მდე {type}",
"select_genres": "ჟანრების არჩევა",
"add_genres": "ჟანრების დამატება",
"country": "ქვეყანა",
"number_of_tracks_generate": "დასაგენერირებელი ტრეკების რაოდენობა",
"acousticness": "Acousticness",
"danceability": "Danceability",
"energy": "Energy",
"instrumentalness": "Instrumentalness",
"liveness": "Liveness",
"loudness": "Loudness",
"speechiness": "Speechiness",
"valence": "Valence",
"popularity": "Popularity",
"key": "Key",
"duration": "Duration (s)",
"tempo": "Tempo (BPM)",
"mode": "Mode",
"time_signature": "Time Signature",
"short": "Short",
"medium": "საშუალო",
"long": "გრძელი",
"min": "მინიმალური",
"max": "მაქსიმალური",
"target": "სამიზნე",
"moderate": "საშუალო",
"deselect_all": "ყველა მონიშვნის გაუქმება",
"select_all": "ყველას მონიშვნა",
"are_you_sure": "Დარწმუნებული ხართ?",
"generating_playlist": "მიმდინარეობს თქვენი მორგებული ფლეილისტის გენერირება...",
"selected_count_tracks": "არჩეულია {count} ტრეკი",
"download_warning": "If you download all Tracks at bulk you're clearly pirating Music & causing damage to the creative society of Music. I hope you are aware of this. Always, try respecting & supporting Artist's hard work",
"download_ip_ban_warning": "BTW, your IP can get blocked on YouTube due excessive download requests than usual. IP block means you can't use YouTube (even if you're logged in) for at least 2-3 months from that IP device. And Spotube doesn't hold any responsibility if this ever happens",
"by_clicking_accept_terms": "By clicking 'accept' you agree to following terms:",
"download_agreement_1": "I know I'm pirating Music. I'm bad",
"download_agreement_2": "I'll support the Artist wherever I can and I'm only doing this because I don't have money to buy their art",
"download_agreement_3": "I'm completely aware that my IP can get blocked on YouTube & I don't hold Spotube or his owners/contributors responsible for any accidents caused by my current action",
"decline": "უარყოფა",
"accept": "დათანხმება",
"details": "დეტალები",
"youtube": "YouTube",
"channel": "Channel",
"likes": "მოწონებები",
"dislikes": "არ მოწონებები",
"views": "ნახვები",
"streamUrl": "სტრიმის ლინკი",
"stop": "გაჩერება",
"sort_newest": "ფალაგება სიახლის მიხედიტ",
"sort_oldest": "დალაგება სიძველის მიხედვით",
"sleep_timer": "ძილის ტაიმერი",
"mins": "{minutes} წუთი",
"hours": "{hours} საათი",
"hour": "{hours} საათი",
"custom_hours": "მორგებული საათები",
"logs": "ლოგები",
"developers": "დეველოპერები",
"not_logged_in": "არ ხარ დალოგინებული",
"search_mode": "ძებნის რეჟიმი",
"audio_source": "აუდიოს წყარო",
"ok": "ოკ",
"failed_to_encrypt": "დაშიფვრა ვერ მოხერხდა",
"encryption_failed_warning": "Spotube uses encryption to securely store your data. But failed to do so. So it'll fallback to insecure storage\nIf you're using linux, please make sure you've any secret-service (gnome-keyring, kde-wallet, keepassxc etc) installed",
"querying_info": "Querying info...",
"piped_api_down": "Piped API is down",
"piped_down_error_instructions": "The Piped instance {pipedInstance} is currently down\n\nEither change the instance or change the 'API type' to official YouTube API\n\nMake sure to restart the app after change",
"you_are_offline": "ამჟამად ხაზგარეშე ხართ",
"connection_restored": "თქვენი ინტერნეტ კავშირი აღდგა",
"use_system_title_bar": "სისტემის სათაურის ზოლის გამოყენება",
"crunching_results": "იტვირთება შედეგები...",
"search_to_get_results": "მოძებნეთ შედეგების მისაღებად",
"use_amoled_mode": "Pitch black dark theme",
"pitch_dark_theme": "AMOLED Mode",
"normalize_audio": "აუდიოს ნორმალიზება",
"change_cover": "Ქავერის შეცვლა",
"add_cover": "Ქავერის ფოტოს დამატება",
"restore_defaults": "ნაგულისხმევი პარამეტრების აღდგენა",
"download_music_codec": "მუსიკის კოდეკის გადმოწერა",
"streaming_music_codec": "სტრიმინგ მუსიკის კოდეკი",
"login_with_lastfm": "Last.fm-ით შესვლა",
"connect": "დაკავშირება",
"disconnect_lastfm": "Last.fm-იდან გამოსვლა",
"disconnect": "გამოსვლა",
"username": "მომხმარებელი",
"password": "პაროლი",
"login": "შესვლა",
"login_with_your_lastfm": "Last.fm ანგარიშით შესვლა",
"scrobble_to_lastfm": "Scrobble to Last.fm",
"go_to_album": "ალბომზე გადასვლა",
"discord_rich_presence": "Discord Rich Presence",
"browse_all": "ყველას ნახვა",
"genres": "ჟანრები",
"explore_genres": "შეისწავლეთ ჟანრები",
"friends": "მეგობრები",
"no_lyrics_available": "უკაცრავად, ამ ტრეკისთვის ტექსტის პოვნა შეუძლებელია",
"start_a_radio": "რადიოს ჩართვა",
"how_to_start_radio": "როგორ გნებავთ რადიოს ჩართვა?",
"replace_queue_question": "გნებავთ ჩაანაცვლოთ არსებული რიგი თუ დაამატოთ მასზე?",
"endless_playback": "დაუსრულებელი დაკვრა",
"delete_playlist": "ფლეილისტის წაშლა",
"delete_playlist_confirmation": "დარწმუნებული ხართ რომ გნებავთ ფლეილისტის წაშლა?",
"local_tracks": "ლოკალური ტრეკები",
"song_link": "ტრეკის ლინკი",
"skip_this_nonsense": "ამ სისულელის გამოტოვება",
"freedom_of_music": "“მუსიკის თავისუფლება”",
"freedom_of_music_palm": "“მუსიკის თავისუფლება შენს ხელის გულზე”",
"get_started": "დავიწყოთ",
"youtube_source_description": "რეკომენდებულია და მუშაობს საუკეთესოდ.",
"piped_source_description": "თავისუფლად გრძნობთ თავს? იგივეა, რაც YouTube, მაგრამ ბევრი თავისუფალი.",
"jiosaavn_source_description": "საუკეთესოა სამხრეთ აზიის რეგიონისთვის.",
"highest_quality": "საუკეთესო ხარისხი: {quality}",
"select_audio_source": "აუდიოს წყაროს არჩევა",
"endless_playback_description": "ახალი სიმთერების ავტომატურად რიგის ბოლოში დამატება",
"choose_your_region": "აირჩიე შენი რეგიონი",
"choose_your_region_description": "This will help Spotube show you the right content\nfor your location.",
"choose_your_language": "აირჩიე ენა",
"help_project_grow": "დაეხმარეთ ამ პროექტს განვითარებაში",
"help_project_grow_description": "Spotube is an open-source project. You can help this project grow by contributing to the project, reporting bugs, or suggesting new features.",
"contribute_on_github": "GitHub-ზე კონტრიბუცია",
"donate_on_open_collective": "Open Collective-ზე დონაცია",
"browse_anonymously": "ანონიმურად ნახვა",
"enable_connect": "დაკავშირების ჩართვა",
"enable_connect_description": "აკონტროლე Spotube სხვა მოწყობილობებიდან",
"devices": "მოწყობილობები",
"select": "არჩევა",
"connect_client_alert": "თქვენ კონტროლირებული ხართ {client} მოწყობილობით",
"this_device": "ეს მოწყობილობა",
"remote": "დისტანციური"
}

View File

@ -3,13 +3,13 @@
"browse": "Göz at",
"search": "Ara",
"library": "Kütüphane",
"lyrics": "Şarkı Sözleri",
"lyrics": "Şarkı sözleri",
"settings": "Ayarlar",
"genre_categories_filter": "Kategorileri veya türleri filtrele...",
"genre_categories_filter": "Kategorileri veya türleri filtreleyin...",
"genre": "Tür",
"personalized": "Kişiselleştirilmiş",
"featured": "Öne Çıkanlar",
"new_releases": "Yeni Çıkanlar",
"featured": "Öne çıkanlar",
"new_releases": "Yeni çıkanlar",
"songs": "Şarkılar",
"playing_track": "{track} oynatılıyor",
"queue_clear_alert": "Bu, mevcut kuyruğu temizleyecektir. {track_length} parça kaldırılacak\nDevam etmek istiyor musunuz?",
@ -20,15 +20,15 @@
"tracks": "Parçalar",
"downloads": "İndirilenler",
"filter_playlists": "Oynatma listelerinizi filtreleyin...",
"liked_tracks": "Beğenilen Parçalar",
"liked_tracks": "Beğenilen parçalar",
"liked_tracks_description": "Beğendiğiniz tüm parçalar",
"create_playlist": "Oynatma Listesi Oluştur",
"create_a_playlist": "Bir oynatma listesi oluşturun",
"create_playlist": "Oynatma listesi oluştur",
"create_a_playlist": "Bir oynatma listesi oluştur",
"update_playlist": "Oynatma listesini güncelle",
"create": "Oluştur",
"cancel": "İptal",
"update": "Güncelle",
"playlist_name": "Oynatma Listesi Adı",
"playlist_name": "Oynatma listesi adı",
"name_of_playlist": "Oynatma listesinin adı",
"description": "Açıklama",
"public": "Halka açık",
@ -39,16 +39,16 @@
"none": "Yok",
"sort_a_z": "A - Z'ye göre sırala",
"sort_z_a": "Z - A'ya göre sırala",
"sort_artist": "Sanatçıya Göre Sırala",
"sort_album": "Albüme Göre Sırala",
"sort_duration": "Süreye Göre Sırala",
"sort_tracks": "Parçaları Sırala",
"currently_downloading": "Şu An İndirilenler ({tracks_length})",
"cancel_all": "Tümünü İptal Et",
"filter_artist": "Sanatçıları filtrele...",
"sort_artist": "Sanatçıya göre sırala",
"sort_album": "Albüme göre sırala",
"sort_duration": "Süreye göre sırala",
"sort_tracks": "Parçaları sırala",
"currently_downloading": "Şu anda indirilenler ({tracks_length})",
"cancel_all": "Tümünü iptal et",
"filter_artist": "Sanatçıları filtreleyin...",
"followers": "{followers} Takipçiler",
"add_artist_to_blacklist": "Sanatçıyı kara listeye ekle",
"top_tracks": "En İyi Parçalar",
"top_tracks": "En iyi parçalar",
"fans_also_like": "Hayranlar ayrıca şunları da beğendi",
"loading": "Yükleniyor...",
"artist": "Sanatçı",
@ -57,7 +57,7 @@
"follow": "Takip et",
"artist_url_copied": "Sanatçı bağlantısı panoya kopyalandı",
"added_to_queue": "Kuyruğa {tracks} parçası eklendi",
"filter_albums": "Albümleri filtrele...",
"filter_albums": "Albümleri filtreleyin...",
"synced": "Senkronize edildi",
"plain": "Sade",
"shuffle": "Karıştır",
@ -68,19 +68,19 @@
"time": "Zaman",
"more_actions": "Daha fazla eylem",
"download_count": "İndir ({count})",
"add_count_to_playlist": "Oynatma Listesine ({count}) ekle",
"add_count_to_queue": "Kuyruğa ({count}) ekle",
"play_count_next": "({count}) sonrakini oynat",
"add_count_to_playlist": "Oynatma Listesine ekle ({count})",
"add_count_to_queue": "Kuyruğa ekle ({count})",
"play_count_next": "Sonrakini oynat ({count})",
"album": "Albüm",
"copied_to_clipboard": "{data} panoya kopyalandı",
"add_to_following_playlists": "{track} parçasını aşağıdaki Oynatma Listelerine ekle",
"add_to_following_playlists": "{track} parçasını aşağıdaki oynatma listelerine ekle",
"add": "Ekle",
"added_track_to_queue": "{track} kuyruğa eklendi",
"add_to_queue": "Kuyruğa ekle",
"track_will_play_next": "{track} bir sonraki çalacak",
"play_next": "Sonrakini oynat",
"removed_track_from_queue": "{track} sıradan kaldırıldı",
"remove_from_queue": "Sıradan kaldır",
"removed_track_from_queue": "{track} kuyruktan kaldırıldı",
"remove_from_queue": "Kuyruktan kaldır",
"remove_from_favorites": "Favorilerden kaldır",
"save_as_favorite": "Favori olarak kaydet",
"add_to_playlist": "Oynatma listesine ekle",
@ -88,7 +88,7 @@
"add_to_blacklist": "Kara listeye ekle",
"remove_from_blacklist": "Kara listeden kaldır",
"share": "Paylaş",
"mini_player": "Mini Oynatıcı",
"mini_player": "Mini oynatıcı",
"slide_to_seek": "İleri veya geri arama yapmak için kaydırın",
"shuffle_playlist": "Oynatma listesini karıştır",
"unshuffle_playlist": "Oynatma listesinin karışıklığını kaldır",
@ -98,27 +98,27 @@
"resume_playback": "Oynatmayı sürdür",
"loop_track": "Döngü parçası",
"repeat_playlist": "Oynatma listesini tekrarla",
"queue": "Sıra",
"alternative_track_sources": "Alternatif yol kaynakları",
"queue": "Kuyruk",
"alternative_track_sources": "Alternatif parça kaynakları",
"download_track": "Parçayı indir",
"tracks_in_queue": "{tracks} parça sırada",
"tracks_in_queue": "{tracks} parça kuyrukta",
"clear_all": "Tümünü temizle",
"show_hide_ui_on_hover": "Fareyle üzerine gelindiğinde kullanıcı arayüzünü göster/gizle",
"always_on_top": "Her zaman üstte",
"exit_mini_player": "Mini oynatıcıdan çık",
"download_location": "İndirme konumu",
"account": "Hesap",
"login_with_spotify": "Spotify hesabınızla giriş yapın",
"login_with_spotify": "Spotify hesabı ile giriş yap",
"connect_with_spotify": "Spotify ile bağlan",
"logout": "Çıkış Yap",
"logout_of_this_account": "Bu hesaptan çıkış yap",
"language_region": "Dil ve Bölge",
"language": "Dil",
"system_default": "Sistem Varsayılanı",
"market_place_region": "Pazaryeri Bölgesi",
"recommendation_country": "Tavsiye Edilen Ülke",
"logout": "Çıkış yap",
"logout_of_this_account": "Hesaptan çıkış yap",
"language_region": "Dil ve bölge",
"language": "Tercih edilen dil",
"system_default": "Sistem varsayılanı",
"market_place_region": "Tercih edilen bölge",
"recommendation_country": "Tavsiye edilen ülke",
"appearance": "Görünüm",
"layout_mode": "Düzen Modu",
"layout_mode": "Düzen modu",
"override_layout_settings": "Duyarlı düzen modu ayarlarını geçersiz kıl",
"adaptive": "Uyarlanabilir",
"compact": "Sıkıştırılmış",
@ -127,35 +127,35 @@
"dark": "Koyu",
"light": "Açık",
"system": "Sistem",
"accent_color": "Vurgu Rengi",
"accent_color": "Vurgu rengi",
"sync_album_color": "Albüm rengini senkronize et",
"sync_album_color_description": "Vurgu rengi olarak albüm resminin baskın rengini kullanır",
"playback": "Oynatma",
"audio_quality": "Ses Kalitesi",
"audio_quality": "Ses kalitesi",
"high": "Yüksek",
"low": "Düşük",
"pre_download_play": "Ön yükleme ve oynatma",
"pre_download_play_description": "Ses akışı yerine baytları indirin ve oynatın (Daha yüksek bant genişliğine sahip kullanıcılar için önerilir)",
"skip_non_music": "Müzik olmayan bölümleri atla (SponsorBlock)",
"pre_download_play": "Önceden indir ve oynat",
"pre_download_play_description": "Ses akışı yerine baytları indir ve oynat (Daha yüksek bant genişliğine sahip kullanıcılar için önerilir)",
"skip_non_music": "Müzik olmayan bölümleri atlat (SponsorBlock)",
"blacklist_description": "Kara listeye alınan parçalar ve sanatçılar",
"wait_for_download_to_finish": "Lütfen mevcut indirme işleminin tamamlanmasını bekleyin",
"desktop": "Masaüstü",
"close_behavior": "Kapatma Davranışı",
"close_behavior": "Kapatma davranışı",
"close": "Kapat",
"minimize_to_tray": "Tepsiye küçült",
"show_tray_icon": "Sistem tepsisi simgesini göster",
"about": "Hakkında",
"u_love_spotube": "Spotube'u sevdiğinizi biliyoruz",
"check_for_updates": "Güncellemeleri kontrol et",
"about_spotube": "Spotube Hakkında",
"about_spotube": "Spotube hakkında",
"blacklist": "Kara liste",
"please_sponsor": "Sponsor Ol/Bağış Yap",
"spotube_description": "Spotube, hafif, platformlar arası, herkes için ücretsiz bir spotify istemcisidir",
"spotube_description": "Spotube, hafif, platformlar arası uyumlu ve herkes için ücretsiz bir Spotify istemcisidir.",
"version": "Sürüm",
"build_number": "Derleme Numarası",
"founder": "Kurucu",
"build_number": "Derleme numarası",
"founder": "Geliştirici",
"repository": "Depo",
"bug_issues": "Hata+Sorunlar",
"bug_issues": "Hata + Sorunlar",
"made_with": "❤️ ile Bangladeş'te yapıldı",
"kingkor_roy_tirtho": "Kingkor Roy Tirtho",
"copyright": "© 2021-{current_year} Kingkor Roy Tirtho",
@ -163,31 +163,31 @@
"add_spotify_credentials": "Başlamak için spotify kimlik bilgilerinizi ekleyin",
"credentials_will_not_be_shared_disclaimer": "Endişelenmeyin, kimlik bilgilerinizden hiçbiri toplanmayacak veya kimseyle paylaşılmayacak",
"know_how_to_login": "Bunu nasıl yapacağınızı bilmiyor musunuz?",
"follow_step_by_step_guide": "Adım Adım kılavuzu takip edin",
"spotify_cookie": "Spotify {name} Çerezi",
"cookie_name_cookie": "{name} Çerezi",
"follow_step_by_step_guide": "Adım adım kılavuzu takip edin",
"spotify_cookie": "Spotify {name} çerezi",
"cookie_name_cookie": "{name} çerezi",
"fill_in_all_fields": "Lütfen tüm alanları doldurun",
"submit": "Gönder",
"submit": "Başvur",
"exit": "Çık",
"previous": "Önceki",
"next": "Sonraki",
"done": "Bitti",
"step_1": "1. Adım",
"first_go_to": "İlk olarak şuraya gidin:",
"login_if_not_logged_in": "ve oturum açmadıysanız Oturum Açın/Kaydolun",
"login_if_not_logged_in": "ve oturum açmadıysanız Oturum açın/Kaydolun",
"step_2": "2. Adım",
"step_2_steps": "1. Giriş yaptıktan sonra, Tarayıcı geliştirme araçlarını açmak için F12 veya Fareye Sağ Tıklayın > İncele'ye basın.\n2. Ardından \"Uygulama\" Sekmesine (Chrome, Edge, Brave vb.) veya \"Depolama\" Sekmesine (Firefox, Palemoon vb.) gidin.\n3. \"Çerezler\" bölümüne ve ardından \"https://accounts.spotify.com\" alt bölümüne gidin",
"step_2_steps": "1. Oturum açtıktan sonra, tarayıcı geliştirme araçlarını açmak için F12'ye veya fareye sağ tıklayın > İncele'ye basın.\n2. Daha sonra \"Uygulama\" sekmesine (Chrome, Edge, Brave vb..) veya \"Depolama\" sekmesine (Firefox, Palemoon vb..) gidin\n3. \"Çerezler\" bölümüne, ardından \"https://accounts.spotify.com\" alt bölümüne gidin",
"step_3": "3. Adım",
"step_3_steps": "\"sp_dc\" Çerezinin değerini kopyalayın",
"success_emoji": "Başarılı🥳",
"success_message": "Artık Spotify hesabınızla başarıyla giriş yaptınız. Aferin, dostum!",
"success_message": "Artık Spotify hesabınızla başarıyla giriş yaptınız. Tebrik ederim!",
"step_4": "4. Adım",
"step_4_steps": "Kopyalanan \"sp_dc\" değerini yapıştırın",
"something_went_wrong": "Bir hata oluştu",
"piped_instance": "Piped Sunucu Örneği",
"piped_instance": "Piped sunucu örneği",
"piped_description": "Parça eşleştirme için kullanılacak Piped sunucu örneği",
"piped_warning": "Bazıları iyi çalışmayabilir. Yani riski size ait olmak üzere kullanın",
"generate_playlist": "Oynatma Listesi Oluştur",
"generate_playlist": "Oynatma listesi oluştur",
"track_exists": "{track} parçası zaten var",
"replace_downloaded_tracks": "İndirilen tüm parçaları değiştir",
"skip_download_tracks": "İndirilen tüm parçaları indirmeyi atla",
@ -195,8 +195,8 @@
"replace": "Değiştir",
"skip": "Atla",
"select_up_to_count_type": "En fazla {count} {type} seçin",
"select_genres": "Türleri Seç",
"add_genres": "Tür Ekle",
"select_genres": "Türleri seç",
"add_genres": "Tür ekle",
"country": "Ülke",
"number_of_tracks_generate": "Oluşturulacak parça sayısı",
"acousticness": "Akustiklik",
@ -212,7 +212,7 @@
"duration": "Süre (sn)",
"tempo": "Tempo (BPM)",
"mode": "Mod",
"time_signature": "Zaman İmzası",
"time_signature": "Zaman imzası",
"short": "Kısa",
"medium": "Orta",
"long": "Uzun",
@ -220,29 +220,29 @@
"max": "Maks",
"target": "Hedef",
"moderate": "Orta",
"deselect_all": "Tüm Seçimleri Kaldır",
"select_all": "Tümünü Seç",
"deselect_all": "Tüm seçimleri kaldır",
"select_all": "Tümünü seç",
"are_you_sure": "Emin misiniz?",
"generating_playlist": "Özel oynatma listeniz oluşturuluyor...",
"selected_count_tracks": "{count} parça seçildi",
"download_warning": "Tüm Parçaları toplu olarak indirirseniz, açıkça Müzik korsanlığı yapıyor ve Müziğin yaratıcı toplumuna zarar veriyorsunuz demektir. Umarım bunun farkındasınızdır. Her zaman, Sanatçının sıkı çalışmasına saygı duymaya ve desteklemeye çalışın",
"download_ip_ban_warning": "Bu arada, IP'niz normalden daha fazla indirme isteği nedeniyle YouTube'da engellenebilir. IP engelleme, YouTube'u (oturum açmış olsanız bile) o IP cihazından en az 2 -3 ay kullanamayacağınız anlamına gelir. Ve eğer böyle bir şey olursa Spotube'un hiçbir sorumluluğu yok",
"download_warning": "Tüm şarkıları toplu olarak indiriyorsanız, açıkça müzik korsanlığı yapıyorsunuz ve müzik dünyasının yaratıcı topluluğuna zarar veriyorsunuz demektir. Umuyorum bunun farkındasınızdır. Her zaman, sanatçıların emeğine saygı göstermeyi ve desteklemeyi deneyin.",
"download_ip_ban_warning": "Ayrıca, normalden fazla indirme istekleri nedeniyle YouTube'da IP'niz engellenebilir. IP engeli, en az 2-3 ay boyunca YouTube'u (hatta oturum açmış olsanız bile) o IP cihazından kullanamayacağınız anlamına gelir. Ve eğer böyle bir durum yaşanırsa, Spotube bundan hiçbir sorumluluk kabul etmez.",
"by_clicking_accept_terms": "\"Kabul et\" e tıklayarak aşağıdaki şartları kabul etmiş olursunuz:",
"download_agreement_1": "Müzik korsanlığı yaptığımı biliyorum. Ben kötüyüm",
"download_agreement_1": "Müzik korsanlığı yaptığımı biliyorum. Ben fakir biriyim.",
"download_agreement_2": "Sanatçıyı elimden geldiğince destekleyeceğim ve bunu sadece sanatını satın alacak param olmadığı için yapıyorum",
"download_agreement_3": "IP adresimin YouTube'da engellenebileceğinin tamamen farkındayım ve mevcut işlemimden kaynaklanan herhangi bir kazadan Spotube'u veya sahiplerini/katkıda bulunanlarını sorumlu tutmuyorum",
"download_agreement_3": "YouTube'da IP'min engellenebileceğinin tamamen farkındayım ve mevcut eylemlerimden kaynaklanan herhangi bir kaza için Spotube'u veya sahiplerini/katkıda bulunanları sorumlu tutmuyorum.",
"decline": "Reddet",
"accept": "Kabul et",
"details": "Detaylar",
"youtube": "YouTube",
"channel": "Kanal",
"likes": "Beğeniler",
"likes": "Beğenenler",
"dislikes": "Beğenmeyenler",
"views": "İzlenmeler",
"streamUrl": "Akış bağlantısı",
"stop": "Durdur",
"sort_newest": "En yeniye göre sırala",
"sort_oldest": "Eklenen en eskiye göre sırala",
"sort_newest": "En yeni eklenene göre sırala.",
"sort_oldest": "En eski eklenene göre sırala",
"sleep_timer": "Uyku Zamanlayıcısı",
"mins": "{minutes} Dakika",
"hours": "{hours} Saatler",
@ -251,11 +251,11 @@
"logs": "Günlükler",
"developers": "Geliştiriciler",
"not_logged_in": "Giriş yapmadınız",
"search_mode": "Arama Modu",
"audio_source": "Ses Kaynağı",
"search_mode": "Arama modu",
"audio_source": "Ses kaynağı",
"ok": "Tamam",
"failed_to_encrypt": "Şifreleme başarısız oldu",
"encryption_failed_warning": "Spotube, verilerinizi güvenli bir şekilde saklamak için şifreleme kullanır. Ama başaramadı. Bu yüzden güvensiz depolamaya geri dönecek\nLinux kullanıyorsanız, lütfen herhangi bir gizli servisin (gnome - anahtarlık, kde - cüzdan, keepassxc vb.) yüklü olduğundan emin olun",
"encryption_failed_warning": "Spotube, verilerinizi güvenli bir şekilde depolamak için şifreleme kullanır. Ancak bunu başaramadı. Bu nedenle, güvensiz depolamaya geri dönecektir\nLinux kullanıyorsanız, lütfen gnome-keyring, kde-wallet, keepassxc vb. herhangi bir gizli servisin yüklü olduğundan emin olun.",
"querying_info": "Bilgi sorgulanıyor...",
"piped_api_down": "Piped API kapalı",
"piped_down_error_instructions": "Piped örneği {pipedInstance} şu anda kapalı\n\nÖrneği değiştirin veya 'API türünü' resmi YouTube API'si olarak değiştirin\n\nDeğişiklikten sonra uygulamayı yeniden başlattığınızdan emin olun",
@ -263,8 +263,8 @@
"connection_restored": "İnternet bağlantınız geri yüklendi",
"use_system_title_bar": "Sistem başlık çubuğunu kullan",
"crunching_results": "Sonuçlar...",
"search_to_get_results": "Sonuç almak için ara",
"use_amoled_mode": "AMOLED Modunu Kullan",
"search_to_get_results": "Sonuç almak için arayın",
"use_amoled_mode": "AMOLED modu kullan",
"pitch_dark_theme": "Zifiri karanlık koyu tema",
"normalize_audio": "Sesi normalleştir",
"change_cover": "Kapağı değiştir",
@ -277,48 +277,48 @@
"disconnect_lastfm": "Last.fm bağlantısını kes",
"disconnect": "Bağlantıyı kes",
"username": "Kullanıcı adı",
"password": "Parola",
"login": "Giriş",
"password": "Şifre",
"login": "Giriş yap",
"login_with_your_lastfm": "Last.fm hesabınızla giriş yapın",
"scrobble_to_lastfm": "Last.fm için Scrobble",
"go_to_album": "Albüme Git",
"discord_rich_presence": "Discord Zengin Varlığı",
"browse_all": "Tümüne Göz At",
"genres": "Müzik Türleri",
"explore_genres": "Türleri Keşfet",
"go_to_album": "Albüme git",
"discord_rich_presence": "Discord zengin varlığı",
"browse_all": "Tümüne göz at",
"genres": "Müzik türleri",
"explore_genres": "Türleri keşfet",
"friends": "Arkadaşlar",
"no_lyrics_available": "Üzgünüz, bu parçanın sözleri bulunamıyor",
"start_a_radio": "Radyo Başlat",
"start_a_radio": "Radyo başlat",
"how_to_start_radio": "Radyoyu nasıl başlatmak istersiniz?",
"replace_queue_question": "Mevcut kuyruğu değiştirmek mi yoksa eklemek mi istersiniz?",
"endless_playback": "Sonsuz Olarak Oynat",
"delete_playlist": "Oynatma Listesini Sil",
"endless_playback": "Sonsuz olarak oynat",
"delete_playlist": "Oynatma listesini sil",
"delete_playlist_confirmation": "Bu oynatma listesini silmek istediğinizden emin misiniz?",
"local_tracks": "Yerel Parçalar",
"song_link": "Şarkı Bağlantısı",
"local_tracks": "Yerel parçalar",
"song_link": "Şarkı bağlantısı",
"skip_this_nonsense": "Bu saçmalığı atla",
"freedom_of_music": "“Müzik Özgürlüğü”",
"freedom_of_music_palm": "“Müzik Özgürlüğü avucunuzun içinde”",
"freedom_of_music": "“Müzik özgürlüğü”",
"freedom_of_music_palm": "“Müzik özgürlüğü avucunuzun içinde”",
"get_started": "Haydi başlayalım",
"youtube_source_description": "Tavsiye edilir ve en iyi şekilde çalışır.",
"piped_source_description": "Özgür hissediyor musunuz? YouTube ile aynı ama çok daha fazla ücretsiz.",
"piped_source_description": "Özgür hissediyor musunuz? YouTube ile aynı, ama çok daha özgür.",
"jiosaavn_source_description": "Güney Asya bölgesi için en iyisi.",
"highest_quality": "En Yüksek Kalite: {quality}",
"select_audio_source": "Ses Kaynağını Seç",
"endless_playback_description": "Yeni şarkıları otomatik olarak \nkuyruğun sonuna ekle",
"highest_quality": "En yüksek kalite: {quality}",
"select_audio_source": "Ses kaynağını seçin",
"endless_playback_description": "Yeni şarkıları otomatik olarak\nkuyruğun sonuna ekle",
"choose_your_region": "Bölgenizi seçin",
"choose_your_region_description": "Bu, Spotube'un size doğru içeriği göstermesine yardımcı olacaktır\nkonumunuz için.",
"choose_your_region_description": "Bu, Spotube'un konumunuza uygun içerikleri göstermesine yardımcı olacaktır.",
"choose_your_language": "Dilinizi seçin",
"help_project_grow": "Bu projenin büyümesine yardımcı ol",
"help_project_grow": "Bu projenin büyümesine yardımcı olun",
"help_project_grow_description": "Spotube açık kaynaklı bir projedir. Projeye katkıda bulunarak, hataları bildirerek veya yeni özellikler önererek bu projenin büyümesine yardımcı olabilirsiniz.",
"contribute_on_github": "GitHub'a katkıda bulunun",
"donate_on_open_collective": "Open Collective'e bağış yap",
"browse_anonymously": "Anonim Olarak Göz at",
"enable_connect": "Bağlantıyı Etkinleştir",
"contribute_on_github": "GitHub'da katkıda bulun",
"donate_on_open_collective": "Open Collective'de bağış yap",
"browse_anonymously": "Anonim olarak giriş yap",
"enable_connect": "Bağlanmayı etkinleştir",
"enable_connect_description": "Spotube'u diğer cihazlardan kontrol edin",
"devices": "Cihazlar",
"select": "Seç",
"connect_client_alert": "{client} tarafından kontrol ediliyorsun.",
"this_device": "Bu Cihaz",
"this_device": "Bu cihaz",
"remote": "Yönet"
}

View File

@ -7,7 +7,7 @@
/// TexturedPolak@github => Polish
/// yuri-val@github => Ukrainian
/// energywave@github, ncvescera@github, OpenCode@github => Italian
/// mdksec@github, mikropsoft@github => Turkish
/// mikropsoft@github => Turkish
/// Stephan-P@github, SecularSteve@github => Dutch
/// doannc2212@github => Vietnamese
/// sappho192@github => Korean
@ -28,11 +28,14 @@ class L10n {
const Locale('de', 'GE'),
const Locale('es', 'ES'),
const Locale('fa', 'IR'),
const Locale('fi', 'FI'),
const Locale('fr', 'FR'),
const Locale('ne', 'NP'),
const Locale('hi', 'IN'),
const Locale('id', 'ID'),
const Locale('it', 'IT'),
const Locale('ja', 'JP'),
const Locale('ka', 'GE'),
const Locale('ko', 'KR'),
const Locale('nl', 'NL'),
const Locale('pl', 'PL'),
@ -43,5 +46,6 @@ class L10n {
const Locale('tr', 'TR'),
const Locale('zh', 'CN'),
const Locale('vi', 'VN'),
const Locale('eu', 'ES'),
];
}

View File

@ -29,7 +29,7 @@ class LibraryPage extends HookConsumerWidget {
leading: ThemedButtonsTabBar(
tabs: [
Tab(text: " ${context.l10n.playlists} "),
Tab(text: " ${context.l10n.local_tracks} "),
Tab(text: " ${context.l10n.local_tab} "),
Tab(
child: Badge(
isLabelVisible: downloadingCount > 0,

View File

@ -0,0 +1,240 @@
import 'package:collection/collection.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:fuzzywuzzy/fuzzywuzzy.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:skeletonizer/skeletonizer.dart';
import 'package:spotube/collections/fake.dart';
import 'package:spotube/collections/spotube_icons.dart';
import 'package:spotube/components/library/user_local_tracks.dart';
import 'package:spotube/components/shared/expandable_search/expandable_search.dart';
import 'package:spotube/components/shared/fallbacks/not_found.dart';
import 'package:spotube/components/shared/inter_scrollbar/inter_scrollbar.dart';
import 'package:spotube/components/shared/page_window_title_bar.dart';
import 'package:spotube/components/shared/sort_tracks_dropdown.dart';
import 'package:spotube/components/shared/track_tile/track_tile.dart';
import 'package:spotube/extensions/artist_simple.dart';
import 'package:spotube/extensions/context.dart';
import 'package:spotube/models/local_track.dart';
import 'package:spotube/provider/local_tracks/local_tracks_provider.dart';
import 'package:spotube/provider/proxy_playlist/proxy_playlist_provider.dart';
import 'package:spotube/utils/service_utils.dart';
class LocalLibraryPage extends HookConsumerWidget {
static const name = "local_library_page";
final String location;
final bool isDownloads;
const LocalLibraryPage(this.location, {super.key, this.isDownloads = false});
Future<void> playLocalTracks(
WidgetRef ref,
List<LocalTrack> tracks, {
LocalTrack? currentTrack,
}) async {
final playlist = ref.read(proxyPlaylistProvider);
final playback = ref.read(proxyPlaylistProvider.notifier);
currentTrack ??= tracks.first;
final isPlaylistPlaying = playlist.containsTracks(tracks);
if (!isPlaylistPlaying) {
await playback.load(
tracks,
initialIndex: tracks.indexWhere((s) => s.id == currentTrack?.id),
autoPlay: true,
);
} else if (isPlaylistPlaying &&
currentTrack.id != null &&
currentTrack.id != playlist.activeTrack?.id) {
await playback.jumpToTrack(currentTrack);
}
}
@override
Widget build(BuildContext context, ref) {
final sortBy = useState<SortBy>(SortBy.none);
final playlist = ref.watch(proxyPlaylistProvider);
final trackSnapshot = ref.watch(localTracksProvider);
final isPlaylistPlaying = playlist.containsTracks(
trackSnapshot.asData?.value.values.flattened.toList() ?? []);
final searchController = useTextEditingController();
useValueListenable(searchController);
final searchFocus = useFocusNode();
final isFiltering = useState(false);
final controller = useScrollController();
return SafeArea(
bottom: false,
child: Scaffold(
appBar: PageWindowTitleBar(
leading: const BackButton(),
centerTitle: true,
title: Text(isDownloads ? context.l10n.downloads : location),
backgroundColor: Colors.transparent,
),
body: Column(
children: [
Padding(
padding: const EdgeInsets.all(8.0),
child: Row(
children: [
const SizedBox(width: 5),
FilledButton(
onPressed: trackSnapshot.asData?.value != null
? () async {
if (trackSnapshot.asData?.value.isNotEmpty ==
true) {
if (!isPlaylistPlaying) {
await playLocalTracks(
ref,
trackSnapshot.asData!.value[location] ?? [],
);
}
}
}
: null,
child: Row(
children: [
Text(context.l10n.play),
Icon(
isPlaylistPlaying
? SpotubeIcons.stop
: SpotubeIcons.play,
)
],
),
),
const Spacer(),
ExpandableSearchButton(
isFiltering: isFiltering.value,
onPressed: (value) => isFiltering.value = value,
searchFocus: searchFocus,
),
const SizedBox(width: 10),
SortTracksDropdown(
value: sortBy.value,
onChanged: (value) {
sortBy.value = value;
},
),
const SizedBox(width: 5),
FilledButton(
child: const Icon(SpotubeIcons.refresh),
onPressed: () {
ref.invalidate(localTracksProvider);
},
)
],
),
),
ExpandableSearchField(
searchController: searchController,
searchFocus: searchFocus,
isFiltering: isFiltering.value,
onChangeFiltering: (value) => isFiltering.value = value,
),
trackSnapshot.when(
data: (tracks) {
final sortedTracks = useMemoized(() {
return ServiceUtils.sortTracks(
tracks[location] ?? <LocalTrack>[], sortBy.value);
}, [sortBy.value, tracks]);
final filteredTracks = useMemoized(() {
if (searchController.text.isEmpty) {
return sortedTracks;
}
return sortedTracks
.map((e) => (
weightedRatio(
"${e.name} - ${e.artists?.asString() ?? ""}",
searchController.text,
),
e,
))
.toList()
.sorted(
(a, b) => b.$1.compareTo(a.$1),
)
.where((e) => e.$1 > 50)
.map((e) => e.$2)
.toList()
.toList();
}, [searchController.text, sortedTracks]);
if (!trackSnapshot.isLoading && filteredTracks.isEmpty) {
return const Expanded(
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [NotFound()],
),
);
}
return Expanded(
child: RefreshIndicator(
onRefresh: () async {
ref.invalidate(localTracksProvider);
},
child: InterScrollbar(
controller: controller,
child: Skeletonizer(
enabled: trackSnapshot.isLoading,
child: ListView.builder(
controller: controller,
physics: const AlwaysScrollableScrollPhysics(),
itemCount: trackSnapshot.isLoading
? 5
: filteredTracks.length,
itemBuilder: (context, index) {
if (trackSnapshot.isLoading) {
return TrackTile(
playlist: playlist,
track: FakeData.track,
index: index,
);
}
final track = filteredTracks[index];
return TrackTile(
index: index,
playlist: playlist,
track: track,
userPlaylist: false,
onTap: () async {
await playLocalTracks(
ref,
sortedTracks,
currentTrack: track,
);
},
);
},
),
),
),
),
);
},
loading: () => Expanded(
child: Skeletonizer(
enabled: true,
child: ListView.builder(
itemCount: 5,
itemBuilder: (context, index) => TrackTile(
track: FakeData.track,
index: index,
playlist: playlist,
),
),
),
),
error: (error, stackTrace) =>
Text(error.toString() + stackTrace.toString()),
)
],
)),
);
}
}

View File

@ -3,6 +3,7 @@ import 'package:file_selector/file_selector.dart';
import 'package:flutter/material.dart';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:go_router/go_router.dart';
import 'package:spotube/collections/spotube_icons.dart';
import 'package:spotube/components/settings/section_card_with_heading.dart';
import 'package:spotube/extensions/context.dart';

View File

@ -0,0 +1,125 @@
import 'dart:io';
import 'package:catcher_2/catcher_2.dart';
import 'package:flutter/foundation.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:metadata_god/metadata_god.dart';
import 'package:mime/mime.dart';
import 'package:path/path.dart';
import 'package:path_provider/path_provider.dart';
import 'package:spotify/spotify.dart';
import 'package:spotube/extensions/track.dart';
import 'package:spotube/models/local_track.dart';
import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
// ignore: depend_on_referenced_packages
import 'package:flutter_rust_bridge/flutter_rust_bridge.dart' show FfiException;
const supportedAudioTypes = [
"audio/webm",
"audio/ogg",
"audio/mpeg",
"audio/mp4",
"audio/opus",
"audio/wav",
"audio/aac",
];
const imgMimeToExt = {
"image/png": ".png",
"image/jpeg": ".jpg",
"image/webp": ".webp",
"image/gif": ".gif",
};
final localTracksProvider =
FutureProvider<Map<String, List<LocalTrack>>>((ref) async {
try {
if (kIsWeb) return {};
final Map<String, List<LocalTrack>> tracks = {};
final downloadLocation = ref.watch(
userPreferencesProvider.select((s) => s.downloadLocation),
);
final downloadDir = Directory(downloadLocation);
if (!await downloadDir.exists()) {
await downloadDir.create(recursive: true);
}
final localLibraryLocations = ref.watch(
userPreferencesProvider.select((s) => s.localLibraryLocation),
);
for (var location in [downloadLocation, ...localLibraryLocations]) {
if (location.isEmpty) continue;
final entities = <FileSystemEntity>[];
if (await Directory(location).exists()) {
try {
entities.addAll(Directory(location).listSync(recursive: true));
} catch (e, stack) {
Catcher2.reportCheckedError(e, stack);
}
}
final filesWithMetadata = (await Future.wait(
entities.map((e) => File(e.path)).where((file) {
final mimetype = lookupMimeType(file.path);
return mimetype != null && supportedAudioTypes.contains(mimetype);
}).map(
(file) async {
try {
final metadata = await MetadataGod.readMetadata(file: file.path);
final imageFile = File(join(
(await getTemporaryDirectory()).path,
"spotube",
basenameWithoutExtension(file.path) +
imgMimeToExt[metadata.picture?.mimeType ?? "image/jpeg"]!,
));
if (!await imageFile.exists() && metadata.picture != null) {
await imageFile.create(recursive: true);
await imageFile.writeAsBytes(
metadata.picture?.data ?? [],
mode: FileMode.writeOnly,
);
}
return {
"metadata": metadata,
"file": file,
"art": imageFile.path
};
} catch (e, stack) {
if (e is FfiException) {
return {"file": file};
}
Catcher2.reportCheckedError(e, stack);
return {};
}
},
),
))
.where((e) => e.isNotEmpty)
.toList();
// ignore: no_leading_underscores_for_local_identifiers
final _tracks = filesWithMetadata
.map(
(fileWithMetadata) => LocalTrack.fromTrack(
track: Track().fromFile(
fileWithMetadata["file"],
metadata: fileWithMetadata["metadata"],
art: fileWithMetadata["art"],
),
path: fileWithMetadata["file"].path,
),
)
.toList();
tracks[location] = _tracks;
}
return tracks;
} catch (e, stack) {
Catcher2.reportCheckedError(e, stack);
return {};
}
});

View File

@ -1,4 +1,4 @@
// ignore_for_file: invalid_use_of_protected_member
// ignore_for_file: invalid_use_of_protected_member, invalid_use_of_visible_for_testing_member
import 'dart:async';

View File

@ -44,7 +44,14 @@ class ProxyPlaylist {
}
bool containsTrack(TrackSimple track) {
return tracks.firstWhereOrNull((element) => element.id == track.id) != null;
return tracks.firstWhereOrNull((element) {
if (element is LocalTrack && track is LocalTrack) {
return element.path == track.path;
}
return element.id == track.id;
}) !=
null;
}
bool containsTracks(Iterable<TrackSimple> tracks) {
@ -63,9 +70,11 @@ class ProxyPlaylist {
/// To make sure proper instance method is used for JSON serialization
/// Otherwise default super.toJson() is used
static Map<String, dynamic> _makeAppropriateTrackJson(Track track) {
return switch (track.runtimeType) {
LocalTrack() => track.toJson(),
SourcedTrack() => track.toJson(),
return switch (track) {
// ignore: unnecessary_cast
LocalTrack() => (track as LocalTrack).toJson(),
// ignore: unnecessary_cast
SourcedTrack() => (track as SourcedTrack).toJson(),
_ => track.toJson(),
};
}

View File

@ -127,7 +127,7 @@ class SyncedLyricsNotifier extends FamilyAsyncNotifier<SubtitleSimple, Track?>
final token = await spotify.getCredentials();
SubtitleSimple lyrics = await getSpotifyLyrics(token.accessToken);
if (lyrics.lyrics.isEmpty) {
if (lyrics.lyrics.isEmpty || lyrics.lyrics.length <= 5) {
lyrics = await getLRCLibLyrics();
}

View File

@ -70,6 +70,11 @@ class UserPreferencesNotifier extends PersistedStateNotifier<UserPreferences> {
state = state.copyWith(downloadLocation: downloadDir);
}
void setLocalLibraryLocation(List<String> localLibraryDirs) {
//if (localLibraryDir.isEmpty) return;
state = state.copyWith(localLibraryLocation: localLibraryDirs);
}
void setLayoutMode(LayoutMode mode) {
state = state.copyWith(layoutMode: mode);
}

View File

@ -84,6 +84,7 @@ class UserPreferences with _$UserPreferences {
@Default(Market.US) Market recommendationMarket,
@Default(SearchMode.youtube) SearchMode searchMode,
@Default("") String downloadLocation,
@Default([]) List<String> localLibraryLocation,
@Default("https://pipedapi.kavin.rocks") String pipedInstance,
@Default(ThemeMode.system) ThemeMode themeMode,
@Default(AudioSource.youtube) AudioSource audioSource,

View File

@ -43,6 +43,7 @@ mixin _$UserPreferences {
Market get recommendationMarket => throw _privateConstructorUsedError;
SearchMode get searchMode => throw _privateConstructorUsedError;
String get downloadLocation => throw _privateConstructorUsedError;
List<String> get localLibraryLocation => throw _privateConstructorUsedError;
String get pipedInstance => throw _privateConstructorUsedError;
ThemeMode get themeMode => throw _privateConstructorUsedError;
AudioSource get audioSource => throw _privateConstructorUsedError;
@ -88,6 +89,7 @@ abstract class $UserPreferencesCopyWith<$Res> {
Market recommendationMarket,
SearchMode searchMode,
String downloadLocation,
List<String> localLibraryLocation,
String pipedInstance,
ThemeMode themeMode,
AudioSource audioSource,
@ -126,6 +128,7 @@ class _$UserPreferencesCopyWithImpl<$Res, $Val extends UserPreferences>
Object? recommendationMarket = null,
Object? searchMode = null,
Object? downloadLocation = null,
Object? localLibraryLocation = null,
Object? pipedInstance = null,
Object? themeMode = null,
Object? audioSource = null,
@ -196,6 +199,10 @@ class _$UserPreferencesCopyWithImpl<$Res, $Val extends UserPreferences>
? _value.downloadLocation
: downloadLocation // ignore: cast_nullable_to_non_nullable
as String,
localLibraryLocation: null == localLibraryLocation
? _value.localLibraryLocation
: localLibraryLocation // ignore: cast_nullable_to_non_nullable
as List<String>,
pipedInstance: null == pipedInstance
? _value.pipedInstance
: pipedInstance // ignore: cast_nullable_to_non_nullable
@ -264,6 +271,7 @@ abstract class _$$UserPreferencesImplCopyWith<$Res>
Market recommendationMarket,
SearchMode searchMode,
String downloadLocation,
List<String> localLibraryLocation,
String pipedInstance,
ThemeMode themeMode,
AudioSource audioSource,
@ -300,6 +308,7 @@ class __$$UserPreferencesImplCopyWithImpl<$Res>
Object? recommendationMarket = null,
Object? searchMode = null,
Object? downloadLocation = null,
Object? localLibraryLocation = null,
Object? pipedInstance = null,
Object? themeMode = null,
Object? audioSource = null,
@ -370,6 +379,10 @@ class __$$UserPreferencesImplCopyWithImpl<$Res>
? _value.downloadLocation
: downloadLocation // ignore: cast_nullable_to_non_nullable
as String,
localLibraryLocation: null == localLibraryLocation
? _value._localLibraryLocation
: localLibraryLocation // ignore: cast_nullable_to_non_nullable
as List<String>,
pipedInstance: null == pipedInstance
? _value.pipedInstance
: pipedInstance // ignore: cast_nullable_to_non_nullable
@ -433,6 +446,7 @@ class _$UserPreferencesImpl implements _UserPreferences {
this.recommendationMarket = Market.US,
this.searchMode = SearchMode.youtube,
this.downloadLocation = "",
final List<String> localLibraryLocation = const [],
this.pipedInstance = "https://pipedapi.kavin.rocks",
this.themeMode = ThemeMode.system,
this.audioSource = AudioSource.youtube,
@ -440,7 +454,8 @@ class _$UserPreferencesImpl implements _UserPreferences {
this.downloadMusicCodec = SourceCodecs.m4a,
this.discordPresence = true,
this.endlessPlayback = true,
this.enableConnect = false});
this.enableConnect = false})
: _localLibraryLocation = localLibraryLocation;
factory _$UserPreferencesImpl.fromJson(Map<String, dynamic> json) =>
_$$UserPreferencesImplFromJson(json);
@ -496,6 +511,16 @@ class _$UserPreferencesImpl implements _UserPreferences {
@override
@JsonKey()
final String downloadLocation;
final List<String> _localLibraryLocation;
@override
@JsonKey()
List<String> get localLibraryLocation {
if (_localLibraryLocation is EqualUnmodifiableListView)
return _localLibraryLocation;
// ignore: implicit_dynamic_type
return EqualUnmodifiableListView(_localLibraryLocation);
}
@override
@JsonKey()
final String pipedInstance;
@ -523,7 +548,7 @@ class _$UserPreferencesImpl implements _UserPreferences {
@override
String toString() {
return 'UserPreferences(audioQuality: $audioQuality, albumColorSync: $albumColorSync, amoledDarkTheme: $amoledDarkTheme, checkUpdate: $checkUpdate, normalizeAudio: $normalizeAudio, showSystemTrayIcon: $showSystemTrayIcon, skipNonMusic: $skipNonMusic, systemTitleBar: $systemTitleBar, closeBehavior: $closeBehavior, accentColorScheme: $accentColorScheme, layoutMode: $layoutMode, locale: $locale, recommendationMarket: $recommendationMarket, searchMode: $searchMode, downloadLocation: $downloadLocation, pipedInstance: $pipedInstance, themeMode: $themeMode, audioSource: $audioSource, streamMusicCodec: $streamMusicCodec, downloadMusicCodec: $downloadMusicCodec, discordPresence: $discordPresence, endlessPlayback: $endlessPlayback, enableConnect: $enableConnect)';
return 'UserPreferences(audioQuality: $audioQuality, albumColorSync: $albumColorSync, amoledDarkTheme: $amoledDarkTheme, checkUpdate: $checkUpdate, normalizeAudio: $normalizeAudio, showSystemTrayIcon: $showSystemTrayIcon, skipNonMusic: $skipNonMusic, systemTitleBar: $systemTitleBar, closeBehavior: $closeBehavior, accentColorScheme: $accentColorScheme, layoutMode: $layoutMode, locale: $locale, recommendationMarket: $recommendationMarket, searchMode: $searchMode, downloadLocation: $downloadLocation, localLibraryLocation: $localLibraryLocation, pipedInstance: $pipedInstance, themeMode: $themeMode, audioSource: $audioSource, streamMusicCodec: $streamMusicCodec, downloadMusicCodec: $downloadMusicCodec, discordPresence: $discordPresence, endlessPlayback: $endlessPlayback, enableConnect: $enableConnect)';
}
@override
@ -560,6 +585,8 @@ class _$UserPreferencesImpl implements _UserPreferences {
other.searchMode == searchMode) &&
(identical(other.downloadLocation, downloadLocation) ||
other.downloadLocation == downloadLocation) &&
const DeepCollectionEquality()
.equals(other._localLibraryLocation, _localLibraryLocation) &&
(identical(other.pipedInstance, pipedInstance) ||
other.pipedInstance == pipedInstance) &&
(identical(other.themeMode, themeMode) ||
@ -597,6 +624,7 @@ class _$UserPreferencesImpl implements _UserPreferences {
recommendationMarket,
searchMode,
downloadLocation,
const DeepCollectionEquality().hash(_localLibraryLocation),
pipedInstance,
themeMode,
audioSource,
@ -647,6 +675,7 @@ abstract class _UserPreferences implements UserPreferences {
final Market recommendationMarket,
final SearchMode searchMode,
final String downloadLocation,
final List<String> localLibraryLocation,
final String pipedInstance,
final ThemeMode themeMode,
final AudioSource audioSource,
@ -698,6 +727,8 @@ abstract class _UserPreferences implements UserPreferences {
@override
String get downloadLocation;
@override
List<String> get localLibraryLocation;
@override
String get pipedInstance;
@override
ThemeMode get themeMode;

View File

@ -43,6 +43,10 @@ _$UserPreferencesImpl _$$UserPreferencesImplFromJson(Map json) =>
$enumDecodeNullable(_$SearchModeEnumMap, json['searchMode']) ??
SearchMode.youtube,
downloadLocation: json['downloadLocation'] as String? ?? "",
localLibraryLocation: (json['localLibraryLocation'] as List<dynamic>?)
?.map((e) => e as String)
.toList() ??
const [],
pipedInstance:
json['pipedInstance'] as String? ?? "https://pipedapi.kavin.rocks",
themeMode: $enumDecodeNullable(_$ThemeModeEnumMap, json['themeMode']) ??
@ -80,6 +84,7 @@ Map<String, dynamic> _$$UserPreferencesImplToJson(
'recommendationMarket': _$MarketEnumMap[instance.recommendationMarket]!,
'searchMode': _$SearchModeEnumMap[instance.searchMode]!,
'downloadLocation': instance.downloadLocation,
'localLibraryLocation': instance.localLibraryLocation,
'pipedInstance': instance.pipedInstance,
'themeMode': _$ThemeModeEnumMap[instance.themeMode]!,
'audioSource': _$AudioSourceEnumMap[instance.audioSource]!,

View File

@ -12,6 +12,7 @@ import 'package:media_kit/media_kit.dart' as mk;
import 'package:spotube/services/audio_player/loop_mode.dart';
import 'package:spotube/services/audio_player/playback_state.dart';
import 'package:spotube/services/sourced_track/sourced_track.dart';
part 'audio_players_streams_mixin.dart';
part 'audio_player_impl.dart';
@ -29,12 +30,18 @@ class SpotubeMedia extends mk.Media {
: "http://${InternetAddress.loopbackIPv4.address}:${PlaybackServer.port}/stream/${track.id}",
extras: {
...?extras,
"track": track.toJson(),
"track": switch (track) {
LocalTrack() => track.toJson(),
SourcedTrack() => track.toJson(),
_ => track.toJson(),
},
},
);
factory SpotubeMedia.fromMedia(mk.Media media) {
final track = Track.fromJson(media.extras?["track"]);
final track = media.uri.startsWith("http")
? Track.fromJson(media.extras?["track"])
: LocalTrack.fromJson(media.extras?["track"]);
return SpotubeMedia(track);
}
}

View File

@ -14,6 +14,7 @@
#include <media_kit_libs_linux/media_kit_libs_linux_plugin.h>
#include <screen_retriever/screen_retriever_plugin.h>
#include <system_theme/system_theme_plugin.h>
#include <system_tray/system_tray_plugin.h>
#include <tray_manager/tray_manager_plugin.h>
#include <url_launcher_linux/url_launcher_plugin.h>
#include <window_manager/window_manager_plugin.h>
@ -44,6 +45,9 @@ void fl_register_plugins(FlPluginRegistry* registry) {
g_autoptr(FlPluginRegistrar) system_theme_registrar =
fl_plugin_registry_get_registrar_for_plugin(registry, "SystemThemePlugin");
system_theme_plugin_register_with_registrar(system_theme_registrar);
g_autoptr(FlPluginRegistrar) system_tray_registrar =
fl_plugin_registry_get_registrar_for_plugin(registry, "SystemTrayPlugin");
system_tray_plugin_register_with_registrar(system_tray_registrar);
g_autoptr(FlPluginRegistrar) tray_manager_registrar =
fl_plugin_registry_get_registrar_for_plugin(registry, "TrayManagerPlugin");
tray_manager_plugin_register_with_registrar(tray_manager_registrar);

View File

@ -11,6 +11,7 @@ list(APPEND FLUTTER_PLUGIN_LIST
media_kit_libs_linux
screen_retriever
system_theme
system_tray
tray_manager
url_launcher_linux
window_manager

View File

@ -21,6 +21,7 @@ import screen_retriever
import shared_preferences_foundation
import sqflite
import system_theme
import system_tray
import tray_manager
import url_launcher_macos
import window_manager
@ -43,6 +44,7 @@ func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) {
SharedPreferencesPlugin.register(with: registry.registrar(forPlugin: "SharedPreferencesPlugin"))
SqflitePlugin.register(with: registry.registrar(forPlugin: "SqflitePlugin"))
SystemThemePlugin.register(with: registry.registrar(forPlugin: "SystemThemePlugin"))
SystemTrayPlugin.register(with: registry.registrar(forPlugin: "SystemTrayPlugin"))
TrayManagerPlugin.register(with: registry.registrar(forPlugin: "TrayManagerPlugin"))
UrlLauncherPlugin.register(with: registry.registrar(forPlugin: "UrlLauncherPlugin"))
WindowManagerPlugin.register(with: registry.registrar(forPlugin: "WindowManagerPlugin"))

View File

@ -41,6 +41,8 @@ PODS:
- FlutterMacOS
- system_theme (0.0.1):
- FlutterMacOS
- system_tray (0.0.1):
- FlutterMacOS
- tray_manager (0.0.1):
- FlutterMacOS
- url_launcher_macos (0.0.1):
@ -70,6 +72,7 @@ DEPENDENCIES:
- shared_preferences_foundation (from `Flutter/ephemeral/.symlinks/plugins/shared_preferences_foundation/darwin`)
- sqflite (from `Flutter/ephemeral/.symlinks/plugins/sqflite/darwin`)
- system_theme (from `Flutter/ephemeral/.symlinks/plugins/system_theme/macos`)
- system_tray (from `Flutter/ephemeral/.symlinks/plugins/system_tray/macos`)
- tray_manager (from `Flutter/ephemeral/.symlinks/plugins/tray_manager/macos`)
- url_launcher_macos (from `Flutter/ephemeral/.symlinks/plugins/url_launcher_macos/macos`)
- window_manager (from `Flutter/ephemeral/.symlinks/plugins/window_manager/macos`)
@ -118,6 +121,8 @@ EXTERNAL SOURCES:
:path: Flutter/ephemeral/.symlinks/plugins/sqflite/darwin
system_theme:
:path: Flutter/ephemeral/.symlinks/plugins/system_theme/macos
system_tray:
:path: Flutter/ephemeral/.symlinks/plugins/system_tray/macos
tray_manager:
:path: Flutter/ephemeral/.symlinks/plugins/tray_manager/macos
url_launcher_macos:
@ -148,6 +153,7 @@ SPEC CHECKSUMS:
shared_preferences_foundation: b4c3b4cddf1c21f02770737f147a3f5da9d39695
sqflite: 673a0e54cc04b7d6dba8d24fb8095b31c3a99eec
system_theme: c7b9f6659a5caa26c9bc2284da096781e9a6fcbc
system_tray: e53c972838c69589ff2e77d6d3abfd71332f9e5d
tray_manager: 9064e219c56d75c476e46b9a21182087930baf90
url_launcher_macos: d2691c7dd33ed713bf3544850a623080ec693d95
window_manager: 3a1844359a6295ab1e47659b1a777e36773cd6e8

View File

@ -1455,12 +1455,13 @@ packages:
source: hosted
version: "1.0.9"
media_kit_native_event_loop:
dependency: transitive
dependency: "direct overridden"
description:
name: media_kit_native_event_loop
sha256: a605cf185499d14d58935b8784955a92a4bf0ff4e19a23de3d17a9106303930e
url: "https://pub.dev"
source: hosted
path: media_kit_native_event_loop
ref: main
resolved-ref: "285f7919bbf4a7d89a62615b14a3766a171ad575"
url: "https://github.com/media-kit/media-kit"
source: git
version: "1.0.8"
menu_base:
dependency: transitive
@ -2157,6 +2158,15 @@ packages:
url: "https://pub.dev"
source: hosted
version: "0.0.2"
system_tray:
dependency: "direct overridden"
description:
path: "."
ref: dc7ef410d5cfec897edf060c1c4baff69f7c181c
resolved-ref: dc7ef410d5cfec897edf060c1c4baff69f7c181c
url: "https://github.com/antler119/system_tray"
source: git
version: "2.0.2"
term_glyph:
dependency: transitive
description:

View File

@ -153,6 +153,17 @@ dev_dependencies:
dependency_overrides:
uuid: ^4.4.0
system_tray:
# TODO: remove this when flutter_desktop_tools gets updated
# to use [MenuItemBase] instead of [MenuItem]
git:
url: https://github.com/antler119/system_tray
ref: dc7ef410d5cfec897edf060c1c4baff69f7c181c
media_kit_native_event_loop: # to fix "macro name must be an identifier"
git:
url: https://github.com/media-kit/media-kit
path: media_kit_native_event_loop
ref: main
flutter:
generate: true

View File

@ -1,89 +1,209 @@
{
"ar": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"bn": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"ca": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"cs": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"de": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"es": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"eu": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"fa": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"fi": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"fr": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"hi": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"id": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"it": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"ja": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"ka": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"ko": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"ne": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"nl": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"pl": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"pt": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"ru": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"th": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"tr": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"uk": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"vi": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
],
"zh": [
"local_library",
"add_library_location",
"remove_library_location",
"local_tab",
"stats"
]
}

View File

@ -16,6 +16,7 @@
#include <permission_handler_windows/permission_handler_windows_plugin.h>
#include <screen_retriever/screen_retriever_plugin.h>
#include <system_theme/system_theme_plugin.h>
#include <system_tray/system_tray_plugin.h>
#include <tray_manager/tray_manager_plugin.h>
#include <url_launcher_windows/url_launcher_windows.h>
#include <window_manager/window_manager_plugin.h>
@ -42,6 +43,8 @@ void RegisterPlugins(flutter::PluginRegistry* registry) {
registry->GetRegistrarForPlugin("ScreenRetrieverPlugin"));
SystemThemePluginRegisterWithRegistrar(
registry->GetRegistrarForPlugin("SystemThemePlugin"));
SystemTrayPluginRegisterWithRegistrar(
registry->GetRegistrarForPlugin("SystemTrayPlugin"));
TrayManagerPluginRegisterWithRegistrar(
registry->GetRegistrarForPlugin("TrayManagerPlugin"));
UrlLauncherWindowsRegisterWithRegistrar(

View File

@ -13,6 +13,7 @@ list(APPEND FLUTTER_PLUGIN_LIST
permission_handler_windows
screen_retriever
system_theme
system_tray
tray_manager
url_launcher_windows
window_manager