Compare commits

..

17 Commits

Author SHA1 Message Date
Richard Hajek
2c82cd258e
Merge 42e954428b into 4838656dcc 2025-11-14 17:07:33 +05:45
Kingkor Roy Tirtho
4838656dcc chore: fix alternative sources not persisting 2025-11-14 16:17:37 +06:00
Kingkor Roy Tirtho
7ad2066684 website: add AppImage link 2025-11-14 15:07:51 +06:00
Kingkor Roy Tirtho
77c32a27cf chor: idkl 2025-11-14 14:26:46 +06:00
Kingkor Roy Tirtho
89c67e4f89 docs: update readme credits 2025-11-14 14:19:12 +06:00
Kingkor Roy Tirtho
b142928412 chore: remove buffer size limit 2025-11-14 13:38:47 +06:00
Kingkor Roy Tirtho
bf2eb0ffac chore: bump version and generate CHANGELOG 2025-11-14 13:36:20 +06:00
Kingkor Roy Tirtho
3f5291ec92 chore: upgrade hetu_std 2025-11-14 13:09:46 +06:00
Kingkor Roy Tirtho
5ea4df932f cd: add back rpm for x86_64 2025-11-13 16:03:08 +06:00
Kingkor Roy Tirtho
3462e32a6c chore: add arch for install deps 2025-11-13 15:39:58 +06:00
Kingkor Roy Tirtho
a452122302 cd: add appimage support and use fastforge 2025-11-13 15:15:13 +06:00
Kingkor Roy Tirtho
11866d532b chore: remove useless dependencies 2025-11-13 14:57:08 +06:00
Kingkor Roy Tirtho
d843ce9ede chore: upgrade yt plugin 2025-11-12 15:55:07 +06:00
Kingkor Roy Tirtho
6884a131c9 fix(playback): use stream instead of chunked serving of audio bytes 2025-11-12 14:35:06 +06:00
Kingkor Roy Tirtho
bb6f4bd57b feat(android): add 16KB page size support 2025-11-12 11:17:34 +06:00
Kingkor Roy Tirtho
cab09e00ce chore: ignore DB queries in migrations 2025-11-12 10:02:37 +06:00
Kingkor Roy Tirtho
4fae9013a7 fix: download not working in different devices and slow 2025-11-11 10:39:13 +06:00
126 changed files with 2084 additions and 2273 deletions

25
.github/Dockerfile vendored
View File

@ -1,25 +0,0 @@
ARG FLUTTER_VERSION
FROM --platform=linux/arm64 krtirtho/flutter_distributor:${FLUTTER_VERSION}
ARG BUILD_VERSION
WORKDIR /app
COPY . .
RUN chown -R $(whoami) /app
RUN rustup target add aarch64-unknown-linux-gnu
RUN flutter pub get
RUN alias dpkg-deb="dpkg-deb --Zxz" &&\
flutter_distributor package --platform=linux --targets=deb --skip-clean
RUN make tar VERSION=${BUILD_VERSION} ARCH=arm64 PKG_ARCH=aarch64
RUN mv build/spotube-linux-*-aarch64.tar.xz dist/ &&\
mv dist/**/spotube-*-linux.deb dist/Spotube-linux-aarch64.deb
CMD [ "sleep", "5000000" ]

View File

@ -37,12 +37,14 @@ jobs:
files: |
dist/Spotube-linux-x86_64.deb
dist/Spotube-linux-x86_64.rpm
dist/Spotube-linux-x86_64.AppImage
dist/spotube-linux-*-x86_64.tar.xz
- os: ubuntu-22.04-arm
platform: linux
arch: arm64
files: |
dist/Spotube-linux-aarch64.deb
dist/Spotube-linux-aarch64.AppImage
dist/spotube-linux-*-aarch64.tar.xz
- os: ubuntu-22.04
platform: android
@ -107,7 +109,7 @@ jobs:
- name: Install ${{matrix.platform}} dependencies
run: |
flutter pub get
dart cli/cli.dart install-dependencies --platform=${{matrix.platform}}
dart cli/cli.dart install-dependencies --platform=${{matrix.platform}} --arch=${{matrix.arch}}
- name: Sign Apk
if: ${{matrix.platform == 'android'}}

View File

@ -1,5 +1,29 @@
# Changelog
## [5.1.0](https://github.com/KRTirtho/spotube/compare/v5.0.0...v5.1.0) (2025-11-14)
### Features
- Show plugin source and set the only plugin as default if no plugin is there
- **playback**: Add dab music source
- **playback**: Add uncompressed flac playback support
- Add plugin audio source models and api service
- **plugins**: Filter plugins by abilities in plugins page and show abilities as badge
- Add setting default audio source support
- Move away from track source query and preferences audio quality and codec
- Add NewPipe support for desktop platforms
- Add default plugin loading capability
- **queue**: Add multi-select and bulk actions to queue ([#2839](https://github.com/KRTirtho/spotube/issues/2839))
- **android**: Add 16KB page size support
### Bug Fixes
- Change plugin download directory to application support
- **playback**: Play next not working
- Downloaded tracks are not tagged with metadata
- Download not working in different devices and slow
- **playback**: Use stream instead of chunked serving of audio bytes
## [5.0.0](https://github.com/KRTirtho/spotube/compare/v4.0.2...v5.0.0) (2025-09-08)
### Features

View File

@ -2,7 +2,7 @@
<img width="600" src="assets/branding/spotube_banner.png" alt="Spotube Logo">
A cross-platform extensible open-source music streaming platform.<br>
Bring your own music metadata/playlist with plugins created by community or by yourself. A small step towards the decentralized music streaming era!
Bring your own music metadata/playlist/audio-source with plugins created by community or by yourself. A small step towards the decentralized music streaming era!
Btw it's not just another Electron app 😉
@ -202,6 +202,7 @@ If you are curious, you can [read the reason of choosing this license](https://d
1. [Invidious](https://invidious.io/) - Invidious is an open source alternative front-end to YouTube.
1. [yt-dlp](https://github.com/yt-dlp/yt-dlp) - A feature-rich command-line audio/video downloader.
1. [NewPipeExtractor](https://github.com/TeamNewPipe/NewPipeExtractor) - NewPipe's core library for extracting data from streaming sites.
1. [YouTubeExplodeDart](https://github.com/Hexer10/youtube_explode_dart) - A port in dart of the youtube explode library. Supports several API functions without the need of Youtube API Key.
1. [LRCLib](https://lrclib.net/) - A public synced lyric API.
1. [Linux](https://www.linux.org) - Linux is a family of open-source Unix-like operating systems based on the Linux kernel, an operating system kernel first released on September 17, 1991, by Linus Torvalds. Linux is typically packaged in a Linux distribution
1. [AUR](https://aur.archlinux.org) - AUR stands for Arch User Repository. It is a community-driven repository for Arch-based Linux distributions users
@ -215,7 +216,6 @@ If you are curious, you can [read the reason of choosing this license](https://d
1. [app_links](https://github.com/llfbandit/app_links) - Android App Links, Deep Links, iOs Universal Links and Custom URL schemes handler for Flutter (desktop included).
1. [args](https://pub.dev/packages/args) - Library for defining parsers for parsing raw command-line arguments into a set of options and values using GNU and POSIX style options.
1. [async](https://pub.dev/packages/async) - Utility functions and classes related to the 'dart:async' library.
1. [audio_service](https://pub.dev/packages/audio_service) - Flutter plugin to play audio in the background while the screen is off.
1. [audio_service_mpris](https://github.com/bdrazhzhov/audio-service-mpris) - audio_service platform interface supporting Media Player Remote Interfacing Specification.
1. [audio_session](https://github.com/ryanheise/audio_session) - Sets the iOS audio session category and Android audio attributes for your app, and manages your app's audio focus, mixing and ducking behaviour.
@ -242,15 +242,11 @@ If you are curious, you can [read the reason of choosing this license](https://d
1. [flutter_inappwebview](https://inappwebview.dev/) - A Flutter plugin that allows you to add an inline webview, to use an headless webview, and to open an in-app browser window.
1. [flutter_native_splash](https://pub.dev/packages/flutter_native_splash) - Customize Flutter's default white native splash screen with background color and splash image. Supports dark mode, full screen, and more.
1. [flutter_riverpod](https://riverpod.dev) - A reactive caching and data-binding framework. Riverpod makes working with asynchronous code a breeze.
1. [flutter_secure_storage](https://pub.dev/packages/flutter_secure_storage) - Flutter Secure Storage provides API to store data in secure storage. Keychain is used in iOS, KeyStore based solution is used in Android.
1. [flutter_sharing_intent](https://github.com/bhagat-techind/flutter_sharing_intent.git) - A flutter plugin that allow flutter apps to receive photos, videos, text, urls or any other file types from another app.
1. [flutter_undraw](https://github.com/KRTirtho/flutter_undraw) - Undraw.co Illustrations for Flutter with customization options
1. [form_builder_validators](https://github.com/flutter-form-builder-ecosystem) - Form Builder Validators set of validators for FlutterFormBuilder. Provides common validators and a way to make your own.
1. [form_validator](https://github.com/TheMisir/form-validator) - Simplest form validation library for flutter's form field widgets
1. [freezed_annotation](https://pub.dev/packages/freezed_annotation) - Annotations for the freezed code-generator. This package does nothing without freezed too.
1. [fuzzywuzzy](https://github.com/sphericalkat/dart-fuzzywuzzy) - An implementation of the popular fuzzywuzzy package in Dart, to suit all your fuzzy string matching/searching needs!
1. [gap](https://github.com/letsar/gap) - Flutter widgets for easily adding gaps inside Flex widgets such as Columns and Rows or scrolling views.
1. [google_fonts](https://pub.dev/packages/google_fonts) - A Flutter package to use fonts from fonts.google.com. Supports HTTP fetching, caching, and asset bundling.
1. [home_widget](https://pub.dev/packages/home_widget) - A plugin to provide a common interface for creating HomeScreen Widgets for Android and iOS.
1. [hooks_riverpod](https://riverpod.dev) - A reactive caching and data-binding framework. Riverpod makes working with asynchronous code a breeze.
1. [html](https://pub.dev/packages/html) - APIs for parsing and manipulating HTML content outside the browser.
@ -258,15 +254,10 @@ If you are curious, you can [read the reason of choosing this license](https://d
1. [http](https://pub.dev/packages/http) - A composable, multi-platform, Future-based API for HTTP requests.
1. [image_picker](https://pub.dev/packages/image_picker) - Flutter plugin for selecting images from the Android and iOS image library, and taking new pictures with the camera.
1. [intl](https://pub.dev/packages/intl) - Contains code to deal with internationalized/localized messages, date and number formatting and parsing, bi-directional text, and other internationalization issues.
1. [invidious](https://pub.dev/packages/invidious) - Invidious API client for Dart and Flutter.
1. [jiosaavn](https://github.com/KRTirtho/jiosaavn) - Unofficial API client for jiosaavn.com
1. [json_annotation](https://pub.dev/packages/json_annotation) - Classes and helper functions that support JSON code generation via the `json_serializable` package.
1. [local_notifier](https://github.com/leanflutter/local_notifier) - This plugin allows Flutter desktop apps to displaying local notifications.
1. [logger](https://pub.dev/packages/logger) - Small, easy to use and extensible logger which prints beautiful logs.
1. [logging](https://pub.dev/packages/logging) - Provides APIs for debugging and error logging, similar to loggers in other languages, such as the Closure JS Logger and java.util.logging.Logger.
1. [lrc](https://pub.dev/packages/lrc) - A Dart-only package that creates, parses, and handles LRC, which is a format that stores song lyrics.
1. [media_kit](https://github.com/media-kit/media-kit) - A cross-platform video player & audio player for Flutter & Dart. Performant, stable, feature-proof & modular.
1. [media_kit_libs_audio](https://github.com/media-kit/media-kit.git) - package:media_kit audio (only) playback native libraries for all platforms.
1. [metadata_god](https://pub.dev/packages/metadata_god) - Plugin for retrieving and writing audio tags/metadata from audio files
1. [mime](https://pub.dev/packages/mime) - Utilities for handling media (MIME) types, including determining a type from a file extension and file contents.
1. [open_file](https://pub.dev/packages/open_file) - A plug-in that can call native APP to open files with string result in flutter, support iOS(UTI) / android(intent) / PC(ffi) / web(dart:html)
@ -275,7 +266,6 @@ If you are curious, you can [read the reason of choosing this license](https://d
1. [path](https://pub.dev/packages/path) - A string-based path manipulation library. All of the path operations you know and love, with solid support for Windows, POSIX (Linux and Mac OS X), and the web.
1. [path_provider](https://pub.dev/packages/path_provider) - Flutter plugin for getting commonly used locations on host platform file systems, such as the temp and app data directories.
1. [permission_handler](https://pub.dev/packages/permission_handler) - Permission plugin for Flutter. This plugin provides a cross-platform (iOS, Android) API to request and check permissions.
1. [piped_client](https://github.com/KRTirtho/piped_client) - API Client for piped.video
1. [riverpod](https://riverpod.dev) - A reactive caching and data-binding framework. Riverpod makes working with asynchronous code a breeze.
1. [scroll_to_index](https://github.com/quire-io/scroll-to-index) - Scroll to a specific child of any scrollable widget in Flutter
1. [shadcn_flutter](https://github.com/sunarya-thito/shadcn_flutter) - Beautifully designed components from Shadcn/UI is now available for Flutter
@ -290,9 +280,6 @@ If you are curious, you can [read the reason of choosing this license](https://d
1. [smtc_windows](https://pub.dev/packages/smtc_windows) - Windows `SystemMediaTransportControls` implementation for Flutter giving access to Windows OS Media Control applet.
1. [sqlite3](https://github.com/simolus3/sqlite3.dart/tree/main/sqlite3) - Provides lightweight yet convenient bindings to SQLite by using dart:ffi
1. [sqlite3_flutter_libs](https://github.com/simolus3/sqlite3.dart/tree/main/sqlite3_flutter_libs) - Flutter plugin to include native sqlite3 libraries with your app
1. [stroke_text](https://github.com/MohamedAbd0/stroke_text) - A Simple Flutter plugin for applying stroke (border) style to a text widget
1. [system_theme](https://github.com/bdlukaa/system_theme/tree/master/system_theme) - A plugin to get the current system theme info. Supports Android, Web, Windows, Linux and macOS
1. [test](https://pub.dev/packages/test) - A full featured library for writing and running Dart tests across platforms.
1. [timezone](https://pub.dev/packages/timezone) - Time zone database and time zone aware DateTime.
1. [titlebar_buttons](https://github.com/gtk-flutter/titlebar_buttons) - A package which provides most of the titlebar buttons from windows, linux and macos.
1. [tray_manager](https://github.com/leanflutter/tray_manager) - This plugin allows Flutter desktop apps to defines system tray.
@ -304,12 +291,17 @@ If you are curious, you can [read the reason of choosing this license](https://d
1. [web_socket_channel](https://pub.dev/packages/web_socket_channel) - StreamChannel wrappers for WebSockets. Provides a cross-platform WebSocketChannel API, a cross-platform implementation of that API that communicates over an underlying StreamChannel.
1. [wikipedia_api](https://github.com/KRTirtho/wikipedia_api) - Wikipedia API for dart and flutter
1. [win32_registry](https://pub.dev/packages/win32_registry) - A package that provides a friendly Dart API for accessing the Windows Registry.
1. [window_manager](https://github.com/leanflutter/window_manager) - This plugin allows Flutter desktop apps to resizing and repositioning the window.
1. [window_manager](https://leanflutter.dev) - This plugin allows Flutter desktop apps to resizing and repositioning the window.
1. [youtube_explode_dart](https://github.com/Hexer10/youtube_explode_dart) - A port in dart of the youtube explode library. Supports several API functions without the need of Youtube API Key.
1. [http_parser](https://pub.dev/packages/http_parser) - A platform-independent package for parsing and serializing HTTP formats.
1. [collection](https://pub.dev/packages/collection) - Collections and utilities functions and classes related to collections.
1. [otp_util](https://github.com/dushiling) - otp_util is a dart package to generate and verify one-time passwords,it It provides two methods TOPT and HOTP.They are Time-based OTPs and Counter-based OTPs.
1. [dio_http2_adapter](https://github.com/cfug/dio) - An adapter that combines HTTP/2 and dio. Supports reusing connections, header compression, etc.
1. [archive](https://pub.dev/packages/archive) - Provides encoders and decoders for various archive and compression formats such as zip, tar, bzip2, gzip, and zlib.
1. [hetu_script](https://github.com/hetu-script/hetu-script) - Hetu is a lightweight scripting language for embedding in Flutter apps.
1. [get_it](https://github.com/flutter-it/get_it) - Simple direct Service Locator that allows to decouple the interface from a concrete implementation and to access the concrete implementation from everywhere in your App"
1. [flutter_markdown_plus](https://pub.dev/packages/flutter_markdown_plus) - A Markdown renderer for Flutter. Create rich text output, including text styles, tables, links, and more, from plain text data formatted with simple Markdown tags.
1. [pub_semver](https://pub.dev/packages/pub_semver) - Versions and version constraints implementing pub's versioning policy. This is very similar to vanilla semver, with a few corner cases.
1. [change_case](https://github.com/mrgnhnt96/change_case) - An extension on String for the missing methods for camelCase, PascalCase, Capital Case, snake_case, param-case, CONSTANT_CASE and others.
1. [flutter_secure_storage](https://pub.dev/packages/flutter_secure_storage) - Flutter Secure Storage provides API to store data in secure storage. Keychain is used in iOS, KeyStore based solution is used in Android.
1. [build_runner](https://pub.dev/packages/build_runner) - A build system for Dart code generation and modular compilation.
1. [envied_generator](https://github.com/petercinibulk/envied) - Generator for the Envied package. See https://pub.dev/packages/envied.
1. [flutter_gen_runner](https://github.com/FlutterGen/flutter_gen) - The Flutter code generator for your assets, fonts, colors, … — Get rid of all String-based APIs.
@ -320,17 +312,23 @@ If you are curious, you can [read the reason of choosing this license](https://d
1. [process_run](https://github.com/tekartik/process_run.dart/blob/master/packages/process_run) - Process run helpers for Linux/Win/Mac and which like feature for finding executables.
1. [pubspec_parse](https://pub.dev/packages/pubspec_parse) - Simple package for parsing pubspec.yaml files with a type-safe API and rich error reporting.
1. [pub_api_client](https://github.com/leoafarias/pub_api_client) - An API Client for Pub to interact with public package information.
1. [xml](https://github.com/renggli/dart-xml) - A lightweight library for parsing, traversing, querying, transforming and building XML documents.
1. [io](https://pub.dev/packages/io) - Utilities for the Dart VM Runtime including support for ANSI colors, file copying, and standard exit code values.
1. [drift_dev](https://drift.simonbinder.eu/) - Dev-dependency for users of drift. Contains the generator and development tools.
1. [test](https://pub.dev/packages/test) - A full featured library for writing and running Dart tests across platforms.
1. [auto_route_generator](https://github.com/Milad-Akarie/auto_route_library) - AutoRoute is a declarative routing solution, where everything needed for navigation is automatically generated for you.
1. [desktop_webview_window](https://github.com/MixinNetwork/flutter-plugins/tree/main/packages/desktop_webview_window) - Show a webview window on your flutter desktop application.
1. [disable_battery_optimization](https://github.com/pvsvamsi/Disable-Battery-Optimizations) - Flutter plugin to check and disable battery optimizations. Also shows custom steps to disable the optimizations in devices like mi, xiaomi, samsung, oppo, huawei, oneplus etc
1. [draggable_scrollbar](https://github.com/fluttercommunity/flutter-draggable-scrollbar) - A scrollbar that can be dragged for quickly navigation through a vertical list. Additional option is showing label next to scrollthumb with information about current item.
1. [flutter_broadcasts](https://github.com/KRTirtho/flutter_broadcasts.git) - A plugin for sending and receiving broadcasts with Android intents and iOS notifications.
1. [scrobblenaut](https://github.com/Nebulino/Scrobblenaut) - A deadly simple LastFM API Wrapper for Dart. So deadly simple that it's gonna hit the mark.
1. [yt_dlp_dart](https://github.com/KRTirtho/yt_dlp_dart.git) - yt-dlp binding in Dart
1. [yt_dlp_dart](https://github.com/KRTirtho/yt_dlp_dart.git) - A starting point for Dart libraries or applications.
1. [flutter_new_pipe_extractor](https://github.com/KRTirtho/flutter_new_pipe_extractor) - NewPipeExtractor binding for Flutter (Android only)
1. [hetu_std](https://github.com/hetu-community/hetu_std.git) - A sample command-line application.
1. [hetu_otp_util](https://github.com/hetu-community/hetu_otp_util.git) - A sample command-line application.
1. [hetu_spotube_plugin](https://github.com/KRTirtho/hetu_spotube_plugin) - A new Flutter package project.
1. [media_kit](https://github.com/media-kit/media-kit) - A cross-platform video player & audio player for Flutter & Dart. Performant, stable, feature-proof & modular.
1. [media_kit_libs_audio](https://github.com/media-kit/media-kit.git) - package:media_kit audio (only) playback native libraries for all platforms.
</details>
<div align="center"><h4>© Copyright Spotube 2025</h4></div>

View File

@ -36,7 +36,7 @@ android {
compileSdkVersion 36
ndkVersion = "27.0.12077973"
ndkVersion = "29.0.14206865"
compileOptions {
coreLibraryDesugaringEnabled true

View File

@ -59,7 +59,7 @@ mixin BuildCommandCommonSteps on Command {
"""
flutter pub get
dart run build_runner build --delete-conflicting-outputs
dart pub global activate flutter_distributor
dart pub global activate fastforge
""",
);
}

View File

@ -37,12 +37,11 @@ class LinuxBuildCommand extends Command with BuildCommandCommonSteps {
await bootstrap();
await shell.run(
"flutter_distributor package --platform=linux --targets=deb",
"fastforge package --platform=linux --targets=deb,appimage",
);
if (architecture == "x86") {
await shell.run(
"flutter_distributor package --platform=linux --targets=rpm",
"fastforge package --platform=linux --targets=rpm",
);
}
@ -116,6 +115,23 @@ class LinuxBuildCommand extends Command with BuildCommandCommonSteps {
await ogRpm.delete();
}
final ogAppImage = File(
join(
cwd.path,
"dist",
pubspec.version.toString(),
"spotube-${pubspec.version}-linux.AppImage",
),
);
await ogAppImage.copy(
join(
cwd.path,
"dist",
"Spotube-linux-$bundleArchName.AppImage",
),
);
await ogAppImage.delete();
stdout.writeln("✅ Linux building done");
}
}

View File

@ -21,7 +21,7 @@ class MacosBuildCommand extends Command with BuildCommandCommonSteps {
"""
flutter build macos
appdmg appdmg.json ${join(cwd.path, "build", "Spotube-macos-universal.dmg")}
flutter_distributor package --platform=macos --targets pkg --skip-clean
fastforge package --platform=macos --targets pkg --skip-clean
""",
);

View File

@ -61,7 +61,7 @@ class WindowsBuildCommand extends Command with BuildCommandCommonSteps {
);
await shell.run(
"flutter_distributor package --platform=windows --targets=exe --skip-clean",
"fastforge package --platform=windows --targets=exe --skip-clean",
);
final ogExe = File(

View File

@ -37,6 +37,8 @@ class InstallDependenciesCommand extends Command {
FutureOr? run() async {
final shell = Shell();
final arch = argResults?.option("arch") == "x86" ? "x86_64" : "aarch64";
switch (argResults!.option("platform")) {
case "windows":
await shell.run(
@ -49,7 +51,10 @@ class InstallDependenciesCommand extends Command {
await shell.run(
"""
sudo apt-get update -y
sudo apt-get install -y tar clang cmake ninja-build pkg-config libgtk-3-dev make python3-pip python3-setuptools desktop-file-utils libgdk-pixbuf2.0-dev fakeroot strace fuse libunwind-dev locate patchelf gir1.2-appindicator3-0.1 libappindicator3-1 libappindicator3-dev libsecret-1-0 libjsoncpp25 libsecret-1-dev libjsoncpp-dev libnotify-bin libnotify-dev mpv libmpv-dev libwebkit2gtk-4.1-0 libwebkit2gtk-4.1-dev libsoup-3.0-0 libsoup-3.0-dev
sudo apt-get install -y wget tar clang cmake ninja-build pkg-config libgtk-3-dev make python3-pip python3-setuptools desktop-file-utils libgdk-pixbuf2.0-dev fakeroot strace fuse libunwind-dev locate patchelf gir1.2-appindicator3-0.1 libappindicator3-1 libappindicator3-dev libsecret-1-0 libjsoncpp25 libsecret-1-dev libjsoncpp-dev libnotify-bin libnotify-dev mpv libmpv-dev libwebkit2gtk-4.1-0 libwebkit2gtk-4.1-dev libsoup-3.0-0 libsoup-3.0-dev
wget -O appimagetool "https://github.com/AppImage/appimagetool/releases/download/continuous/appimagetool-$arch.AppImage"
chmod +x appimagetool
sudo mv appimagetool /usr/local/bin/
""",
);
break;

View File

@ -12,13 +12,12 @@ class ReplaceDownloadedDialog extends ConsumerWidget {
@override
Widget build(BuildContext context, ref) {
final groupValue = ref.watch(replaceDownloadedFileState);
final replaceAll = ref.watch(replaceDownloadedFileState);
return AlertDialog(
title: Text(context.l10n.track_exists(track.name)),
content: RadioGroup(
value: groupValue,
value: replaceAll,
onChanged: (value) {
ref.read(replaceDownloadedFileState.notifier).state = value;
},

View File

@ -89,7 +89,7 @@ class TrackPresentationActionsSection extends HookConsumerWidget {
) ??
false;
if (confirmed != true) return;
downloader.batchAddToQueue(fullTrackObjects);
downloader.addAllToQueue(fullTrackObjects);
notifier.deselectAllTracks();
if (!context.mounted) return;
showToastForAction(context, action, fullTrackObjects.length);

View File

@ -1,5 +1,3 @@
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
@ -44,7 +42,7 @@ class TrackOptions extends HookConsumerWidget {
:isActiveTrack,
:isAuthenticated,
:isLiked,
:progressNotifier
:downloadTask
) = ref.watch(trackOptionsStateProvider(track));
final isLocalTrack = track is SpotubeLocalTrackObject;
@ -211,12 +209,19 @@ class TrackOptions extends HookConsumerWidget {
},
enabled: !isInDownloadQueue,
leading: isInDownloadQueue
? HookBuilder(builder: (context) {
final progress = useListenable(progressNotifier);
? StreamBuilder(
stream: downloadTask?.downloadedBytesStream,
builder: (context, snapshot) {
final progress = downloadTask?.totalSizeBytes == null ||
downloadTask?.totalSizeBytes == 0
? 0
: (snapshot.data ?? 0) /
downloadTask!.totalSizeBytes!;
return CircularProgressIndicator(
value: progress?.value,
value: progress.toDouble(),
);
})
},
)
: const Icon(SpotubeIcons.download),
title: Text(context.l10n.download_track),
),

168
lib/extensions/dio.dart Normal file
View File

@ -0,0 +1,168 @@
import 'dart:io';
import 'package:dio/dio.dart';
import 'package:path/path.dart';
import 'package:path_provider/path_provider.dart';
extension ChunkDownloaderDioExtension on Dio {
Future<Response> chunkDownload(
String urlPath,
dynamic savePath, {
ProgressCallback? onReceiveProgress,
Map<String, dynamic>? queryParameters,
CancelToken? cancelToken,
bool deleteOnError = true,
FileAccessMode fileAccessMode = FileAccessMode.write,
String lengthHeader = Headers.contentLengthHeader,
Object? data,
Options? options,
int connections = 4,
}) async {
final targetFile = File(savePath.toString());
final tempRootDir = await getTemporaryDirectory();
final tempSaveDir = Directory(
join(
tempRootDir.path,
'Spotube',
'.chunk_dl_${targetFile.uri.pathSegments.last}',
),
);
if (await tempSaveDir.exists()) await tempSaveDir.delete(recursive: true);
await tempSaveDir.create(recursive: true);
try {
int? totalLength;
bool supportsRange = false;
Response? headResp;
try {
headResp = await head(
urlPath,
queryParameters: queryParameters,
options: Options(
headers: {'Range': 'bytes=0-0'},
followRedirects: true,
),
);
} catch (_) {
// Some servers reject HEAD -> ignore
}
final lengthStr = headResp?.headers[lengthHeader]?.first;
if (lengthStr != null) {
final parsed = int.tryParse(lengthStr);
if (parsed != null && parsed > 1) {
totalLength = parsed;
}
}
supportsRange = headResp?.statusCode == 206 ||
headResp?.headers.value(HttpHeaders.acceptRangesHeader) == 'bytes';
if (totalLength == null || totalLength <= 1) {
final resp = await get<ResponseBody>(
urlPath,
options: Options(
responseType: ResponseType.stream,
),
queryParameters: queryParameters,
cancelToken: cancelToken,
);
final len = int.tryParse(resp.headers[lengthHeader]?.first ?? '');
if (len == null || len <= 1) {
// cant safely chunk fallback
return download(
urlPath,
savePath,
onReceiveProgress: onReceiveProgress,
queryParameters: queryParameters,
cancelToken: cancelToken,
deleteOnError: deleteOnError,
options: options,
data: data,
);
}
totalLength = len;
supportsRange =
resp.headers.value(HttpHeaders.acceptRangesHeader)?.toLowerCase() ==
'bytes';
}
if (!supportsRange || connections <= 1) {
return download(
urlPath,
savePath,
onReceiveProgress: onReceiveProgress,
queryParameters: queryParameters,
cancelToken: cancelToken,
deleteOnError: deleteOnError,
options: options,
data: data,
);
}
final chunkSize = (totalLength / connections).ceil();
int downloaded = 0;
final partFiles = List.generate(
connections,
(i) => File(join(tempSaveDir.path, 'part_$i')),
);
final futures = List.generate(connections, (i) async {
final start = i * chunkSize;
final end = (i + 1) * chunkSize - 1;
if (start >= totalLength!) return;
final resp = await get<ResponseBody>(
urlPath,
options: Options(
responseType: ResponseType.stream,
headers: {'Range': 'bytes=$start-$end'},
),
queryParameters: queryParameters,
cancelToken: cancelToken,
);
final file = partFiles[i];
if (await file.exists()) await file.delete();
await file.create(recursive: true);
final sink = file.openWrite();
await for (final chunk in resp.data!.stream) {
sink.add(chunk);
downloaded += chunk.length;
onReceiveProgress?.call(downloaded, totalLength);
}
await sink.close();
});
await Future.wait(futures);
final targetSink = targetFile.openWrite();
for (final f in partFiles) {
await targetSink.addStream(f.openRead());
}
await targetSink.close();
await tempSaveDir.delete(recursive: true);
return Response(
requestOptions: RequestOptions(path: urlPath),
data: targetFile,
statusCode: 200,
statusMessage: 'Chunked download completed ($connections connections)',
);
} catch (e) {
if (deleteOnError) {
if (await targetFile.exists()) await targetFile.delete();
if (await tempSaveDir.exists()) {
await tempSaveDir.delete(recursive: true);
}
}
rethrow;
}
}
}

View File

@ -477,5 +477,18 @@
"available_plugins": "الإضافات المتوفّرة",
"configure_your_own_metadata_plugin": "تهيئة مزوّد بيانات للقائمة/الألبوم/الفنان/المصدر خاص بك",
"audio_scrobblers": "أجهزة تتبع الصوت",
"scrobbling": "التتبع"
"scrobbling": "التتبع",
"download_music_format": "تنسيق تنزيل الموسيقى",
"streaming_music_format": "تنسيق بث الموسيقى",
"download_music_quality": "جودة تنزيل الموسيقى",
"streaming_music_quality": "جودة بث الموسيقى",
"default_metadata_source": "مصدر البيانات الوصفية الافتراضي",
"set_default_metadata_source": "تعيين مصدر البيانات الوصفية الافتراضي",
"default_audio_source": "مصدر الصوت الافتراضي",
"set_default_audio_source": "تعيين مصدر الصوت الافتراضي",
"plugins": "الإضافات",
"configure_plugins": "قم بتكوين مزود البيانات الوصفية ومكونات مصدر الصوت الخاصة بك",
"source": "المصدر: ",
"uncompressed": "غير مضغوط",
"dab_music_source_description": "لمحبي الصوتيات. يوفر تدفقات صوتية عالية الجودة/بدون فقدان. مطابقة دقيقة للمسارات بناءً على ISRC."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "উপলব্ধ প্লাগইনগুলো",
"configure_your_own_metadata_plugin": "নিজস্ব প্লেলিস্ট/অ্যালবাম/শিল্পী/ফিড মেটাডেটা প্রদানকারী কনফিগার করুন",
"audio_scrobblers": "অডিও স্ক্রোব্বলার্স",
"scrobbling": "স্ক্রোব্বলিং"
"scrobbling": "স্ক্রোব্বলিং",
"download_music_format": "গান ডাউনলোডের বিন্যাস",
"streaming_music_format": "গান স্ট্রিমিং এর বিন্যাস",
"download_music_quality": "গান ডাউনলোডের মান",
"streaming_music_quality": "গান স্ট্রিমিং এর মান",
"default_metadata_source": "ডিফল্ট মেটাডেটা উৎস",
"set_default_metadata_source": "ডিফল্ট মেটাডেটা উৎস সেট করুন",
"default_audio_source": "ডিফল্ট অডিও উৎস",
"set_default_audio_source": "ডিফল্ট অডিও উৎস সেট করুন",
"plugins": "প্লাগইন",
"configure_plugins": "আপনার নিজের মেটাডেটা প্রদানকারী এবং অডিও উৎস প্লাগইন কনফিগার করুন",
"source": "উৎস: ",
"uncompressed": "অ-সংকুচিত",
"dab_music_source_description": "অডিওফাইলদের জন্য। উচ্চ-মানের/লসলেস অডিও স্ট্রিম প্রদান করে। সঠিক ISRC ভিত্তিক ট্র্যাক ম্যাচিং।"
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Complements disponibles",
"configure_your_own_metadata_plugin": "Configura el teu propi proveïdor de metadades per llistes/reproduccions àlbum/artista/flux",
"audio_scrobblers": "Scrobblers dàudio",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Format de descàrrega de música",
"streaming_music_format": "Format de reproducció de música en temps real",
"download_music_quality": "Qualitat de descàrrega de música",
"streaming_music_quality": "Qualitat de reproducció de música en temps real",
"default_metadata_source": "Font de metadades per defecte",
"set_default_metadata_source": "Estableix la font de metadades per defecte",
"default_audio_source": "Font d'àudio per defecte",
"set_default_audio_source": "Estableix la font d'àudio per defecte",
"plugins": "Connectors",
"configure_plugins": "Configura els teus propis connectors de proveïdor de metadades i de font d'àudio",
"source": "Font: ",
"uncompressed": "Sense comprimir",
"dab_music_source_description": "Per als audiòfils. Ofereix fluxos d'àudio d'alta qualitat/sense pèrdua. Coincidència precisa de pistes basada en ISRC."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Dostupné pluginy",
"configure_your_own_metadata_plugin": "Nakonfigurujte si vlastního poskytovatele metadat pro playlist/album/umělec/fid",
"audio_scrobblers": "Audio scrobblers",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Formát stahování hudby",
"streaming_music_format": "Formát streamování hudby",
"download_music_quality": "Kvalita stahování hudby",
"streaming_music_quality": "Kvalita streamování hudby",
"default_metadata_source": "Výchozí zdroj metadat",
"set_default_metadata_source": "Nastavit výchozí zdroj metadat",
"default_audio_source": "Výchozí zdroj zvuku",
"set_default_audio_source": "Nastavit výchozí zdroj zvuku",
"plugins": "Pluginy",
"configure_plugins": "Konfigurujte své vlastní pluginy poskytovatele metadat a zdroje zvuku",
"source": "Zdroj: ",
"uncompressed": "Nekomprimováno",
"dab_music_source_description": "Pro audiofily. Poskytuje vysoce kvalitní/bezztrátové zvukové toky. Přesná shoda skladeb na základě ISRC."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Verfügbare Plugins",
"configure_your_own_metadata_plugin": "Eigenen Anbieter für Playlist-/Album-/Künstler-/Feed-Metadaten konfigurieren",
"audio_scrobblers": "Audio-Scrobbler",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Musik-Downloadformat",
"streaming_music_format": "Musik-Streamingformat",
"download_music_quality": "Musik-Downloadqualität",
"streaming_music_quality": "Musik-Streamingqualität",
"default_metadata_source": "Standard-Metadatenquelle",
"set_default_metadata_source": "Standard-Metadatenquelle festlegen",
"default_audio_source": "Standard-Audioquelle",
"set_default_audio_source": "Standard-Audioquelle festlegen",
"plugins": "Plugins",
"configure_plugins": "Richte deine eigenen Metadatenanbieter- und Audioquellen-Plugins ein",
"source": "Quelle: ",
"uncompressed": "Unkomprimiert",
"dab_music_source_description": "Für Audiophile. Bietet hochwertige/verlustfreie Audiostreams. Präzises ISRC-basiertes Track-Matching."
}

View File

@ -264,8 +264,10 @@
"change_cover": "Change cover",
"add_cover": "Add cover",
"restore_defaults": "Restore defaults",
"download_music_codec": "Download music codec",
"streaming_music_codec": "Streaming music codec",
"download_music_format": "Download music format",
"streaming_music_format": "Streaming music format",
"download_music_quality": "Download music quality",
"streaming_music_quality": "Streaming music quality",
"login_with_lastfm": "Login with Last.fm",
"connect": "Connect",
"disconnect_lastfm": "Disconnect Last.fm",

View File

@ -477,5 +477,18 @@
"available_plugins": "Complementos disponibles",
"configure_your_own_metadata_plugin": "Configura tu propio proveedor de metadatos para listas/álbum/artista/feeds",
"audio_scrobblers": "Scrobblers de audio",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Formato de descarga de música",
"streaming_music_format": "Formato de transmisión de música",
"download_music_quality": "Calidad de descarga de música",
"streaming_music_quality": "Calidad de transmisión de música",
"default_metadata_source": "Fuente de metadatos predeterminada",
"set_default_metadata_source": "Establecer fuente de metadatos predeterminada",
"default_audio_source": "Fuente de audio predeterminada",
"set_default_audio_source": "Establecer fuente de audio predeterminada",
"plugins": "Plugins",
"configure_plugins": "Configura tus propios plugins de proveedor de metadatos y fuente de audio",
"source": "Fuente: ",
"uncompressed": "Sin comprimir",
"dab_music_source_description": "Para audiófilos. Proporciona transmisiones de audio de alta calidad/sin pérdida. Coincidencia precisa de pistas basada en ISRC."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Eskaintzen diren pluginak",
"configure_your_own_metadata_plugin": "Konfiguratu zureko playlists-/album-/artista-/feed-metadaten hornitzailea",
"audio_scrobblers": "Audio scrobbler-ak",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Musika deskargatzeko formatua",
"streaming_music_format": "Musika streaming bidezko formatua",
"download_music_quality": "Musika deskargaren kalitatea",
"streaming_music_quality": "Streaming bidezko musika kalitatea",
"default_metadata_source": "Metadatu-iturburu lehenetsia",
"set_default_metadata_source": "Ezarri metadatu-iturburu lehenetsia",
"default_audio_source": "Audio-iturburu lehenetsia",
"set_default_audio_source": "Ezarri audio-iturburu lehenetsia",
"plugins": "Pluginak",
"configure_plugins": "Konfiguratu zure metadatu-hornitzaile eta audio-iturburu pluginak",
"source": "Iturburua: ",
"uncompressed": "Konprimitu gabea",
"dab_music_source_description": "Audiozaleentzat. Kalitate handiko/galerarik gabeko audio-streamak eskaintzen ditu. ISRC oinarritutako pistaren parekatze zehatza."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "افزونه‌های موجود",
"configure_your_own_metadata_plugin": "پیکربندی ارائه‌دهندهٔ متادیتا برای پلی‌لیست/آلبوم/هنرمند/فید به‌صورت سفارشی",
"audio_scrobblers": "اسکراب‌بلرهای صوتی",
"scrobbling": "اسکراب‌بلینگ"
"scrobbling": "اسکراب‌بلینگ",
"download_music_format": "فرمت دانلود موسیقی",
"streaming_music_format": "فرمت پخش آنلاین موسیقی",
"download_music_quality": "کیفیت دانلود موسیقی",
"streaming_music_quality": "کیفیت پخش آنلاین موسیقی",
"default_metadata_source": "منبع پیش‌فرض فراداده",
"set_default_metadata_source": "تنظیم منبع پیش‌فرض فراداده",
"default_audio_source": "منبع پیش‌فرض صوت",
"set_default_audio_source": "تنظیم منبع پیش‌فرض صوت",
"plugins": "افزونه‌ها",
"configure_plugins": "افزونه‌های منبع صوت و ارائه‌دهنده فراداده خود را پیکربندی کنید",
"source": "منبع: ",
"uncompressed": "بدون فشرده‌سازی",
"dab_music_source_description": "مخصوص علاقه‌مندان صدا. ارائه‌دهنده استریم‌های باکیفیت/بدون افت. تطبیق دقیق آهنگ بر اساس ISRC."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Saatavilla olevat lisäosat",
"configure_your_own_metadata_plugin": "Määritä oma soittolistan/albumin/artistin/syötteen metatietojen tarjoaja",
"audio_scrobblers": "Äänen scrobblerit",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Musiikin latausmuoto",
"streaming_music_format": "Musiikin suoratoistomuoto",
"download_music_quality": "Musiikin latauslaatu",
"streaming_music_quality": "Musiikin suoratoistolaadun",
"default_metadata_source": "Oletusarvoinen metatietolähde",
"set_default_metadata_source": "Aseta oletusmetatietolähde",
"default_audio_source": "Oletusarvoinen äänilähde",
"set_default_audio_source": "Aseta oletusäänilähde",
"plugins": "Laajennukset",
"configure_plugins": "Määritä omat metatietojen tarjoaja- ja äänilähdelaajennukset",
"source": "Lähde: ",
"uncompressed": "Pakkaamaton",
"dab_music_source_description": "Audiofiileille. Tarjoaa korkealaatuisia/häviöttömiä äänivirtoja. Tarkka ISRC-pohjainen kappaleiden tunnistus."
}

View File

@ -478,5 +478,18 @@
"available_plugins": "Plugins disponibles",
"configure_your_own_metadata_plugin": "Configurer votre propre fournisseur de métadonnées de playlist/album/artiste/flux",
"audio_scrobblers": "Scrobblers audio",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Format de téléchargement de musique",
"streaming_music_format": "Format de streaming de musique",
"download_music_quality": "Qualité de téléchargement de musique",
"streaming_music_quality": "Qualité de streaming de musique",
"default_metadata_source": "Source de métadonnées par défaut",
"set_default_metadata_source": "Définir la source de métadonnées par défaut",
"default_audio_source": "Source audio par défaut",
"set_default_audio_source": "Définir la source audio par défaut",
"plugins": "Plugins",
"configure_plugins": "Configurez vos propres plugins de fournisseur de métadonnées et de source audio",
"source": "Source : ",
"uncompressed": "Non compressé",
"dab_music_source_description": "Pour les audiophiles. Fournit des flux audio de haute qualité/sans perte. Correspondance précise des pistes basée sur ISRC."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "उपलब्ध प्लगइन",
"configure_your_own_metadata_plugin": "अपनी खुद की प्लेलिस्ट/एल्बम/कलाकार/फ़ीड मेटाडेटा प्रदाता कॉन्फ़िगर करें",
"audio_scrobblers": "ऑडियो स्क्रॉबलर्स",
"scrobbling": "स्क्रॉबलिंग"
"scrobbling": "स्क्रॉबलिंग",
"download_music_format": "संगीत डाउनलोड प्रारूप",
"streaming_music_format": "संगीत स्ट्रीमिंग प्रारूप",
"download_music_quality": "संगीत डाउनलोड गुणवत्ता",
"streaming_music_quality": "संगीत स्ट्रीमिंग गुणवत्ता",
"default_metadata_source": "डिफ़ॉल्ट मेटाडेटा स्रोत",
"set_default_metadata_source": "डिफ़ॉल्ट मेटाडेटा स्रोत सेट करें",
"default_audio_source": "डिफ़ॉल्ट ऑडियो स्रोत",
"set_default_audio_source": "डिफ़ॉल्ट ऑडियो स्रोत सेट करें",
"plugins": "प्लगइन्स",
"configure_plugins": "अपने स्वयं के मेटाडेटा प्रदाता और ऑडियो स्रोत प्लगइन्स कॉन्फ़िगर करें",
"source": "स्रोत: ",
"uncompressed": "असंपीड़ित",
"dab_music_source_description": "ऑडियोफाइलों के लिए। उच्च-गुणवत्ता/बिना हानि वाले ऑडियो स्ट्रीम प्रदान करता है। सटीक ISRC आधारित ट्रैक मिलान।"
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Plugin yang tersedia",
"configure_your_own_metadata_plugin": "Konfigurasi penyedia metadata playlist/album/artis/feed Anda sendiri",
"audio_scrobblers": "Scrobblers Audio",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Format unduh musik",
"streaming_music_format": "Format streaming musik",
"download_music_quality": "Kualitas unduh musik",
"streaming_music_quality": "Kualitas streaming musik",
"default_metadata_source": "Sumber metadata default",
"set_default_metadata_source": "Atur sumber metadata default",
"default_audio_source": "Sumber audio default",
"set_default_audio_source": "Atur sumber audio default",
"plugins": "Plugin",
"configure_plugins": "Konfigurasi plugin penyedia metadata dan sumber audio Anda sendiri",
"source": "Sumber: ",
"uncompressed": "Tidak terkompresi",
"dab_music_source_description": "Untuk audiophile. Menyediakan aliran audio berkualitas tinggi/tanpa kehilangan. Pencocokkan trek yang akurat berdasarkan ISRC."
}

View File

@ -478,5 +478,18 @@
"available_plugins": "Plugin disponibili",
"configure_your_own_metadata_plugin": "Configura il tuo provider di metadati per playlist/album/artista/feed",
"audio_scrobblers": "Scrobbler audio",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Formato download musica",
"streaming_music_format": "Formato streaming musica",
"download_music_quality": "Qualità download musica",
"streaming_music_quality": "Qualità streaming musica",
"default_metadata_source": "Fonte metadati predefinita",
"set_default_metadata_source": "Imposta fonte metadati predefinita",
"default_audio_source": "Fonte audio predefinita",
"set_default_audio_source": "Imposta fonte audio predefinita",
"plugins": "Plugin",
"configure_plugins": "Configura i tuoi plugin per fornitore metadati e fonte audio",
"source": "Fonte: ",
"uncompressed": "Non compresso",
"dab_music_source_description": "Per audiophile. Fornisce flussi audio di alta qualità/senza perdita. Abbinamento traccia accurato basato su ISRC."
}

View File

@ -476,5 +476,18 @@
"available_plugins": "利用可能なプラグイン",
"configure_your_own_metadata_plugin": "独自のプレイリスト/アルバム/アーティスト/フィードのメタデータプロバイダーを構成",
"audio_scrobblers": "オーディオスクロッブラー",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "音楽ダウンロード形式",
"streaming_music_format": "音楽ストリーミング形式",
"download_music_quality": "音楽ダウンロード品質",
"streaming_music_quality": "音楽ストリーミング品質",
"default_metadata_source": "デフォルトメタデータソース",
"set_default_metadata_source": "デフォルトメタデータソースを設定",
"default_audio_source": "デフォルトオーディオソース",
"set_default_audio_source": "デフォルトオーディオソースを設定",
"plugins": "プラグイン",
"configure_plugins": "独自のメタデータプロバイダーとオーディオソースプラグインを設定",
"source": "ソース: ",
"uncompressed": "非圧縮",
"dab_music_source_description": "オーディオファイル向け。高品質/ロスレスオーディオストリームを提供。正確なISRCベースのトラックマッチング。"
}

View File

@ -477,5 +477,18 @@
"available_plugins": "ხელმისაწვდომი პლაგინები",
"configure_your_own_metadata_plugin": "დააყენეთ თქვენი საკუთარი პლეილისტის/ალბომის/არტისტის/ფიდის მეტამონაცემების პროვაიდერი",
"audio_scrobblers": "აუდიო სქრობლერები",
"scrobbling": "სქრობლინგი"
"scrobbling": "სქრობლინგი",
"download_music_format": "მუსიკის ჩამოტვირთვის ფორმატი",
"streaming_music_format": "სტრიმინგის მუსიკის ფორმატი",
"download_music_quality": "ჩამოტვირთვის ხარისხი",
"streaming_music_quality": "სტრიმინგის ხარისხი",
"default_metadata_source": "ნაგულისხმევი მეტამონაცემების წყარო",
"set_default_metadata_source": "ნაგულისხმევი მეტამონაცემების წყაროს დაყენება",
"default_audio_source": "ნაგულისხმევი აუდიო წყარო",
"set_default_audio_source": "ნაგულისხმევი აუდიო წყაროს დაყენება",
"plugins": "პლაგინები",
"configure_plugins": "თქვენი საკუთარი მეტამონაცემებისა და აუდიო წყაროს პლაგინების კონფიგურაცია",
"source": "წყარო: ",
"uncompressed": "შეუკუმშავი",
"dab_music_source_description": "აუდიოფილებისთვის. უზრუნველყოფს მაღალი ხარისხის/უკომპრესო აუდიო სტრიმებს. ზუსტი შესაბამისობა ISRC-ის მიხედვით."
}

View File

@ -478,5 +478,18 @@
"available_plugins": "사용 가능한 플러그인",
"configure_your_own_metadata_plugin": "자신만의 플레이리스트/앨범/아티스트/피드 메타데이터 제공자 구성",
"audio_scrobblers": "오디오 스크로블러",
"scrobbling": "스크로블링"
"scrobbling": "스크로블링",
"download_music_format": "다운로드 음악 포맷",
"streaming_music_format": "스트리밍 음악 포맷",
"download_music_quality": "다운로드 음질",
"streaming_music_quality": "스트리밍 음질",
"default_metadata_source": "기본 메타데이터 소스",
"set_default_metadata_source": "기본 메타데이터 소스 설정",
"default_audio_source": "기본 오디오 소스",
"set_default_audio_source": "기본 오디오 소스 설정",
"plugins": "플러그인",
"configure_plugins": "직접 메타데이터 제공자와 오디오 소스 플러그인을 구성하세요",
"source": "출처: ",
"uncompressed": "비압축",
"dab_music_source_description": "오디오파일을 위한 소스입니다. 고음질/무손실 오디오 스트림을 제공하며 ISRC 기반으로 정확한 트랙 매칭을 지원합니다."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "उपलब्ध प्लगइनहरू",
"configure_your_own_metadata_plugin": "तपाईंको आफ्नै प्लेलिस्ट/एल्बम/कलाकार/फिड मेटाडेटा प्रदायक कन्फिगर गर्नुहोस्",
"audio_scrobblers": "अडियो स्क्रब्बलरहरू",
"scrobbling": "स्क्रब्बलिंग"
"scrobbling": "स्क्रब्बलिंग",
"download_music_format": "सङ्गीत डाउनलोड ढाँचा",
"streaming_music_format": "स्ट्रिमिङ सङ्गीत ढाँचा",
"download_music_quality": "डाउनलोड गुणस्तर",
"streaming_music_quality": "स्ट्रिमिङ गुणस्तर",
"default_metadata_source": "पूर्वनिर्धारित मेटाडाटा स्रोत",
"set_default_metadata_source": "पूर्वनिर्धारित मेटाडाटा स्रोत सेट गर्नुहोस्",
"default_audio_source": "पूर्वनिर्धारित अडियो स्रोत",
"set_default_audio_source": "पूर्वनिर्धारित अडियो स्रोत सेट गर्नुहोस्",
"plugins": "प्लगइनहरू",
"configure_plugins": "आफ्नै मेटाडाटा प्रदायक र अडियो स्रोत प्लगइनहरू कन्फिगर गर्नुहोस्",
"source": "स्रोत: ",
"uncompressed": "असंक्षिप्त",
"dab_music_source_description": "अडियोप्रेमीहरूका लागि। उच्च गुणस्तर/लसलेस अडियो स्ट्रिमहरू उपलब्ध गराउँछ। ISRC-मा आधारित सटीक ट्र्याक मिलान।"
}

View File

@ -477,5 +477,19 @@
"available_plugins": "Beschikbare plugins",
"configure_your_own_metadata_plugin": "Configureer uw eigen metadata-aanbieder voor afspeellijst/album/artiest/feed",
"audio_scrobblers": "Audioscrobblers",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Download muziekformaat",
"streaming_music_format": "Streaming muziekformaat",
"download_music_quality": "Downloadkwaliteit",
"streaming_music_quality": "Streamingkwaliteit",
"default_metadata_source": "Standaard metadata-bron",
"set_default_metadata_source": "Standaard metadata-bron instellen",
"default_audio_source": "Standaard audiobron",
"set_default_audio_source": "Standaard audiobron instellen",
"plugins": "Plug-ins",
"configure_plugins": "Configureer je eigen metadata- en audiobron-plug-ins",
"source": "Bron: ",
"uncompressed": "Ongecomprimeerd",
"dab_music_source_description": "Voor audiofielen. Biedt hoge kwaliteit/lossless audiostreams. Nauwkeurige trackmatching op basis van ISRC.",
"audio_source": "Audiobron"
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Dostępne wtyczki",
"configure_your_own_metadata_plugin": "Skonfiguruj własnego dostawcę metadanych dla playlisty/albumu/artysty/kanału",
"audio_scrobblers": "Scrobblery audio",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Format pobierania muzyki",
"streaming_music_format": "Format strumieniowania muzyki",
"download_music_quality": "Jakość pobierania",
"streaming_music_quality": "Jakość strumieniowania",
"default_metadata_source": "Domyślne źródło metadanych",
"set_default_metadata_source": "Ustaw domyślne źródło metadanych",
"default_audio_source": "Domyślne źródło audio",
"set_default_audio_source": "Ustaw domyślne źródło audio",
"plugins": "Wtyczki",
"configure_plugins": "Skonfiguruj własne wtyczki dostawców metadanych i źródeł audio",
"source": "Źródło: ",
"uncompressed": "Nieskompresowany",
"dab_music_source_description": "Dla audiofilów. Oferuje strumienie audio wysokiej jakości/lossless. Precyzyjne dopasowanie utworów na podstawie ISRC."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Plugins disponíveis",
"configure_your_own_metadata_plugin": "Configure seu próprio provedor de metadados de playlist/álbum/artista/feed",
"audio_scrobblers": "Scrobblers de áudio",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Formato de download de música",
"streaming_music_format": "Formato de streaming de música",
"download_music_quality": "Qualidade de download",
"streaming_music_quality": "Qualidade de streaming",
"default_metadata_source": "Fonte padrão de metadados",
"set_default_metadata_source": "Definir fonte padrão de metadados",
"default_audio_source": "Fonte de áudio padrão",
"set_default_audio_source": "Definir fonte de áudio padrão",
"plugins": "Plugins",
"configure_plugins": "Configure seus próprios plugins de provedores de metadados e fontes de áudio",
"source": "Fonte: ",
"uncompressed": "Não comprimido",
"dab_music_source_description": "Para audiófilos. Fornece streams de áudio de alta qualidade/sem perdas. Correspondência precisa de faixas baseada em ISRC."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Доступные плагины",
"configure_your_own_metadata_plugin": "Настройте свой собственный поставщик метаданных для плейлиста/альбома/артиста/ленты",
"audio_scrobblers": "Аудио скробблеры",
"scrobbling": "Скробблинг"
"scrobbling": "Скробблинг",
"download_music_format": "Формат загрузки музыки",
"streaming_music_format": "Формат потоковой музыки",
"download_music_quality": "Качество загрузки",
"streaming_music_quality": "Качество стриминга",
"default_metadata_source": "Источник метаданных по умолчанию",
"set_default_metadata_source": "Задать источник метаданных по умолчанию",
"default_audio_source": "Источник аудио по умолчанию",
"set_default_audio_source": "Задать источник аудио по умолчанию",
"plugins": "Плагины",
"configure_plugins": "Настройте собственные плагины провайдеров метаданных и источников аудио",
"source": "Источник: ",
"uncompressed": "Несжатый",
"dab_music_source_description": "Для аудиофилов. Предоставляет высококачественные/lossless аудиопотоки. Точное совпадение треков по ISRC."
}

View File

@ -475,5 +475,18 @@
"available_plugins": "கிடைக்கக்கூடிய பிளகின்கள்",
"configure_your_own_metadata_plugin": "உங்கள் சொந்த பிளேலிஸ்ட்/ஆல்பம்/கலைஞர்/ஊட்ட மெட்டாடேட்டா வழங்குநரை உள்ளமைக்கவும்",
"audio_scrobblers": "ஆடியோ ஸ்க்ரோப்ளர்கள்",
"scrobbling": "ஸ்க்ரோப்ளிங்"
"scrobbling": "ஸ்க்ரோப்ளிங்",
"download_music_format": "இசை பதிவிறக்க வடிவம்",
"streaming_music_format": "இசை ஸ்ட்ரீமிங் வடிவம்",
"download_music_quality": "பதிவிறக்க தரம்",
"streaming_music_quality": "ஸ்ட்ரீமிங் தரம்",
"default_metadata_source": "இயல்புநிலை மெட்டாடேட்டா மூலம்",
"set_default_metadata_source": "இயல்புநிலை மெட்டாடேட்டா மூலத்தை அமை",
"default_audio_source": "இயல்புநிலை ஆடியோ மூலம்",
"set_default_audio_source": "இயல்புநிலை ஆடியோ மூலத்தை அமை",
"plugins": "செருகுநிரல்கள்",
"configure_plugins": "உங்கள் சொந்த மெட்டாடேட்டா வழங்குநர் மற்றும் ஆடியோ மூல செருகுநிரல்களை அமைக்கவும்",
"source": "மூலம்: ",
"uncompressed": "அழுத்தப்படாத",
"dab_music_source_description": "ஆடியோஃபைல்களுக்காக. உயர்தர/லாஸ்லெஸ் ஆடியோ ஸ்ட்ரீம்களை வழங்குகிறது. ISRC அடிப்படையில் துல்லியமான பாடல் பொருத்தம்."
}

View File

@ -478,5 +478,18 @@
"available_plugins": "ปลั๊กอินที่มีอยู่",
"configure_your_own_metadata_plugin": "กำหนดค่าผู้ให้บริการเมตาดาต้าเพลย์ลิสต์/อัลบั้ม/ศิลปิน/ฟีดของคุณเอง",
"audio_scrobblers": "เครื่อง scrobbler เสียง",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "รูปแบบการดาวน์โหลดเพลง",
"streaming_music_format": "รูปแบบการสตรีมเพลง",
"download_music_quality": "คุณภาพการดาวน์โหลด",
"streaming_music_quality": "คุณภาพการสตรีม",
"default_metadata_source": "แหล่งเมตาดาต้าพื้นฐาน",
"set_default_metadata_source": "ตั้งค่าแหล่งเมตาดาต้าพื้นฐาน",
"default_audio_source": "แหล่งเสียงพื้นฐาน",
"set_default_audio_source": "ตั้งค่าแหล่งเสียงพื้นฐาน",
"plugins": "ปลั๊กอิน",
"configure_plugins": "กำหนดค่าปลั๊กอินผู้ให้บริการเมตาดาต้าและแหล่งเสียงของคุณเอง",
"source": "แหล่งที่มา: ",
"uncompressed": "ไม่บีบอัด",
"dab_music_source_description": "สำหรับคนรักเสียงเพลง ให้สตรีมเสียงคุณภาพสูง/ไร้การสูญเสียการบีบอัด การจับคู่แทร็กแม่นยำตาม ISRC"
}

View File

@ -475,5 +475,18 @@
"available_plugins": "Mga available na plugin",
"configure_your_own_metadata_plugin": "I-configure ang iyong sariling playlist/album/artist/feed metadata provider",
"audio_scrobblers": "Mga Audio Scrobbler",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "I-download na format ng musika",
"streaming_music_format": "Format ng streaming ng musika",
"download_music_quality": "Kalidad ng i-download na musika",
"streaming_music_quality": "Kalidad ng streaming ng musika",
"default_metadata_source": "Default na pinagmulan ng metadata",
"set_default_metadata_source": "Itakda ang default na pinagmulan ng metadata",
"default_audio_source": "Default na pinagmulan ng audio",
"set_default_audio_source": "Itakda ang default na pinagmulan ng audio",
"plugins": "Mga plugin",
"configure_plugins": "I-configure ang sarili mong metadata provider at mga audio source plugin",
"source": "Pinagmulan: ",
"uncompressed": "Hindi naka-compress",
"dab_music_source_description": "Para sa mga audiophile. Nagbibigay ng de-kalidad/walang loss na audio streams. Tumpak na pagtutugma ng track batay sa ISRC."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Mevcut eklentiler",
"configure_your_own_metadata_plugin": "Kendi çalma listenizi/albümünüzü/sanatçınızı/akış meta veri sağlayıcınızı yapılandırın",
"audio_scrobblers": "Ses Scrobbler'lar",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Müzik indirme formatı",
"streaming_music_format": "Müzik akış formatı",
"download_music_quality": "İndirilen müzik kalitesi",
"streaming_music_quality": "Yayınlanan müzik kalitesi",
"default_metadata_source": "Varsayılan meta veri kaynağı",
"set_default_metadata_source": "Varsayılan meta veri kaynağını ayarla",
"default_audio_source": "Varsayılan ses kaynağı",
"set_default_audio_source": "Varsayılan ses kaynağını ayarla",
"plugins": "Eklentiler",
"configure_plugins": "Kendi meta veri sağlayıcı ve ses kaynağı eklentilerinizi yapılandırın",
"source": "Kaynak: ",
"uncompressed": "Sıkıştırılmamış",
"dab_music_source_description": "Audiophile'ler için. Yüksek kaliteli/kayıpsız ses akışları sağlar. Doğru ISRC tabanlı parça eşleştirme."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Доступні плагіни",
"configure_your_own_metadata_plugin": "Налаштуйте свій власний провайдер метаданих для плейлиста/альбому/виконавця/стрічки",
"audio_scrobblers": "Аудіо скробблери",
"scrobbling": "Скроблінг"
"scrobbling": "Скроблінг",
"download_music_format": "Формат завантаження музики",
"streaming_music_format": "Формат потокової музики",
"download_music_quality": "Якість завантаженої музики",
"streaming_music_quality": "Якість потокової музики",
"default_metadata_source": "Джерело метаданих за замовчуванням",
"set_default_metadata_source": "Встановити джерело метаданих за замовчуванням",
"default_audio_source": "Джерело аудіо за замовчуванням",
"set_default_audio_source": "Встановити джерело аудіо за замовчуванням",
"plugins": "Плагіни",
"configure_plugins": "Налаштуйте власні плагіни метаданих і аудіоджерела",
"source": "Джерело: ",
"uncompressed": "Без стиснення",
"dab_music_source_description": "Для аудіофілів. Забезпечує високоякісні/без втрат аудіопотоки. Точна відповідність треків на основі ISRC."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "Các plugin có sẵn",
"configure_your_own_metadata_plugin": "Cấu hình nhà cung cấp siêu dữ liệu danh sách phát/album/nghệ sĩ/nguồn cấp dữ liệu của riêng bạn",
"audio_scrobblers": "Bộ scrobbler âm thanh",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "Định dạng nhạc tải về",
"streaming_music_format": "Định dạng nhạc phát trực tuyến",
"download_music_quality": "Chất lượng nhạc tải về",
"streaming_music_quality": "Chất lượng nhạc phát trực tuyến",
"default_metadata_source": "Nguồn siêu dữ liệu mặc định",
"set_default_metadata_source": "Đặt nguồn siêu dữ liệu mặc định",
"default_audio_source": "Nguồn âm thanh mặc định",
"set_default_audio_source": "Đặt nguồn âm thanh mặc định",
"plugins": "Tiện ích bổ sung",
"configure_plugins": "Cấu hình nhà cung cấp siêu dữ liệu và tiện ích nguồn âm thanh riêng",
"source": "Nguồn: ",
"uncompressed": "Không nén",
"dab_music_source_description": "Dành cho người yêu âm nhạc chất lượng cao. Cung cấp luồng âm thanh chất lượng cao/không nén. Phù hợp bài hát dựa trên ISRC chính xác."
}

View File

@ -477,5 +477,18 @@
"available_plugins": "可用插件",
"configure_your_own_metadata_plugin": "配置您自己的播放列表/专辑/艺人/订阅元数据提供者",
"audio_scrobblers": "音频 Scrobblers",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "下载音乐格式",
"streaming_music_format": "流媒体音乐格式",
"download_music_quality": "下载音乐质量",
"streaming_music_quality": "流媒体音乐质量",
"default_metadata_source": "默认元数据源",
"set_default_metadata_source": "设置默认元数据源",
"default_audio_source": "默认音频源",
"set_default_audio_source": "设置默认音频源",
"plugins": "插件",
"configure_plugins": "配置您自己的元数据提供者和音频源插件",
"source": "来源:",
"uncompressed": "无损",
"dab_music_source_description": "适合发烧友。提供高质量/无损音频流。基于 ISRC 的精确曲目匹配。"
}

View File

@ -477,5 +477,18 @@
"available_plugins": "可用的外掛程式",
"configure_your_own_metadata_plugin": "設定您自己的播放清單/專輯/藝人/動態中繼資料供應商",
"audio_scrobblers": "音訊 Scrobblers",
"scrobbling": "Scrobbling"
"scrobbling": "Scrobbling",
"download_music_format": "下載音樂格式",
"streaming_music_format": "串流音樂格式",
"download_music_quality": "下載音樂品質",
"streaming_music_quality": "串流音樂品質",
"default_metadata_source": "預設中繼資料來源",
"set_default_metadata_source": "設定預設中繼資料來源",
"default_audio_source": "預設音訊來源",
"set_default_audio_source": "設定預設音訊來源",
"plugins": "外掛程式",
"configure_plugins": "配置您自己的中繼資料提供者和音訊來源外掛程式",
"source": "來源:",
"uncompressed": "未壓縮",
"dab_music_source_description": "適合音響發燒友。提供高品質/無損音訊串流。精確的 ISRC 曲目比對。"
}

View File

@ -1743,17 +1743,29 @@ abstract class AppLocalizations {
/// **'Restore defaults'**
String get restore_defaults;
/// No description provided for @download_music_codec.
/// No description provided for @download_music_format.
///
/// In en, this message translates to:
/// **'Download music codec'**
String get download_music_codec;
/// **'Download music format'**
String get download_music_format;
/// No description provided for @streaming_music_codec.
/// No description provided for @streaming_music_format.
///
/// In en, this message translates to:
/// **'Streaming music codec'**
String get streaming_music_codec;
/// **'Streaming music format'**
String get streaming_music_format;
/// No description provided for @download_music_quality.
///
/// In en, this message translates to:
/// **'Download music quality'**
String get download_music_quality;
/// No description provided for @streaming_music_quality.
///
/// In en, this message translates to:
/// **'Streaming music quality'**
String get streaming_music_quality;
/// No description provided for @login_with_lastfm.
///

View File

@ -874,10 +874,16 @@ class AppLocalizationsAr extends AppLocalizations {
String get restore_defaults => 'استعادة الإعدادات الافتراضية';
@override
String get download_music_codec => 'تنزيل ترميز الموسيقى';
String get download_music_format => 'تنسيق تنزيل الموسيقى';
@override
String get streaming_music_codec => 'ترميز الموسيقى بالتدفق';
String get streaming_music_format => 'تنسيق بث الموسيقى';
@override
String get download_music_quality => 'جودة تنزيل الموسيقى';
@override
String get streaming_music_quality => 'جودة بث الموسيقى';
@override
String get login_with_lastfm => 'تسجيل الدخول باستخدام Last.fm';
@ -1443,16 +1449,17 @@ class AppLocalizationsAr extends AppLocalizations {
'تقوم هذه الإضافة بتتبع مقاطعك الموسيقية لإنشاء سجل الاستماع الخاص بك.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'مصدر البيانات الوصفية الافتراضي';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'تعيين مصدر البيانات الوصفية الافتراضي';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'مصدر الصوت الافتراضي';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'تعيين مصدر الصوت الافتراضي';
@override
String get set_default => 'تعيين كافتراضي';
@ -1513,7 +1520,7 @@ class AppLocalizationsAr extends AppLocalizations {
'المدخل لا يتوافق مع التنسيق المطلوب';
@override
String get plugins => 'Plugins';
String get plugins => 'الإضافات';
@override
String get paste_plugin_download_url =>
@ -1539,7 +1546,7 @@ class AppLocalizationsAr extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'قم بتكوين مزود البيانات الوصفية ومكونات مصدر الصوت الخاصة بك';
@override
String get audio_scrobblers => 'أجهزة تتبع الصوت';
@ -1548,12 +1555,12 @@ class AppLocalizationsAr extends AppLocalizations {
String get scrobbling => 'التتبع';
@override
String get source => 'Source: ';
String get source => 'المصدر: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'غير مضغوط';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'لمحبي الصوتيات. يوفر تدفقات صوتية عالية الجودة/بدون فقدان. مطابقة دقيقة للمسارات بناءً على ISRC.';
}

View File

@ -873,10 +873,16 @@ class AppLocalizationsBn extends AppLocalizations {
String get restore_defaults => 'ডিফল্ট সেটিংস পুনরুদ্ধার করুন';
@override
String get download_music_codec => 'সঙ্গীত কোডেক ডাউনলোড করুন';
String get download_music_format => 'গান ডাউনলোডের বিন্যাস';
@override
String get streaming_music_codec => 'স্ট্রিমিং সঙ্গীত কোডেক';
String get streaming_music_format => 'গান স্ট্রিমিং এর বিন্যাস';
@override
String get download_music_quality => 'গান ডাউনলোডের মান';
@override
String get streaming_music_quality => 'গান স্ট্রিমিং এর মান';
@override
String get login_with_lastfm => 'Last.fm দিয়ে লগইন করুন';
@ -1443,16 +1449,16 @@ class AppLocalizationsBn extends AppLocalizations {
'এই প্লাগইনটি আপনার সঙ্গীত স্ক্রোব্বল করে আপনার শোনা ইতিহাস তৈরি করে।';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'ডিফল্ট মেটাডেটা উৎস';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'ডিফল্ট মেটাডেটা উৎস সেট করুন';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'ডিফল্ট অডিও উৎস';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'ডিফল্ট অডিও উৎস সেট করুন';
@override
String get set_default => 'ডিফল্ট হিসাবে নির্ধারণ করুন';
@ -1514,7 +1520,7 @@ class AppLocalizationsBn extends AppLocalizations {
'ইনপুট প্রয়োজনীয় ফরম্যাটের সাথে মেলে না';
@override
String get plugins => 'Plugins';
String get plugins => 'প্লাগইন';
@override
String get paste_plugin_download_url =>
@ -1540,7 +1546,7 @@ class AppLocalizationsBn extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'আপনার নিজের মেটাডেটা প্রদানকারী এবং অডিও উৎস প্লাগইন কনফিগার করুন';
@override
String get audio_scrobblers => 'অডিও স্ক্রোব্বলার্স';
@ -1549,12 +1555,12 @@ class AppLocalizationsBn extends AppLocalizations {
String get scrobbling => 'স্ক্রোব্বলিং';
@override
String get source => 'Source: ';
String get source => 'উৎস: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'অ-সংকুচিত';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'অডিওফাইলদের জন্য। উচ্চ-মানের/লসলেস অডিও স্ট্রিম প্রদান করে। সঠিক ISRC ভিত্তিক ট্র্যাক ম্যাচিং।';
}

View File

@ -876,10 +876,18 @@ class AppLocalizationsCa extends AppLocalizations {
String get restore_defaults => 'Restaura els valors per defecte';
@override
String get download_music_codec => 'Descarrega el codec de música';
String get download_music_format => 'Format de descàrrega de música';
@override
String get streaming_music_codec => 'Codec de música en streaming';
String get streaming_music_format =>
'Format de reproducció de música en temps real';
@override
String get download_music_quality => 'Qualitat de descàrrega de música';
@override
String get streaming_music_quality =>
'Qualitat de reproducció de música en temps real';
@override
String get login_with_lastfm => 'Inicia la sessió amb Last.fm';
@ -1450,16 +1458,18 @@ class AppLocalizationsCa extends AppLocalizations {
'Aquest complement fa scrobbling de la teva música per generar lhistorial descoltes.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Font de metadades per defecte';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'Estableix la font de metadades per defecte';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Font d\'àudio per defecte';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source =>
'Estableix la font d\'àudio per defecte';
@override
String get set_default => 'Establir com a predeterminat';
@ -1523,7 +1533,7 @@ class AppLocalizationsCa extends AppLocalizations {
'Lentrada no coincideix amb el format requerit';
@override
String get plugins => 'Plugins';
String get plugins => 'Connectors';
@override
String get paste_plugin_download_url =>
@ -1549,7 +1559,7 @@ class AppLocalizationsCa extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Configura els teus propis connectors de proveïdor de metadades i de font d\'àudio';
@override
String get audio_scrobblers => 'Scrobblers dàudio';
@ -1558,12 +1568,12 @@ class AppLocalizationsCa extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Font: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Sense comprimir';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Per als audiòfils. Ofereix fluxos d\'àudio d\'alta qualitat/sense pèrdua. Coincidència precisa de pistes basada en ISRC.';
}

View File

@ -869,10 +869,16 @@ class AppLocalizationsCs extends AppLocalizations {
String get restore_defaults => 'Obnovit výchozí';
@override
String get download_music_codec => 'Kodek pro stahování';
String get download_music_format => 'Formát stahování hudby';
@override
String get streaming_music_codec => 'Kodek pro streamování';
String get streaming_music_format => 'Formát streamování hudby';
@override
String get download_music_quality => 'Kvalita stahování hudby';
@override
String get streaming_music_quality => 'Kvalita streamování hudby';
@override
String get login_with_lastfm => 'Přihlásit se pomocí Last.fm';
@ -1442,16 +1448,16 @@ class AppLocalizationsCs extends AppLocalizations {
'Tento plugin scrobbles vaši hudbu pro vytvoření historie poslechů.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Výchozí zdroj metadat';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'Nastavit výchozí zdroj metadat';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Výchozí zdroj zvuku';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Nastavit výchozí zdroj zvuku';
@override
String get set_default => 'Nastavit jako výchozí';
@ -1514,7 +1520,7 @@ class AppLocalizationsCs extends AppLocalizations {
'Vstup neodpovídá požadovanému formátu';
@override
String get plugins => 'Plugins';
String get plugins => 'Pluginy';
@override
String get paste_plugin_download_url =>
@ -1540,7 +1546,7 @@ class AppLocalizationsCs extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Konfigurujte své vlastní pluginy poskytovatele metadat a zdroje zvuku';
@override
String get audio_scrobblers => 'Audio scrobblers';
@ -1549,12 +1555,12 @@ class AppLocalizationsCs extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Zdroj: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Nekomprimováno';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Pro audiofily. Poskytuje vysoce kvalitní/bezztrátové zvukové toky. Přesná shoda skladeb na základě ISRC.';
}

View File

@ -879,10 +879,16 @@ class AppLocalizationsDe extends AppLocalizations {
String get restore_defaults => 'Standardeinstellungen wiederherstellen';
@override
String get download_music_codec => 'Musik-Codec herunterladen';
String get download_music_format => 'Musik-Downloadformat';
@override
String get streaming_music_codec => 'Streaming-Musik-Codec';
String get streaming_music_format => 'Musik-Streamingformat';
@override
String get download_music_quality => 'Musik-Downloadqualität';
@override
String get streaming_music_quality => 'Musik-Streamingqualität';
@override
String get login_with_lastfm => 'Mit Last.fm anmelden';
@ -1455,16 +1461,17 @@ class AppLocalizationsDe extends AppLocalizations {
'Dieses Plugin scrobbelt Ihre Musik, um Ihre Hörhistorie zu erstellen.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Standard-Metadatenquelle';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'Standard-Metadatenquelle festlegen';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Standard-Audioquelle';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Standard-Audioquelle festlegen';
@override
String get set_default => 'Als Standard festlegen';
@ -1552,7 +1559,7 @@ class AppLocalizationsDe extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Richte deine eigenen Metadatenanbieter- und Audioquellen-Plugins ein';
@override
String get audio_scrobblers => 'Audio-Scrobbler';
@ -1561,12 +1568,12 @@ class AppLocalizationsDe extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Quelle: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Unkomprimiert';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Für Audiophile. Bietet hochwertige/verlustfreie Audiostreams. Präzises ISRC-basiertes Track-Matching.';
}

View File

@ -871,10 +871,16 @@ class AppLocalizationsEn extends AppLocalizations {
String get restore_defaults => 'Restore defaults';
@override
String get download_music_codec => 'Download music codec';
String get download_music_format => 'Download music format';
@override
String get streaming_music_codec => 'Streaming music codec';
String get streaming_music_format => 'Streaming music format';
@override
String get download_music_quality => 'Download music quality';
@override
String get streaming_music_quality => 'Streaming music quality';
@override
String get login_with_lastfm => 'Login with Last.fm';

View File

@ -876,10 +876,16 @@ class AppLocalizationsEs extends AppLocalizations {
String get restore_defaults => 'Restaurar valores predeterminados';
@override
String get download_music_codec => 'Descargar códec de música';
String get download_music_format => 'Formato de descarga de música';
@override
String get streaming_music_codec => 'Códec de música en streaming';
String get streaming_music_format => 'Formato de transmisión de música';
@override
String get download_music_quality => 'Calidad de descarga de música';
@override
String get streaming_music_quality => 'Calidad de transmisión de música';
@override
String get login_with_lastfm => 'Iniciar sesión con Last.fm';
@ -1452,16 +1458,18 @@ class AppLocalizationsEs extends AppLocalizations {
'Este complemento scrobblea tu música para generar tu historial de reproducción.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Fuente de metadatos predeterminada';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'Establecer fuente de metadatos predeterminada';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Fuente de audio predeterminada';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source =>
'Establecer fuente de audio predeterminada';
@override
String get set_default => 'Establecer como predeterminado';
@ -1552,7 +1560,7 @@ class AppLocalizationsEs extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Configura tus propios plugins de proveedor de metadatos y fuente de audio';
@override
String get audio_scrobblers => 'Scrobblers de audio';
@ -1561,12 +1569,12 @@ class AppLocalizationsEs extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Fuente: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Sin comprimir';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Para audiófilos. Proporciona transmisiones de audio de alta calidad/sin pérdida. Coincidencia precisa de pistas basada en ISRC.';
}

View File

@ -876,10 +876,16 @@ class AppLocalizationsEu extends AppLocalizations {
String get restore_defaults => 'Berrezarri berezko balioak';
@override
String get download_music_codec => 'Deskargatutako musikaren codec-a';
String get download_music_format => 'Musika deskargatzeko formatua';
@override
String get streaming_music_codec => 'Streaming musikaren codec-a';
String get streaming_music_format => 'Musika streaming bidezko formatua';
@override
String get download_music_quality => 'Musika deskargaren kalitatea';
@override
String get streaming_music_quality => 'Streaming bidezko musika kalitatea';
@override
String get login_with_lastfm => 'Hasi saioa Last.fm-n';
@ -1451,16 +1457,17 @@ class AppLocalizationsEu extends AppLocalizations {
'Plugin honek zure musika scrobbled egiten du zure entzuteen historia sortzeko.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Metadatu-iturburu lehenetsia';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'Ezarri metadatu-iturburu lehenetsia';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Audio-iturburu lehenetsia';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Ezarri audio-iturburu lehenetsia';
@override
String get set_default => 'Lehenetsi gisa ezarri';
@ -1524,7 +1531,7 @@ class AppLocalizationsEu extends AppLocalizations {
'Sarrera ezin da beharrezko formatutik desberdina izan';
@override
String get plugins => 'Plugins';
String get plugins => 'Pluginak';
@override
String get paste_plugin_download_url =>
@ -1550,7 +1557,7 @@ class AppLocalizationsEu extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Konfiguratu zure metadatu-hornitzaile eta audio-iturburu pluginak';
@override
String get audio_scrobblers => 'Audio scrobbler-ak';
@ -1559,12 +1566,12 @@ class AppLocalizationsEu extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Iturburua: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Konprimitu gabea';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Audiozaleentzat. Kalitate handiko/galerarik gabeko audio-streamak eskaintzen ditu. ISRC oinarritutako pistaren parekatze zehatza.';
}

View File

@ -870,10 +870,16 @@ class AppLocalizationsFa extends AppLocalizations {
String get restore_defaults => 'بازیابی پیش فرض ها';
@override
String get download_music_codec => 'دانلود کدک موسیقی';
String get download_music_format => 'فرمت دانلود موسیقی';
@override
String get streaming_music_codec => 'کدک موسیقی استریمینگ';
String get streaming_music_format => 'فرمت پخش آنلاین موسیقی';
@override
String get download_music_quality => 'کیفیت دانلود موسیقی';
@override
String get streaming_music_quality => 'کیفیت پخش آنلاین موسیقی';
@override
String get login_with_lastfm => 'ورود با Last.fm';
@ -1441,16 +1447,16 @@ class AppLocalizationsFa extends AppLocalizations {
'این افزونه موسیقی شما را اسکراب می‌کند تا تاریخچهٔ شنیداری‌تان را تولید کند.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'منبع پیش‌فرض فراداده';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'تنظیم منبع پیش‌فرض فراداده';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'منبع پیش‌فرض صوت';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'تنظیم منبع پیش‌فرض صوت';
@override
String get set_default => 'تنظیم به عنوان پیش‌فرض';
@ -1512,7 +1518,7 @@ class AppLocalizationsFa extends AppLocalizations {
'ورودی با قالب مورد نیاز تطابق ندارد';
@override
String get plugins => 'Plugins';
String get plugins => 'افزونه‌ها';
@override
String get paste_plugin_download_url =>
@ -1538,7 +1544,7 @@ class AppLocalizationsFa extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'افزونه‌های منبع صوت و ارائه‌دهنده فراداده خود را پیکربندی کنید';
@override
String get audio_scrobblers => 'اسکراب‌بلرهای صوتی';
@ -1547,12 +1553,12 @@ class AppLocalizationsFa extends AppLocalizations {
String get scrobbling => 'اسکراب‌بلینگ';
@override
String get source => 'Source: ';
String get source => 'منبع: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'بدون فشرده‌سازی';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'مخصوص علاقه‌مندان صدا. ارائه‌دهنده استریم‌های باکیفیت/بدون افت. تطبیق دقیق آهنگ بر اساس ISRC.';
}

View File

@ -872,10 +872,16 @@ class AppLocalizationsFi extends AppLocalizations {
String get restore_defaults => 'Palauta oletukset';
@override
String get download_music_codec => 'Ladatun musiikin codefc';
String get download_music_format => 'Musiikin latausmuoto';
@override
String get streaming_music_codec => 'Suoratoistetun musiikin codec';
String get streaming_music_format => 'Musiikin suoratoistomuoto';
@override
String get download_music_quality => 'Musiikin latauslaatu';
@override
String get streaming_music_quality => 'Musiikin suoratoistolaadun';
@override
String get login_with_lastfm => 'Kirjaudu sisään Last.fm:llä';
@ -1443,16 +1449,16 @@ class AppLocalizationsFi extends AppLocalizations {
'Tämä lisäosa scrobblaa musiikkisi luodakseen kuunteluhistoriasi.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Oletusarvoinen metatietolähde';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'Aseta oletusmetatietolähde';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Oletusarvoinen äänilähde';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Aseta oletusäänilähde';
@override
String get set_default => 'Aseta oletukseksi';
@ -1512,7 +1518,7 @@ class AppLocalizationsFi extends AppLocalizations {
String get input_does_not_match_format => 'Syöte ei vastaa vaadittua muotoa';
@override
String get plugins => 'Plugins';
String get plugins => 'Laajennukset';
@override
String get paste_plugin_download_url =>
@ -1538,7 +1544,7 @@ class AppLocalizationsFi extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Määritä omat metatietojen tarjoaja- ja äänilähdelaajennukset';
@override
String get audio_scrobblers => 'Äänen scrobblerit';
@ -1547,12 +1553,12 @@ class AppLocalizationsFi extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Lähde: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Pakkaamaton';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Audiofiileille. Tarjoaa korkealaatuisia/häviöttömiä äänivirtoja. Tarkka ISRC-pohjainen kappaleiden tunnistus.';
}

View File

@ -880,10 +880,16 @@ class AppLocalizationsFr extends AppLocalizations {
String get restore_defaults => 'Restaurer les valeurs par défaut';
@override
String get download_music_codec => 'Télécharger le codec musical';
String get download_music_format => 'Format de téléchargement de musique';
@override
String get streaming_music_codec => 'Codec de musique en streaming';
String get streaming_music_format => 'Format de streaming de musique';
@override
String get download_music_quality => 'Qualité de téléchargement de musique';
@override
String get streaming_music_quality => 'Qualité de streaming de musique';
@override
String get login_with_lastfm => 'Se connecter avec Last.fm';
@ -1457,16 +1463,17 @@ class AppLocalizationsFr extends AppLocalizations {
'Ce plugin scrobble votre musique pour générer votre historique d\'écoute.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Source de métadonnées par défaut';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'Définir la source de métadonnées par défaut';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Source audio par défaut';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Définir la source audio par défaut';
@override
String get set_default => 'Définir par défaut';
@ -1557,7 +1564,7 @@ class AppLocalizationsFr extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Configurez vos propres plugins de fournisseur de métadonnées et de source audio';
@override
String get audio_scrobblers => 'Scrobblers audio';
@ -1569,9 +1576,9 @@ class AppLocalizationsFr extends AppLocalizations {
String get source => 'Source : ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Non compressé';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Pour les audiophiles. Fournit des flux audio de haute qualité/sans perte. Correspondance précise des pistes basée sur ISRC.';
}

View File

@ -872,10 +872,16 @@ class AppLocalizationsHi extends AppLocalizations {
String get restore_defaults => 'डिफ़ॉल्ट सेटिंग्स को बहाल करें';
@override
String get download_music_codec => 'संगीत कोडेक डाउनलोड करें';
String get download_music_format => 'संगीत डाउनलोड प्रारूप';
@override
String get streaming_music_codec => 'स्ट्रीमिंग संगीत कोडेक';
String get streaming_music_format => 'संगीत स्ट्रीमिंग प्रारूप';
@override
String get download_music_quality => 'संगीत डाउनलोड गुणवत्ता';
@override
String get streaming_music_quality => 'संगीत स्ट्रीमिंग गुणवत्ता';
@override
String get login_with_lastfm => 'Last.fm से लॉगिन करें';
@ -1448,16 +1454,16 @@ class AppLocalizationsHi extends AppLocalizations {
'यह प्लगइन आपके सुनने के इतिहास को उत्पन्न करने के लिए आपके संगीत को स्क्रॉबल करता है।';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'डिफ़ॉल्ट मेटाडेटा स्रोत';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'डिफ़ॉल्ट मेटाडेटा स्रोत सेट करें';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'डिफ़ॉल्ट ऑडियो स्रोत';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'डिफ़ॉल्ट ऑडियो स्रोत सेट करें';
@override
String get set_default => 'डिफ़ॉल्ट सेट करें';
@ -1518,7 +1524,7 @@ class AppLocalizationsHi extends AppLocalizations {
'इनपुट आवश्यक प्रारूप से मेल नहीं खाता है';
@override
String get plugins => 'Plugins';
String get plugins => 'प्लगइन्स';
@override
String get paste_plugin_download_url =>
@ -1544,7 +1550,7 @@ class AppLocalizationsHi extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'अपने स्वयं के मेटाडेटा प्रदाता और ऑडियो स्रोत प्लगइन्स कॉन्फ़िगर करें';
@override
String get audio_scrobblers => 'ऑडियो स्क्रॉबलर्स';
@ -1553,12 +1559,12 @@ class AppLocalizationsHi extends AppLocalizations {
String get scrobbling => 'स्क्रॉबलिंग';
@override
String get source => 'Source: ';
String get source => 'स्रोत: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'असंपीड़ित';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'ऑडियोफाइलों के लिए। उच्च-गुणवत्ता/बिना हानि वाले ऑडियो स्ट्रीम प्रदान करता है। सटीक ISRC आधारित ट्रैक मिलान।';
}

View File

@ -874,10 +874,16 @@ class AppLocalizationsId extends AppLocalizations {
String get restore_defaults => 'Kembalikan semula';
@override
String get download_music_codec => 'Unduh codec musik';
String get download_music_format => 'Format unduh musik';
@override
String get streaming_music_codec => 'Streaming codec musik';
String get streaming_music_format => 'Format streaming musik';
@override
String get download_music_quality => 'Kualitas unduh musik';
@override
String get streaming_music_quality => 'Kualitas streaming musik';
@override
String get login_with_lastfm => 'Masuk dengan Last.fm';
@ -1449,16 +1455,16 @@ class AppLocalizationsId extends AppLocalizations {
'Plugin ini scrobble musik Anda untuk menghasilkan riwayat mendengarkan Anda.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Sumber metadata default';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'Atur sumber metadata default';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Sumber audio default';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Atur sumber audio default';
@override
String get set_default => 'Atur sebagai bawaan';
@ -1520,7 +1526,7 @@ class AppLocalizationsId extends AppLocalizations {
'Masukan tidak cocok dengan format yang diperlukan';
@override
String get plugins => 'Plugins';
String get plugins => 'Plugin';
@override
String get paste_plugin_download_url =>
@ -1546,7 +1552,7 @@ class AppLocalizationsId extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Konfigurasi plugin penyedia metadata dan sumber audio Anda sendiri';
@override
String get audio_scrobblers => 'Scrobblers Audio';
@ -1555,12 +1561,12 @@ class AppLocalizationsId extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Sumber: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Tidak terkompresi';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Untuk audiophile. Menyediakan aliran audio berkualitas tinggi/tanpa kehilangan. Pencocokkan trek yang akurat berdasarkan ISRC.';
}

View File

@ -874,10 +874,16 @@ class AppLocalizationsIt extends AppLocalizations {
String get restore_defaults => 'Ripristina default';
@override
String get download_music_codec => 'Codec musicale scaricamento';
String get download_music_format => 'Formato download musica';
@override
String get streaming_music_codec => 'Codec musicale streaming';
String get streaming_music_format => 'Formato streaming musica';
@override
String get download_music_quality => 'Qualità download musica';
@override
String get streaming_music_quality => 'Qualità streaming musica';
@override
String get login_with_lastfm => 'Accesso a Last.fm';
@ -1448,16 +1454,17 @@ class AppLocalizationsIt extends AppLocalizations {
'Questo plugin scrobbla la tua musica per generare la tua cronologia di ascolti.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Fonte metadati predefinita';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'Imposta fonte metadati predefinita';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Fonte audio predefinita';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Imposta fonte audio predefinita';
@override
String get set_default => 'Imposta come predefinito';
@ -1519,7 +1526,7 @@ class AppLocalizationsIt extends AppLocalizations {
'L\'input non corrisponde al formato richiesto';
@override
String get plugins => 'Plugins';
String get plugins => 'Plugin';
@override
String get paste_plugin_download_url =>
@ -1545,7 +1552,7 @@ class AppLocalizationsIt extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Configura i tuoi plugin per fornitore metadati e fonte audio';
@override
String get audio_scrobblers => 'Scrobbler audio';
@ -1554,12 +1561,12 @@ class AppLocalizationsIt extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Fonte: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Non compresso';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Per audiophile. Fornisce flussi audio di alta qualità/senza perdita. Abbinamento traccia accurato basato su ISRC.';
}

View File

@ -861,10 +861,16 @@ class AppLocalizationsJa extends AppLocalizations {
String get restore_defaults => '設定を初期化';
@override
String get download_music_codec => 'ダウンロード用の音声コーデック';
String get download_music_format => '音楽ダウンロード形式';
@override
String get streaming_music_codec => 'ストリーミング用の音声コーデック';
String get streaming_music_format => '音楽ストリーミング形式';
@override
String get download_music_quality => '音楽ダウンロード品質';
@override
String get streaming_music_quality => '音楽ストリーミング品質';
@override
String get login_with_lastfm => 'Last.fmでログイン';
@ -1416,16 +1422,16 @@ class AppLocalizationsJa extends AppLocalizations {
String get plugin_scrobbling_info => 'このプラグインは、あなたの音楽をscrobbleして視聴履歴を生成します。';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'デフォルトメタデータソース';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'デフォルトメタデータソースを設定';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'デフォルトオーディオソース';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'デフォルトオーディオソースを設定';
@override
String get set_default => 'デフォルトに設定';
@ -1483,7 +1489,7 @@ class AppLocalizationsJa extends AppLocalizations {
String get input_does_not_match_format => '入力が必須フォーマットと一致しません';
@override
String get plugins => 'Plugins';
String get plugins => 'プラグイン';
@override
String get paste_plugin_download_url =>
@ -1508,8 +1514,7 @@ class AppLocalizationsJa extends AppLocalizations {
String get available_plugins => '利用可能なプラグイン';
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
String get configure_plugins => '独自のメタデータプロバイダーとオーディオソースプラグインを設定';
@override
String get audio_scrobblers => 'オーディオスクロッブラー';
@ -1518,12 +1523,12 @@ class AppLocalizationsJa extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'ソース: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => '非圧縮';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'オーディオファイル向け。高品質/ロスレスオーディオストリームを提供。正確なISRCベースのトラックマッチング。';
}

View File

@ -872,10 +872,16 @@ class AppLocalizationsKa extends AppLocalizations {
String get restore_defaults => 'ნაგულისხმევი პარამეტრების აღდგენა';
@override
String get download_music_codec => 'მუსიკის კოდეკის გადმოწერა';
String get download_music_format => 'მუსიკის ჩამოტვირთვის ფორმატი';
@override
String get streaming_music_codec => 'სტრიმინგ მუსიკის კოდეკი';
String get streaming_music_format => 'სტრიმინგის მუსიკის ფორმატი';
@override
String get download_music_quality => 'ჩამოტვირთვის ხარისხი';
@override
String get streaming_music_quality => 'სტრიმინგის ხარისხი';
@override
String get login_with_lastfm => 'Last.fm-ით შესვლა';
@ -1448,16 +1454,17 @@ class AppLocalizationsKa extends AppLocalizations {
'ეს პლაგინი აწარმოებს თქვენი მუსიკის სქრობლინგს, რათა შექმნას თქვენი მოსმენის ისტორია.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'ნაგულისხმევი მეტამონაცემების წყარო';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'ნაგულისხმევი მეტამონაცემების წყაროს დაყენება';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'ნაგულისხმევი აუდიო წყარო';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'ნაგულისხმევი აუდიო წყაროს დაყენება';
@override
String get set_default => 'ნაგულისხმევად დაყენება';
@ -1520,7 +1527,7 @@ class AppLocalizationsKa extends AppLocalizations {
'შეყვანა არ ემთხვევა საჭირო ფორმატს';
@override
String get plugins => 'Plugins';
String get plugins => 'პლაგინები';
@override
String get paste_plugin_download_url =>
@ -1546,7 +1553,7 @@ class AppLocalizationsKa extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'თქვენი საკუთარი მეტამონაცემებისა და აუდიო წყაროს პლაგინების კონფიგურაცია';
@override
String get audio_scrobblers => 'აუდიო სქრობლერები';
@ -1555,12 +1562,12 @@ class AppLocalizationsKa extends AppLocalizations {
String get scrobbling => 'სქრობლინგი';
@override
String get source => 'Source: ';
String get source => 'წყარო: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'შეუკუმშავი';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'აუდიოფილებისთვის. უზრუნველყოფს მაღალი ხარისხის/უკომპრესო აუდიო სტრიმებს. ზუსტი შესაბამისობა ISRC-ის მიხედვით.';
}

View File

@ -862,10 +862,16 @@ class AppLocalizationsKo extends AppLocalizations {
String get restore_defaults => '기본값으로 복원';
@override
String get download_music_codec => '다운로드 음악 코덱';
String get download_music_format => '다운로드 음악 포맷';
@override
String get streaming_music_codec => '스트리밍 음악 코덱';
String get streaming_music_format => '스트리밍 음악 포맷';
@override
String get download_music_quality => '다운로드 음질';
@override
String get streaming_music_quality => '스트리밍 음질';
@override
String get login_with_lastfm => 'Last.fm에 로그인';
@ -1421,16 +1427,16 @@ class AppLocalizationsKo extends AppLocalizations {
String get plugin_scrobbling_info => '이 플러그인은 음악을 스크로블하여 청취 기록을 생성합니다.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => '기본 메타데이터 소스';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => '기본 메타데이터 소스 설정';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => '기본 오디오 소스';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => '기본 오디오 소스 설정';
@override
String get set_default => '기본값으로 설정';
@ -1488,7 +1494,7 @@ class AppLocalizationsKo extends AppLocalizations {
String get input_does_not_match_format => '입력이 필요한 형식과 일치하지 않습니다';
@override
String get plugins => 'Plugins';
String get plugins => '플러그인';
@override
String get paste_plugin_download_url =>
@ -1512,8 +1518,7 @@ class AppLocalizationsKo extends AppLocalizations {
String get available_plugins => '사용 가능한 플러그인';
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
String get configure_plugins => '직접 메타데이터 제공자와 오디오 소스 플러그인을 구성하세요';
@override
String get audio_scrobblers => '오디오 스크로블러';
@ -1522,12 +1527,12 @@ class AppLocalizationsKo extends AppLocalizations {
String get scrobbling => '스크로블링';
@override
String get source => 'Source: ';
String get source => '출처: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => '비압축';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'오디오파일을 위한 소스입니다. 고음질/무손실 오디오 스트림을 제공하며 ISRC 기반으로 정확한 트랙 매칭을 지원합니다.';
}

View File

@ -880,10 +880,16 @@ class AppLocalizationsNe extends AppLocalizations {
String get restore_defaults => 'पूर्वनिर्धारितहरू पुनः स्थापित गर्नुहोस्';
@override
String get download_music_codec => 'साङ्गीत कोडेक डाउनलोड गर्नुहोस्';
String get download_music_format => 'सङ्गीत डाउनलोड ढाँचा';
@override
String get streaming_music_codec => 'स्ट्रिमिङ साङ्गीत कोडेक';
String get streaming_music_format => 'स्ट्रिमिङ सङ्गीत ढाँचा';
@override
String get download_music_quality => 'डाउनलोड गुणस्तर';
@override
String get streaming_music_quality => 'स्ट्रिमिङ गुणस्तर';
@override
String get login_with_lastfm => 'लास्ट.एफ.एम सँग लगइन गर्नुहोस्';
@ -1454,16 +1460,18 @@ class AppLocalizationsNe extends AppLocalizations {
'यो प्लगइनले तपाईंको सुन्ने इतिहास उत्पन्न गर्न तपाईंको संगीतलाई स्क्रब्बल गर्दछ।';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'पूर्वनिर्धारित मेटाडाटा स्रोत';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'पूर्वनिर्धारित मेटाडाटा स्रोत सेट गर्नुहोस्';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'पूर्वनिर्धारित अडियो स्रोत';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source =>
'पूर्वनिर्धारित अडियो स्रोत सेट गर्नुहोस्';
@override
String get set_default => 'पूर्वनिर्धारित सेट गर्नुहोस्';
@ -1524,7 +1532,7 @@ class AppLocalizationsNe extends AppLocalizations {
String get input_does_not_match_format => 'इनपुट आवश्यक ढाँचासँग मेल खाँदैन';
@override
String get plugins => 'Plugins';
String get plugins => 'प्लगइनहरू';
@override
String get paste_plugin_download_url =>
@ -1550,7 +1558,7 @@ class AppLocalizationsNe extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'आफ्नै मेटाडाटा प्रदायक र अडियो स्रोत प्लगइनहरू कन्फिगर गर्नुहोस्';
@override
String get audio_scrobblers => 'अडियो स्क्रब्बलरहरू';
@ -1559,12 +1567,12 @@ class AppLocalizationsNe extends AppLocalizations {
String get scrobbling => 'स्क्रब्बलिंग';
@override
String get source => 'Source: ';
String get source => 'स्रोत: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'असंक्षिप्त';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'अडियोप्रेमीहरूका लागि। उच्च गुणस्तर/लसलेस अडियो स्ट्रिमहरू उपलब्ध गराउँछ। ISRC-मा आधारित सटीक ट्र्याक मिलान।';
}

View File

@ -815,7 +815,7 @@ class AppLocalizationsNl extends AppLocalizations {
String get search_mode => 'Zoekmodus';
@override
String get audio_source => 'Audio Source';
String get audio_source => 'Audiobron';
@override
String get ok => 'Oké';
@ -872,10 +872,16 @@ class AppLocalizationsNl extends AppLocalizations {
String get restore_defaults => 'Standaardwaarden herstellen';
@override
String get download_music_codec => 'Download-codec';
String get download_music_format => 'Download muziekformaat';
@override
String get streaming_music_codec => 'Streaming-codec';
String get streaming_music_format => 'Streaming muziekformaat';
@override
String get download_music_quality => 'Downloadkwaliteit';
@override
String get streaming_music_quality => 'Streamingkwaliteit';
@override
String get login_with_lastfm => 'Inloggen met Last.fm';
@ -1446,16 +1452,16 @@ class AppLocalizationsNl extends AppLocalizations {
'Deze plugin scrobblet uw muziek om uw luistergeschiedenis te genereren.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Standaard metadata-bron';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'Standaard metadata-bron instellen';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Standaard audiobron';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Standaard audiobron instellen';
@override
String get set_default => 'Instellen als standaard';
@ -1518,7 +1524,7 @@ class AppLocalizationsNl extends AppLocalizations {
'Invoer komt niet overeen met het vereiste formaat';
@override
String get plugins => 'Plugins';
String get plugins => 'Plug-ins';
@override
String get paste_plugin_download_url =>
@ -1544,7 +1550,7 @@ class AppLocalizationsNl extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Configureer je eigen metadata- en audiobron-plug-ins';
@override
String get audio_scrobblers => 'Audioscrobblers';
@ -1553,12 +1559,12 @@ class AppLocalizationsNl extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Bron: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Ongecomprimeerd';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Voor audiofielen. Biedt hoge kwaliteit/lossless audiostreams. Nauwkeurige trackmatching op basis van ISRC.';
}

View File

@ -873,10 +873,16 @@ class AppLocalizationsPl extends AppLocalizations {
String get restore_defaults => 'Przywróć domyślne';
@override
String get download_music_codec => 'Pobierz kodek muzyczny';
String get download_music_format => 'Format pobierania muzyki';
@override
String get streaming_music_codec => 'Kodek strumieniowy muzyki';
String get streaming_music_format => 'Format strumieniowania muzyki';
@override
String get download_music_quality => 'Jakość pobierania';
@override
String get streaming_music_quality => 'Jakość strumieniowania';
@override
String get login_with_lastfm => 'Zaloguj się z Last.fm';
@ -1449,16 +1455,16 @@ class AppLocalizationsPl extends AppLocalizations {
'Ta wtyczka scrobbluje Twoją muzykę, aby wygenerować historię odsłuchań.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Domyślne źródło metadanych';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'Ustaw domyślne źródło metadanych';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Domyślne źródło audio';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Ustaw domyślne źródło audio';
@override
String get set_default => 'Ustaw jako domyślną';
@ -1520,7 +1526,7 @@ class AppLocalizationsPl extends AppLocalizations {
'Wprowadzony tekst nie pasuje do wymaganego formatu';
@override
String get plugins => 'Plugins';
String get plugins => 'Wtyczki';
@override
String get paste_plugin_download_url =>
@ -1546,7 +1552,7 @@ class AppLocalizationsPl extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Skonfiguruj własne wtyczki dostawców metadanych i źródeł audio';
@override
String get audio_scrobblers => 'Scrobblery audio';
@ -1555,12 +1561,12 @@ class AppLocalizationsPl extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Źródło: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Nieskompresowany';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Dla audiofilów. Oferuje strumienie audio wysokiej jakości/lossless. Precyzyjne dopasowanie utworów na podstawie ISRC.';
}

View File

@ -873,10 +873,16 @@ class AppLocalizationsPt extends AppLocalizations {
String get restore_defaults => 'Restaurar padrões';
@override
String get download_music_codec => 'Descarregar codec de música';
String get download_music_format => 'Formato de download de música';
@override
String get streaming_music_codec => 'Codec de streaming de música';
String get streaming_music_format => 'Formato de streaming de música';
@override
String get download_music_quality => 'Qualidade de download';
@override
String get streaming_music_quality => 'Qualidade de streaming';
@override
String get login_with_lastfm => 'Iniciar sessão com o Last.fm';
@ -1446,16 +1452,16 @@ class AppLocalizationsPt extends AppLocalizations {
'Este plugin faz o scrobbling de sua música para gerar seu histórico de audição.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Fonte padrão de metadados';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'Definir fonte padrão de metadados';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Fonte de áudio padrão';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Definir fonte de áudio padrão';
@override
String get set_default => 'Definir como padrão';
@ -1543,7 +1549,7 @@ class AppLocalizationsPt extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Configure seus próprios plugins de provedores de metadados e fontes de áudio';
@override
String get audio_scrobblers => 'Scrobblers de áudio';
@ -1552,12 +1558,12 @@ class AppLocalizationsPt extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Fonte: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Não comprimido';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Para audiófilos. Fornece streams de áudio de alta qualidade/sem perdas. Correspondência precisa de faixas baseada em ISRC.';
}

View File

@ -874,10 +874,16 @@ class AppLocalizationsRu extends AppLocalizations {
String get restore_defaults => 'Восстановить настройки по умолчанию';
@override
String get download_music_codec => 'Загрузить кодек для музыки';
String get download_music_format => 'Формат загрузки музыки';
@override
String get streaming_music_codec => 'Кодек потоковой передачи музыки';
String get streaming_music_format => 'Формат потоковой музыки';
@override
String get download_music_quality => 'Качество загрузки';
@override
String get streaming_music_quality => 'Качество стриминга';
@override
String get login_with_lastfm => 'Войти с помощью Last.fm';
@ -1448,16 +1454,17 @@ class AppLocalizationsRu extends AppLocalizations {
'Этот плагин скробблит вашу музыку для создания вашей истории прослушиваний.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Источник метаданных по умолчанию';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'Задать источник метаданных по умолчанию';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Источник аудио по умолчанию';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Задать источник аудио по умолчанию';
@override
String get set_default => 'Установить по умолчанию';
@ -1520,7 +1527,7 @@ class AppLocalizationsRu extends AppLocalizations {
'Введенные данные не соответствуют требуемому формату';
@override
String get plugins => 'Plugins';
String get plugins => 'Плагины';
@override
String get paste_plugin_download_url =>
@ -1546,7 +1553,7 @@ class AppLocalizationsRu extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Настройте собственные плагины провайдеров метаданных и источников аудио';
@override
String get audio_scrobblers => 'Аудио скробблеры';
@ -1555,12 +1562,12 @@ class AppLocalizationsRu extends AppLocalizations {
String get scrobbling => 'Скробблинг';
@override
String get source => 'Source: ';
String get source => 'Источник: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Несжатый';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Для аудиофилов. Предоставляет высококачественные/lossless аудиопотоки. Точное совпадение треков по ISRC.';
}

View File

@ -879,10 +879,16 @@ class AppLocalizationsTa extends AppLocalizations {
String get restore_defaults => 'இயல்புநிலைகளை மீட்டமை';
@override
String get download_music_codec => 'இசை கோடெக்கை பதிவிறக்கு';
String get download_music_format => 'இசை பதிவிறக்க வடிவம்';
@override
String get streaming_music_codec => 'இசை கோடெக்கை ஸ்ட்ரீம் செய்';
String get streaming_music_format => 'இசை ஸ்ட்ரீமிங் வடிவம்';
@override
String get download_music_quality => 'பதிவிறக்க தரம்';
@override
String get streaming_music_quality => 'ஸ்ட்ரீமிங் தரம்';
@override
String get login_with_lastfm => 'Last.fm உடன் உள்நுழைக';
@ -1455,16 +1461,17 @@ class AppLocalizationsTa extends AppLocalizations {
'இந்த பிளகின் உங்கள் கேட்பதின் வரலாற்றை உருவாக்க உங்கள் இசையை ஸ்க்ரோப்ள் செய்கிறது.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'இயல்புநிலை மெட்டாடேட்டா மூலம்';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'இயல்புநிலை மெட்டாடேட்டா மூலத்தை அமை';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'இயல்புநிலை ஆடியோ மூலம்';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'இயல்புநிலை ஆடியோ மூலத்தை அமை';
@override
String get set_default => 'இயல்புநிலையாக அமைக்கவும்';
@ -1526,7 +1533,7 @@ class AppLocalizationsTa extends AppLocalizations {
'உள்ளீடு தேவையான வடிவத்துடன் பொருந்தவில்லை';
@override
String get plugins => 'Plugins';
String get plugins => 'செருகுநிரல்கள்';
@override
String get paste_plugin_download_url =>
@ -1552,7 +1559,7 @@ class AppLocalizationsTa extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'உங்கள் சொந்த மெட்டாடேட்டா வழங்குநர் மற்றும் ஆடியோ மூல செருகுநிரல்களை அமைக்கவும்';
@override
String get audio_scrobblers => 'ஆடியோ ஸ்க்ரோப்ளர்கள்';
@ -1561,12 +1568,12 @@ class AppLocalizationsTa extends AppLocalizations {
String get scrobbling => 'ஸ்க்ரோப்ளிங்';
@override
String get source => 'Source: ';
String get source => 'மூலம்: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'அழுத்தப்படாத';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'ஆடியோஃபைல்களுக்காக. உயர்தர/லாஸ்லெஸ் ஆடியோ ஸ்ட்ரீம்களை வழங்குகிறது. ISRC அடிப்படையில் துல்லியமான பாடல் பொருத்தம்.';
}

View File

@ -872,10 +872,16 @@ class AppLocalizationsTh extends AppLocalizations {
String get restore_defaults => 'คืนค่าเริ่มต้น';
@override
String get download_music_codec => 'ดาวน์โหลดโคเดคเพลง';
String get download_music_format => 'รูปแบบการดาวน์โหลดเพลง';
@override
String get streaming_music_codec => 'สตรีมมิ่งโคเดคเพลง';
String get streaming_music_format => 'รูปแบบการสตรีมเพลง';
@override
String get download_music_quality => 'คุณภาพการดาวน์โหลด';
@override
String get streaming_music_quality => 'คุณภาพการสตรีม';
@override
String get login_with_lastfm => 'เข้าสู่ระบบด้วย Last.fm';
@ -1440,16 +1446,16 @@ class AppLocalizationsTh extends AppLocalizations {
'ปลั๊กอินนี้จะ scrobble เพลงของคุณเพื่อสร้างประวัติการฟังของคุณ';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'แหล่งเมตาดาต้าพื้นฐาน';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'ตั้งค่าแหล่งเมตาดาต้าพื้นฐาน';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'แหล่งเสียงพื้นฐาน';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'ตั้งค่าแหล่งเสียงพื้นฐาน';
@override
String get set_default => 'ตั้งค่าเริ่มต้น';
@ -1509,7 +1515,7 @@ class AppLocalizationsTh extends AppLocalizations {
String get input_does_not_match_format => 'อินพุตไม่ตรงกับรูปแบบที่ต้องการ';
@override
String get plugins => 'Plugins';
String get plugins => 'ปลั๊กอิน';
@override
String get paste_plugin_download_url =>
@ -1535,7 +1541,7 @@ class AppLocalizationsTh extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'กำหนดค่าปลั๊กอินผู้ให้บริการเมตาดาต้าและแหล่งเสียงของคุณเอง';
@override
String get audio_scrobblers => 'เครื่อง scrobbler เสียง';
@ -1544,12 +1550,12 @@ class AppLocalizationsTh extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'แหล่งที่มา: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'ไม่บีบอัด';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'สำหรับคนรักเสียงเพลง ให้สตรีมเสียงคุณภาพสูง/ไร้การสูญเสียการบีบอัด การจับคู่แทร็กแม่นยำตาม ISRC';
}

View File

@ -878,10 +878,16 @@ class AppLocalizationsTl extends AppLocalizations {
String get restore_defaults => 'Ibalik ang mga default';
@override
String get download_music_codec => 'Codec para sa pag-download ng musika';
String get download_music_format => 'I-download na format ng musika';
@override
String get streaming_music_codec => 'Codec para sa pag-stream ng musika';
String get streaming_music_format => 'Format ng streaming ng musika';
@override
String get download_music_quality => 'Kalidad ng i-download na musika';
@override
String get streaming_music_quality => 'Kalidad ng streaming ng musika';
@override
String get login_with_lastfm => 'Mag-login gamit ang Last.fm';
@ -1456,16 +1462,18 @@ class AppLocalizationsTl extends AppLocalizations {
'Sinis-scrobble ng plugin na ito ang iyong musika upang mabuo ang iyong kasaysayan ng pakikinig.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Default na pinagmulan ng metadata';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'Itakda ang default na pinagmulan ng metadata';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Default na pinagmulan ng audio';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source =>
'Itakda ang default na pinagmulan ng audio';
@override
String get set_default => 'Itakda bilang default';
@ -1527,7 +1535,7 @@ class AppLocalizationsTl extends AppLocalizations {
'Ang input ay hindi tumutugma sa kinakailangang format';
@override
String get plugins => 'Plugins';
String get plugins => 'Mga plugin';
@override
String get paste_plugin_download_url =>
@ -1553,7 +1561,7 @@ class AppLocalizationsTl extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'I-configure ang sarili mong metadata provider at mga audio source plugin';
@override
String get audio_scrobblers => 'Mga Audio Scrobbler';
@ -1562,12 +1570,12 @@ class AppLocalizationsTl extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Pinagmulan: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Hindi naka-compress';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Para sa mga audiophile. Nagbibigay ng de-kalidad/walang loss na audio streams. Tumpak na pagtutugma ng track batay sa ISRC.';
}

View File

@ -875,10 +875,16 @@ class AppLocalizationsTr extends AppLocalizations {
String get restore_defaults => 'Varsayılanları geri yükle';
@override
String get download_music_codec => 'Müzik codec bileşenini indir';
String get download_music_format => 'Müzik indirme formatı';
@override
String get streaming_music_codec => 'Müzik codec\'i akışı';
String get streaming_music_format => 'Müzik akış formatı';
@override
String get download_music_quality => 'İndirilen müzik kalitesi';
@override
String get streaming_music_quality => 'Yayınlanan müzik kalitesi';
@override
String get login_with_lastfm => 'Last.fm ile giriş yap';
@ -1450,16 +1456,17 @@ class AppLocalizationsTr extends AppLocalizations {
'Bu eklenti, dinleme geçmişinizi oluşturmak için müziğinizi scrobble eder.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Varsayılan meta veri kaynağı';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'Varsayılan meta veri kaynağını ayarla';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Varsayılan ses kaynağı';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Varsayılan ses kaynağını ayarla';
@override
String get set_default => 'Varsayılan olarak ayarla';
@ -1520,7 +1527,7 @@ class AppLocalizationsTr extends AppLocalizations {
String get input_does_not_match_format => 'Girdi, gerekli biçimle eşleşmiyor';
@override
String get plugins => 'Plugins';
String get plugins => 'Eklentiler';
@override
String get paste_plugin_download_url =>
@ -1546,7 +1553,7 @@ class AppLocalizationsTr extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Kendi meta veri sağlayıcı ve ses kaynağı eklentilerinizi yapılandırın';
@override
String get audio_scrobblers => 'Ses Scrobbler\'lar';
@ -1555,12 +1562,12 @@ class AppLocalizationsTr extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Kaynak: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Sıkıştırılmamış';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Audiophile\'ler için. Yüksek kaliteli/kayıpsız ses akışları sağlar. Doğru ISRC tabanlı parça eşleştirme.';
}

View File

@ -875,10 +875,16 @@ class AppLocalizationsUk extends AppLocalizations {
String get restore_defaults => 'Відновити налаштування за замовчуванням';
@override
String get download_music_codec => 'Завантажити кодек для музики';
String get download_music_format => 'Формат завантаження музики';
@override
String get streaming_music_codec => 'Кодек потокової передачі музики';
String get streaming_music_format => 'Формат потокової музики';
@override
String get download_music_quality => 'Якість завантаженої музики';
@override
String get streaming_music_quality => 'Якість потокової музики';
@override
String get login_with_lastfm => 'Увійти з Last.fm';
@ -1446,16 +1452,18 @@ class AppLocalizationsUk extends AppLocalizations {
'Цей плагін скроббить вашу музику, щоб створити вашу історію прослуховувань.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Джерело метаданих за замовчуванням';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source =>
'Встановити джерело метаданих за замовчуванням';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Джерело аудіо за замовчуванням';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source =>
'Встановити джерело аудіо за замовчуванням';
@override
String get set_default => 'Встановити за замовчуванням';
@ -1516,7 +1524,7 @@ class AppLocalizationsUk extends AppLocalizations {
'Введені дані не відповідають необхідному формату';
@override
String get plugins => 'Plugins';
String get plugins => 'Плагіни';
@override
String get paste_plugin_download_url =>
@ -1542,7 +1550,7 @@ class AppLocalizationsUk extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Налаштуйте власні плагіни метаданих і аудіоджерела';
@override
String get audio_scrobblers => 'Аудіо скробблери';
@ -1551,12 +1559,12 @@ class AppLocalizationsUk extends AppLocalizations {
String get scrobbling => 'Скроблінг';
@override
String get source => 'Source: ';
String get source => 'Джерело: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Без стиснення';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Для аудіофілів. Забезпечує високоякісні/без втрат аудіопотоки. Точна відповідність треків на основі ISRC.';
}

View File

@ -875,10 +875,16 @@ class AppLocalizationsVi extends AppLocalizations {
String get restore_defaults => 'Khôi phục mặc định';
@override
String get download_music_codec => 'Định dạng tải xuống';
String get download_music_format => 'Định dạng nhạc tải về';
@override
String get streaming_music_codec => 'Định dạng nghe';
String get streaming_music_format => 'Định dạng nhạc phát trực tuyến';
@override
String get download_music_quality => 'Chất lượng nhạc tải về';
@override
String get streaming_music_quality => 'Chất lượng nhạc phát trực tuyến';
@override
String get login_with_lastfm => 'Đăng nhập bằng tài khoản Last.fm';
@ -1450,16 +1456,16 @@ class AppLocalizationsVi extends AppLocalizations {
'Plugin này scrobble nhạc của bạn để tạo lịch sử nghe của bạn.';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => 'Nguồn siêu dữ liệu mặc định';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => 'Đặt nguồn siêu dữ liệu mặc định';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => 'Nguồn âm thanh mặc định';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => 'Đặt nguồn âm thanh mặc định';
@override
String get set_default => 'Đặt làm mặc định';
@ -1522,7 +1528,7 @@ class AppLocalizationsVi extends AppLocalizations {
'Đầu vào không khớp với định dạng yêu cầu';
@override
String get plugins => 'Plugins';
String get plugins => 'Tiện ích bổ sung';
@override
String get paste_plugin_download_url =>
@ -1548,7 +1554,7 @@ class AppLocalizationsVi extends AppLocalizations {
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
'Cấu hình nhà cung cấp siêu dữ liệu và tiện ích nguồn âm thanh riêng';
@override
String get audio_scrobblers => 'Bộ scrobbler âm thanh';
@ -1557,12 +1563,12 @@ class AppLocalizationsVi extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => 'Nguồn: ';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => 'Không nén';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'Dành cho người yêu âm nhạc chất lượng cao. Cung cấp luồng âm thanh chất lượng cao/không nén. Phù hợp bài hát dựa trên ISRC chính xác.';
}

View File

@ -859,10 +859,16 @@ class AppLocalizationsZh extends AppLocalizations {
String get restore_defaults => '恢复默认值';
@override
String get download_music_codec => '下载音乐编解码器';
String get download_music_format => '下载音乐格式';
@override
String get streaming_music_codec => '流媒体音乐编解码器';
String get streaming_music_format => '流媒体音乐格式';
@override
String get download_music_quality => '下载音乐质量';
@override
String get streaming_music_quality => '流媒体音乐质量';
@override
String get login_with_lastfm => '使用 Last.fm 登录';
@ -1412,16 +1418,16 @@ class AppLocalizationsZh extends AppLocalizations {
String get plugin_scrobbling_info => '此插件会 scrobble 您的音乐以生成您的收听历史记录。';
@override
String get default_metadata_source => 'Default metadata source';
String get default_metadata_source => '默认元数据源';
@override
String get set_default_metadata_source => 'Set default metadata source';
String get set_default_metadata_source => '设置默认元数据源';
@override
String get default_audio_source => 'Default audio source';
String get default_audio_source => '默认音频源';
@override
String get set_default_audio_source => 'Set default audio source';
String get set_default_audio_source => '设置默认音频源';
@override
String get set_default => '设为默认';
@ -1478,7 +1484,7 @@ class AppLocalizationsZh extends AppLocalizations {
String get input_does_not_match_format => '输入与所需格式不匹配';
@override
String get plugins => 'Plugins';
String get plugins => '插件';
@override
String get paste_plugin_download_url =>
@ -1502,8 +1508,7 @@ class AppLocalizationsZh extends AppLocalizations {
String get available_plugins => '可用插件';
@override
String get configure_plugins =>
'Configure your own metadata provider and audio source plugins';
String get configure_plugins => '配置您自己的元数据提供者和音频源插件';
@override
String get audio_scrobblers => '音频 Scrobblers';
@ -1512,14 +1517,14 @@ class AppLocalizationsZh extends AppLocalizations {
String get scrobbling => 'Scrobbling';
@override
String get source => 'Source: ';
String get source => '来源:';
@override
String get uncompressed => 'Uncompressed';
String get uncompressed => '无损';
@override
String get dab_music_source_description =>
'For audiophiles. Provides high-quality/lossless audio streams. Accurate ISRC based track matching.';
'适合发烧友。提供高质量/无损音频流。基于 ISRC 的精确曲目匹配。';
}
/// The translations for Chinese, as used in Taiwan (`zh_TW`).
@ -2377,10 +2382,16 @@ class AppLocalizationsZhTw extends AppLocalizationsZh {
String get restore_defaults => '恢復預設值';
@override
String get download_music_codec => '下載音樂編解碼器';
String get download_music_format => '下載音樂格式';
@override
String get streaming_music_codec => '串流音樂編解碼器';
String get streaming_music_format => '串流音樂格式';
@override
String get download_music_quality => '下載音樂品質';
@override
String get streaming_music_quality => '串流音樂品質';
@override
String get login_with_lastfm => '使用 Last.fm 登入';
@ -2929,6 +2940,18 @@ class AppLocalizationsZhTw extends AppLocalizationsZh {
@override
String get plugin_scrobbling_info => '此外掛程式會 Scrobble 您的音樂以產生您的收聽記錄。';
@override
String get default_metadata_source => '預設中繼資料來源';
@override
String get set_default_metadata_source => '設定預設中繼資料來源';
@override
String get default_audio_source => '預設音訊來源';
@override
String get set_default_audio_source => '設定預設音訊來源';
@override
String get set_default => '設為預設';
@ -2983,6 +3006,9 @@ class AppLocalizationsZhTw extends AppLocalizationsZh {
@override
String get input_does_not_match_format => '輸入不符合所需格式';
@override
String get plugins => '外掛程式';
@override
String get paste_plugin_download_url =>
'貼上下載網址、GitHub/Codeberg 儲存庫網址或 .smplug 檔案的直接連結';
@ -3004,9 +3030,22 @@ class AppLocalizationsZhTw extends AppLocalizationsZh {
@override
String get available_plugins => '可用的外掛程式';
@override
String get configure_plugins => '配置您自己的中繼資料提供者和音訊來源外掛程式';
@override
String get audio_scrobblers => '音訊 Scrobblers';
@override
String get scrobbling => 'Scrobbling';
@override
String get source => '來源:';
@override
String get uncompressed => '未壓縮';
@override
String get dab_music_source_description =>
'適合音響發燒友。提供高品質/無損音訊串流。精確的 ISRC 曲目比對。';
}

View File

@ -19,6 +19,7 @@ import 'package:spotube/services/kv_store/kv_store.dart';
import 'package:flutter/widgets.dart' hide Table, Key, View;
import 'package:spotube/modules/settings/color_scheme_picker_dialog.dart';
import 'package:drift/native.dart';
import 'package:spotube/services/logger/logger.dart';
import 'package:spotube/services/youtube_engine/newpipe_engine.dart';
import 'package:spotube/services/youtube_engine/youtube_explode_engine.dart';
import 'package:spotube/services/youtube_engine/yt_dlp_engine.dart';
@ -200,26 +201,41 @@ class AppDatabase extends _$AppDatabase {
});
},
from8To9: (m, schema) async {
await m.renameTable(schema.pluginsTable, "metadata_plugins_table");
await m.renameColumn(
await m
.renameTable(schema.pluginsTable, "metadata_plugins_table")
.catchError((e, stack) => AppLogger.reportError(e, stack));
await m
.renameColumn(
schema.pluginsTable,
"selected",
pluginsTable.selectedForMetadata,
);
await m.addColumn(
)
.catchError((e, stack) => AppLogger.reportError(e, stack));
await m
.addColumn(
schema.pluginsTable,
pluginsTable.selectedForAudioSource,
);
)
.catchError((e, stack) => AppLogger.reportError(e, stack));
},
from9To10: (m, schema) async {
await m.dropColumn(schema.preferencesTable, "piped_instance");
await m.dropColumn(schema.preferencesTable, "invidious_instance");
await m.addColumn(
await m
.dropColumn(schema.preferencesTable, "piped_instance")
.catchError((e, stack) => AppLogger.reportError(e, stack));
await m
.dropColumn(schema.preferencesTable, "invidious_instance")
.catchError((e, stack) => AppLogger.reportError(e, stack));
await m
.addColumn(
schema.sourceMatchTable,
sourceMatchTable.sourceInfo,
);
await customStatement("DROP INDEX IF EXISTS uniq_track_match;");
await m.dropColumn(schema.sourceMatchTable, "source_id");
)
.catchError((e, stack) => AppLogger.reportError(e, stack));
await customStatement("DROP INDEX IF EXISTS uniq_track_match;")
.catchError((e, stack) => AppLogger.reportError(e, stack));
await m
.dropColumn(schema.sourceMatchTable, "source_id")
.catchError((e, stack) => AppLogger.reportError(e, stack));
},
),
);

View File

@ -28,7 +28,7 @@ class HomeNewReleasesSection extends HookConsumerWidget {
if (newReleases.error
case MetadataPluginException(
errorCode: MetadataPluginErrorCode.noDefaultPlugin,
errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin,
message: _,
)) {
return const SizedBox.shrink();

View File

@ -48,7 +48,7 @@ class HomePageBrowseSection extends HookConsumerWidget {
if (browseSections.error
case MetadataPluginException(
errorCode: MetadataPluginErrorCode.noDefaultPlugin,
errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin,
message: _,
)) {
return const SliverFillRemaining(

View File

@ -7,44 +7,19 @@ import 'package:spotube/collections/spotube_icons.dart';
import 'package:spotube/components/image/universal_image.dart';
import 'package:spotube/components/links/artist_link.dart';
import 'package:spotube/components/ui/button_tile.dart';
import 'package:spotube/extensions/context.dart';
import 'package:spotube/models/metadata/metadata.dart';
import 'package:spotube/provider/download_manager_provider.dart';
import 'package:spotube/services/download_manager/download_status.dart';
import 'package:spotube/services/sourced_track/sourced_track.dart';
class DownloadItem extends HookConsumerWidget {
final SpotubeFullTrackObject track;
final DownloadTask task;
const DownloadItem({
super.key,
required this.track,
required this.task,
});
@override
Widget build(BuildContext context, ref) {
final downloadManager = ref.watch(downloadManagerProvider);
final taskStatus = useState<DownloadStatus?>(null);
useEffect(() {
if (track is! SourcedTrack) return null;
final notifier = downloadManager.getStatusNotifier(track);
taskStatus.value = notifier?.value;
void listener() {
taskStatus.value = notifier?.value;
}
notifier?.addListener(listener);
return () {
notifier?.removeListener(listener);
};
}, [track]);
final isQueryingSourceInfo =
taskStatus.value == null || track is! SourcedTrack;
final downloadManager = ref.watch(downloadManagerProvider.notifier);
return ButtonTile(
style: ButtonVariance.ghost,
@ -55,64 +30,46 @@ class DownloadItem extends HookConsumerWidget {
child: UniversalImage(
height: 40,
width: 40,
path: track.album.images.asUrlString(
path: task.track.album.images.asUrlString(
placeholder: ImagePlaceholder.albumArt,
),
),
),
),
title: Text(track.name),
title: Text(task.track.name),
subtitle: ArtistLink(
artists: track.artists,
artists: task.track.artists,
mainAxisAlignment: WrapAlignment.start,
onOverflowArtistClick: () {
context.navigateTo(TrackRoute(trackId: track.id));
context.navigateTo(TrackRoute(trackId: task.track.id));
},
),
trailing: isQueryingSourceInfo
? Text(context.l10n.querying_info).small()
: switch (taskStatus.value!) {
trailing: switch (task.status) {
DownloadStatus.downloading => HookBuilder(builder: (context) {
final taskProgress = useListenable(useMemoized(
() => downloadManager.getProgressNotifier(track),
[track],
));
return StreamBuilder(
stream: task.downloadedBytesStream,
builder: (context, asyncSnapshot) {
final progress =
task.totalSizeBytes == null || task.totalSizeBytes == 0
? 0
: (asyncSnapshot.data ?? 0) / task.totalSizeBytes!;
return Row(
children: [
CircularProgressIndicator(
value: taskProgress?.value ?? 0,
value: progress.toDouble(),
),
const SizedBox(width: 10),
IconButton.ghost(
icon: const Icon(SpotubeIcons.pause),
onPressed: () {
downloadManager.pause(track);
}),
const SizedBox(width: 10),
IconButton.ghost(
icon: const Icon(SpotubeIcons.close),
onPressed: () {
downloadManager.cancel(track);
downloadManager.cancel(task.track);
}),
],
);
});
}),
DownloadStatus.paused => Row(
mainAxisSize: MainAxisSize.min,
children: [
IconButton.ghost(
icon: const Icon(SpotubeIcons.play),
onPressed: () {
downloadManager.resume(track);
}),
const SizedBox(width: 10),
IconButton.ghost(
icon: const Icon(SpotubeIcons.close),
onPressed: () {
downloadManager.cancel(track);
})
],
),
DownloadStatus.failed || DownloadStatus.canceled => SizedBox(
width: 100,
child: Row(
@ -125,7 +82,7 @@ class DownloadItem extends HookConsumerWidget {
IconButton.ghost(
icon: const Icon(SpotubeIcons.refresh),
onPressed: () {
downloadManager.retry(track);
downloadManager.retry(task.track);
},
),
],
@ -136,7 +93,7 @@ class DownloadItem extends HookConsumerWidget {
DownloadStatus.queued => IconButton.ghost(
icon: const Icon(SpotubeIcons.close),
onPressed: () {
downloadManager.removeFromQueue(track);
downloadManager.cancel(task.track);
}),
},
);

View File

@ -43,8 +43,12 @@ class PlayerActions extends HookConsumerWidget {
final downloader = ref.watch(downloadManagerProvider.notifier);
final isInQueue = useMemoized(() {
if (playlist.activeTrack is! SpotubeFullTrackObject) return false;
return downloader
.isActive(playlist.activeTrack! as SpotubeFullTrackObject);
final downloadTask =
downloader.getTaskByTrackId(playlist.activeTrack!.id);
return const [
DownloadStatus.queued,
DownloadStatus.downloading,
].contains(downloadTask?.status);
}, [
playlist.activeTrack,
downloader,

View File

@ -1,6 +1,7 @@
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'package:spotube/components/fallbacks/not_found.dart';
import 'package:spotube/components/image/universal_image.dart';
import 'package:spotube/components/inter_scrollbar/inter_scrollbar.dart';
import 'package:spotube/components/ui/button_tile.dart';
@ -9,8 +10,7 @@ import 'package:spotube/extensions/context.dart';
import 'package:spotube/extensions/duration.dart';
import 'package:spotube/models/metadata/metadata.dart';
import 'package:spotube/provider/audio_player/audio_player.dart';
import 'package:spotube/provider/audio_player/querying_track_info.dart';
import 'package:spotube/provider/server/active_track_sources.dart';
import 'package:spotube/provider/server/sourced_track_provider.dart';
class SiblingTracksSheet extends HookConsumerWidget {
final bool floating;
@ -23,38 +23,45 @@ class SiblingTracksSheet extends HookConsumerWidget {
Widget build(BuildContext context, ref) {
final controller = useScrollController();
final isFetchingActiveTrack = ref.watch(queryingTrackInfoProvider);
final activeTrackSources = ref.watch(activeTrackSourcesProvider);
final activeTrackNotifier = activeTrackSources.asData?.value?.notifier;
final activeTrack = activeTrackSources.asData?.value?.track;
final activeTrackSource = activeTrackSources.asData?.value?.source;
final activeTrack =
ref.watch(audioPlayerProvider.select((e) => e.activeTrack));
if (activeTrack == null || activeTrack is! SpotubeFullTrackObject) {
return const SafeArea(child: NotFound());
}
return HookBuilder(builder: (context) {
final sourcedTrack = ref.watch(sourcedTrackProvider(activeTrack));
final sourcedTrackNotifier =
ref.watch(sourcedTrackProvider(activeTrack).notifier);
final siblings = useMemoized<List<SpotubeAudioSourceMatchObject>>(
() => !isFetchingActiveTrack
? [
if (activeTrackSource != null) activeTrackSource.info,
...?activeTrackSource?.siblings,
() => !sourcedTrack.isLoading
? <SpotubeAudioSourceMatchObject>[
if (sourcedTrack.asData?.value != null)
sourcedTrack.asData!.value.info,
...?sourcedTrack.asData?.value.siblings,
]
: <SpotubeAudioSourceMatchObject>[],
[activeTrackSource, isFetchingActiveTrack],
[sourcedTrack],
);
final previousActiveTrack = usePrevious(activeTrack);
useEffect(() {
/// Populate sibling when active track changes
if (previousActiveTrack?.id == activeTrack?.id) return;
if (activeTrackSource != null && activeTrackSource.siblings.isEmpty) {
activeTrackNotifier?.copyWithSibling();
if (sourcedTrack.asData?.value != null &&
sourcedTrack.asData?.value.siblings.isEmpty == true) {
sourcedTrackNotifier.copyWithSibling();
}
return null;
}, [activeTrack, previousActiveTrack]);
}, [sourcedTrack]);
return SafeArea(
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
Padding(
padding: const EdgeInsets.symmetric(horizontal: 8.0, vertical: 16),
padding:
const EdgeInsets.symmetric(horizontal: 8.0, vertical: 16),
child: Row(
spacing: 5,
children: [
@ -68,7 +75,7 @@ class SiblingTracksSheet extends HookConsumerWidget {
),
AnimatedSwitcher(
duration: const Duration(milliseconds: 300),
child: activeTrackSources.isLoading
child: sourcedTrack.isLoading
? const SizedBox(
width: double.infinity,
child: LinearProgressIndicator(),
@ -107,21 +114,20 @@ class SiblingTracksSheet extends HookConsumerWidget {
: null,
trailing:
Text(sourceInfo.duration.toHumanReadableString()),
subtitle: Flexible(
child: Text(
subtitle: Text(
sourceInfo.artists.join(", "),
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
),
enabled: !isFetchingActiveTrack,
selected: !isFetchingActiveTrack &&
sourceInfo.id == activeTrackSource?.info.id,
enabled: !sourcedTrack.isLoading,
selected: !sourcedTrack.isLoading &&
sourceInfo.id == sourcedTrack.asData?.value.info.id,
onPressed: () async {
if (!isFetchingActiveTrack &&
sourceInfo.id != activeTrackSource?.info.id) {
await activeTrackNotifier
?.swapWithSibling(sourceInfo);
if (!sourcedTrack.isLoading &&
sourceInfo.id !=
sourcedTrack.asData?.value.info.id) {
await sourcedTrackNotifier
.swapWithSibling(sourceInfo);
await ref
.read(audioPlayerProvider.notifier)
.swapActiveSource();
@ -144,5 +150,6 @@ class SiblingTracksSheet extends HookConsumerWidget {
],
),
);
});
}
}

View File

@ -24,7 +24,12 @@ class SidebarFooter extends HookConsumerWidget implements NavigationBarItem {
final theme = Theme.of(context);
final router = AutoRouter.of(context, watch: true);
final mediaQuery = MediaQuery.of(context);
final downloadCount = ref.watch(downloadManagerProvider).$downloadCount;
final downloadCount = ref
.watch(downloadManagerProvider)
.where((e) =>
e.status == DownloadStatus.downloading ||
e.status == DownloadStatus.queued)
.length;
final userSnapshot = ref.watch(metadataPluginUserProvider);
final data = userSnapshot.asData?.value;

View File

@ -25,7 +25,12 @@ class SpotubeNavigationBar extends HookConsumerWidget {
Widget build(BuildContext context, ref) {
final mediaQuery = MediaQuery.of(context);
final downloadCount = ref.watch(downloadManagerProvider).$downloadCount;
final downloadCount = ref
.watch(downloadManagerProvider)
.where((e) =>
e.status == DownloadStatus.downloading ||
e.status == DownloadStatus.queued)
.length;
final layoutMode =
ref.watch(userPreferencesProvider.select((s) => s.layoutMode));

View File

@ -1,46 +0,0 @@
import 'dart:async';
import 'package:flutter_hooks/flutter_hooks.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart';
import 'package:spotube/components/dialogs/replace_downloaded_dialog.dart';
import 'package:spotube/provider/download_manager_provider.dart';
void useDownloaderDialogs(WidgetRef ref) {
final context = useContext();
final showingDialogCompleter = useRef(Completer()..complete());
final downloader = ref.watch(downloadManagerProvider);
useEffect(() {
downloader.onFileExists = (track) async {
if (!context.mounted) return false;
if (!showingDialogCompleter.value.isCompleted) {
await showingDialogCompleter.value.future;
}
final replaceAll = ref.read(replaceDownloadedFileState);
if (replaceAll != null) return replaceAll;
showingDialogCompleter.value = Completer();
if (context.mounted) {
final result = await showDialog<bool>(
context: context,
builder: (context) => ReplaceDownloadedDialog(
track: track,
),
) ??
false;
showingDialogCompleter.value.complete();
return result;
}
// it'll never reach here as root_app is always mounted
return false;
};
return null;
}, [downloader]);
}

View File

@ -17,7 +17,12 @@ class LibraryPage extends HookConsumerWidget {
@override
Widget build(BuildContext context, ref) {
final downloadingCount = ref.watch(downloadManagerProvider).$downloadCount;
final downloadingCount = ref
.watch(downloadManagerProvider)
.where((e) =>
e.status == DownloadStatus.downloading ||
e.status == DownloadStatus.queued)
.length;
final router = context.watchRouter;
final sidebarLibraryTileList = useMemoized(
() => [

View File

@ -55,7 +55,7 @@ class UserAlbumsPage extends HookConsumerWidget {
if (albumsQuery.error
case MetadataPluginException(
errorCode: MetadataPluginErrorCode.noDefaultPlugin,
errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin,
message: _,
)) {
return const Center(child: NoDefaultMetadataPlugin());

View File

@ -60,7 +60,7 @@ class UserArtistsPage extends HookConsumerWidget {
if (artistQuery.error
case MetadataPluginException(
errorCode: MetadataPluginErrorCode.noDefaultPlugin,
errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin,
message: _,
)) {
return const Center(child: NoDefaultMetadataPlugin());

View File

@ -14,9 +14,8 @@ class UserDownloadsPage extends HookConsumerWidget {
@override
Widget build(BuildContext context, ref) {
final downloadManager = ref.watch(downloadManagerProvider);
final history = downloadManager.$history;
final downloadQueue = ref.watch(downloadManagerProvider);
final downloadManagerNotifier = ref.watch(downloadManagerProvider.notifier);
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
@ -28,16 +27,15 @@ class UserDownloadsPage extends HookConsumerWidget {
children: [
Expanded(
child: AutoSizeText(
context.l10n
.currently_downloading(downloadManager.$downloadCount),
context.l10n.currently_downloading(downloadQueue.length),
maxLines: 1,
).semiBold(),
),
const SizedBox(width: 10),
Button.destructive(
onPressed: downloadManager.$downloadCount == 0
onPressed: downloadQueue.isEmpty
? null
: downloadManager.cancelAll,
: downloadManagerNotifier.clearAll,
child: Text(context.l10n.cancel_all),
),
],
@ -46,9 +44,12 @@ class UserDownloadsPage extends HookConsumerWidget {
Expanded(
child: SafeArea(
child: ListView.builder(
itemCount: history.length,
itemCount: downloadQueue.length,
padding: const EdgeInsets.only(bottom: 200),
itemBuilder: (context, index) {
return DownloadItem(track: history.elementAt(index).query);
return DownloadItem(
task: downloadQueue.elementAt(index),
);
},
),
),

View File

@ -14,6 +14,7 @@ import 'package:skeletonizer/skeletonizer.dart';
import 'package:spotube/collections/fake.dart';
import 'package:spotube/collections/spotube_icons.dart';
import 'package:spotube/components/button/back_button.dart';
import 'package:spotube/components/track_presentation/presentation_actions.dart';
import 'package:spotube/extensions/constrains.dart';
import 'package:spotube/extensions/string.dart';
import 'package:spotube/hooks/controllers/use_shadcn_text_editing_controller.dart';
@ -68,6 +69,37 @@ class LocalLibraryPage extends HookConsumerWidget {
}
}
Future<void> shufflePlayLocalTracks(
WidgetRef ref,
List<SpotubeLocalTrackObject> tracks,
) async {
final playlist = ref.read(audioPlayerProvider);
final playback = ref.read(audioPlayerProvider.notifier);
final isPlaylistPlaying = playlist.containsTracks(tracks);
final shuffledTracks = tracks.shuffled();
if (isPlaylistPlaying) return;
await playback.load(
shuffledTracks,
initialIndex: 0,
autoPlay: true,
);
}
Future<void> addToQueueLocalTracks(
BuildContext context,
WidgetRef ref,
List<SpotubeLocalTrackObject> tracks,
) async {
final playlist = ref.read(audioPlayerProvider);
final playback = ref.read(audioPlayerProvider.notifier);
final isPlaylistPlaying = playlist.containsTracks(tracks);
if (isPlaylistPlaying) return;
await playback.addTracks(tracks);
if (!context.mounted) return;
showToastForAction(context, "add-to-queue", tracks.length);
}
@override
Widget build(BuildContext context, ref) {
final scale = context.theme.scaling;
@ -75,8 +107,12 @@ class LocalLibraryPage extends HookConsumerWidget {
final sortBy = useState<SortBy>(SortBy.none);
final playlist = ref.watch(audioPlayerProvider);
final trackSnapshot = ref.watch(localTracksProvider);
final isPlaylistPlaying = playlist.containsTracks(
trackSnapshot.asData?.value.values.flattened.toList() ?? []);
final isPlaylistPlaying = useMemoized(
() => playlist.containsTracks(
trackSnapshot.asData?.value[location] ?? [],
),
[playlist, trackSnapshot, location],
);
final searchController = useShadcnTextEditingController();
useValueListenable(searchController);
@ -162,7 +198,7 @@ class LocalLibraryPage extends HookConsumerWidget {
),
);
if (accepted ?? false) return;
if (accepted != true) return;
final cacheDir = Directory(
await UserPreferencesNotifier.getMusicCacheDir(),
@ -171,6 +207,8 @@ class LocalLibraryPage extends HookConsumerWidget {
if (cacheDir.existsSync()) {
await cacheDir.delete(recursive: true);
}
ref.invalidate(localTracksProvider);
},
),
IconButton.outline(
@ -222,7 +260,10 @@ class LocalLibraryPage extends HookConsumerWidget {
child: Row(
children: [
const Gap(5),
Button.primary(
Tooltip(
tooltip:
TooltipContainer(child: Text(context.l10n.play)).call,
child: IconButton.primary(
onPressed: trackSnapshot.asData?.value != null
? () async {
if (trackSnapshot.asData?.value.isNotEmpty ==
@ -230,18 +271,68 @@ class LocalLibraryPage extends HookConsumerWidget {
if (!isPlaylistPlaying) {
await playLocalTracks(
ref,
trackSnapshot.asData!.value[location] ?? [],
trackSnapshot.asData!.value[location] ??
[],
);
}
}
}
: null,
leading: Icon(
icon: Icon(
isPlaylistPlaying
? SpotubeIcons.stop
: SpotubeIcons.play,
),
child: Text(context.l10n.play),
),
),
const Gap(5),
Tooltip(
tooltip:
TooltipContainer(child: Text(context.l10n.shuffle))
.call,
child: IconButton.outline(
onPressed: trackSnapshot.asData?.value != null
? () async {
if (trackSnapshot.asData?.value.isNotEmpty ==
true) {
if (!isPlaylistPlaying) {
await shufflePlayLocalTracks(
ref,
trackSnapshot.asData!.value[location] ??
[],
);
}
}
}
: null,
enabled: !isPlaylistPlaying,
icon: const Icon(SpotubeIcons.shuffle),
),
),
const Gap(5),
Tooltip(
tooltip: TooltipContainer(
child: Text(context.l10n.add_to_queue))
.call,
child: IconButton.outline(
onPressed: trackSnapshot.asData?.value != null
? () async {
if (trackSnapshot.asData?.value.isNotEmpty ==
true) {
if (!isPlaylistPlaying) {
await addToQueueLocalTracks(
context,
ref,
trackSnapshot.asData!.value[location] ??
[],
);
}
}
}
: null,
enabled: !isPlaylistPlaying,
icon: const Icon(SpotubeIcons.queueAdd),
),
),
const Spacer(),
if (constraints.smAndDown)

View File

@ -13,7 +13,6 @@ import 'package:spotube/extensions/context.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/platform.dart';
// ignore: depend_on_referenced_packages
enum SortBy {
none,

View File

@ -83,7 +83,7 @@ class UserPlaylistsPage extends HookConsumerWidget {
if (playlistsQuery.error
case MetadataPluginException(
errorCode: MetadataPluginErrorCode.noDefaultPlugin,
errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin,
message: _,
)) {
return const Center(child: NoDefaultMetadataPlugin());

View File

@ -9,7 +9,6 @@ import 'package:spotube/modules/root/bottom_player.dart';
import 'package:spotube/modules/root/sidebar/sidebar.dart';
import 'package:spotube/modules/root/spotube_navigation_bar.dart';
import 'package:spotube/hooks/configurators/use_endless_playback.dart';
import 'package:spotube/modules/root/use_downloader_dialogs.dart';
import 'package:spotube/modules/root/use_global_subscriptions.dart';
import 'package:spotube/provider/glance/glance.dart';
@ -25,7 +24,6 @@ class RootAppPage extends HookConsumerWidget {
ref.listen(glanceProvider, (_, __) {});
useGlobalSubscriptions(ref);
useDownloaderDialogs(ref);
useEndlessPlayback(ref);
useCheckYtDlpInstalled(ref);

View File

@ -85,7 +85,7 @@ class SearchPage extends HookConsumerWidget {
child: Builder(builder: (context) {
if (searchChipSnapshot.error
case MetadataPluginException(
errorCode: MetadataPluginErrorCode.noDefaultPlugin,
errorCode: MetadataPluginErrorCode.noDefaultMetadataPlugin,
message: _
)) {
return const NoDefaultMetadataPlugin();

View File

@ -67,8 +67,8 @@ class SettingsPlaybackSection extends HookConsumerWidget {
),
if (sourcePresets.presets.isNotEmpty) ...[
AdaptiveSelectTile(
secondary: const Icon(SpotubeIcons.api),
title: Text(context.l10n.streaming_music_codec),
secondary: const Icon(SpotubeIcons.plugin),
title: Text(context.l10n.streaming_music_format),
value: sourcePresets.selectedStreamingContainerIndex,
options: [
for (final MapEntry(:key, value: preset)
@ -81,8 +81,8 @@ class SettingsPlaybackSection extends HookConsumerWidget {
},
),
AdaptiveSelectTile(
secondary: const Icon(SpotubeIcons.api),
title: const Text("Streaming music quality"),
secondary: const Icon(SpotubeIcons.audioQuality),
title: Text(context.l10n.streaming_music_quality),
value: sourcePresets.selectedStreamingQualityIndex,
options: [
for (final MapEntry(:key, value: quality) in sourcePresets
@ -98,8 +98,8 @@ class SettingsPlaybackSection extends HookConsumerWidget {
},
),
AdaptiveSelectTile(
secondary: const Icon(SpotubeIcons.api),
title: Text(context.l10n.download_music_codec),
secondary: const Icon(SpotubeIcons.plugin),
title: Text(context.l10n.download_music_format),
value: sourcePresets.selectedDownloadingContainerIndex,
options: [
for (final MapEntry(:key, value: preset)
@ -112,8 +112,8 @@ class SettingsPlaybackSection extends HookConsumerWidget {
},
),
AdaptiveSelectTile(
secondary: const Icon(SpotubeIcons.api),
title: const Text("Downloading music quality"),
secondary: const Icon(SpotubeIcons.audioQuality),
title: Text(context.l10n.download_music_quality),
value: sourcePresets.selectedStreamingQualityIndex,
options: [
for (final MapEntry(:key, value: quality) in sourcePresets

View File

@ -144,40 +144,8 @@ class AudioPlayerNotifier extends Notifier<AudioPlayerState> {
}),
audioPlayer.playlistStream.listen((playlist) async {
try {
// Playlist and state has to be in sync. This is only meant for
// the shuffle/re-ordering indices to be in sync
if (playlist.medias.length != state.tracks.length) {
AppLogger.log.w(
"Playlist length does not match state tracks length. Ignoring... "
"Playlist length: ${playlist.medias.length}, "
"State tracks length: ${state.tracks.length}",
);
return;
}
final trackGroupedById = groupBy(
state.tracks,
(query) => query.id,
);
final tracks = <SpotubeTrackObject>[];
for (final media in playlist.medias) {
final track = trackGroupedById[SpotubeMedia.media(media).track.id]
?.firstOrNull;
if (track != null) {
tracks.add(track);
}
}
if (tracks.length != state.tracks.length) {
AppLogger.log.w("Mismatch in tracks after reordering/shuffling.");
final missingTracks =
state.tracks.where((track) => !tracks.contains(track)).toList();
AppLogger.log.w(
"Missing tracks: ${missingTracks.map((e) => e.id).join(", ")}",
);
}
final tracks =
playlist.medias.map((e) => SpotubeMedia.media(e).track).toList();
state = state.copyWith(
tracks: tracks,
@ -434,13 +402,31 @@ class AudioPlayerNotifier extends Notifier<AudioPlayerState> {
return;
}
final currentIndex = state.currentIndex;
final currentTrack = state.activeTrack as SpotubeFullTrackObject;
final swappedMedia = SpotubeMedia(currentTrack);
final oldState = state;
await audioPlayer.stop();
await audioPlayer.addTrackAt(swappedMedia, currentIndex + 1);
await audioPlayer.skipToNext();
await audioPlayer.removeTrack(currentIndex);
await load(
oldState.tracks,
initialIndex: oldState.currentIndex,
autoPlay: true,
);
state = state.copyWith(
collections: oldState.collections,
loopMode: oldState.loopMode,
playing: oldState.playing,
shuffled: false,
);
await audioPlayer.setLoopMode(oldState.loopMode);
await _updatePlayerState(
AudioPlayerStateTableCompanion(
tracks: Value(state.tracks),
currentIndex: Value(state.currentIndex),
collections: Value(state.collections),
loopMode: Value(state.loopMode),
playing: Value(state.playing),
shuffled: Value(state.shuffled),
),
);
}
Future<void> jumpToTrack(SpotubeTrackObject track) async {

View File

@ -51,11 +51,17 @@ class AudioPlayerState with _$AudioPlayerState {
}
bool containsTrack(SpotubeTrackObject track) {
return tracks.any((t) => t.id == track.id);
return tracks.isNotEmpty &&
tracks.any(
(t) =>
t is SpotubeLocalTrackObject && track is SpotubeLocalTrackObject
? t.path == track.path
: t.id == track.id,
);
}
bool containsTracks(List<SpotubeTrackObject> tracks) {
return tracks.every(containsTrack);
return this.tracks.isNotEmpty && tracks.every(containsTrack);
}
bool containsCollection(String collectionId) {

View File

@ -1,268 +1,285 @@
import 'dart:async';
import 'dart:io';
import 'package:collection/collection.dart';
import 'package:dio/dio.dart';
import 'package:hooks_riverpod/hooks_riverpod.dart';
import 'package:metadata_god/metadata_god.dart';
import 'package:path/path.dart';
import 'package:shadcn_flutter/shadcn_flutter.dart' hide join;
import 'package:spotube/collections/routes.dart';
import 'package:spotube/components/dialogs/replace_downloaded_dialog.dart';
import 'package:spotube/extensions/dio.dart';
import 'package:spotube/models/metadata/metadata.dart';
import 'package:spotube/provider/metadata_plugin/audio_source/quality_presets.dart';
import 'package:spotube/provider/server/sourced_track_provider.dart';
import 'package:spotube/services/logger/logger.dart';
import 'package:collection/collection.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter_riverpod/flutter_riverpod.dart';
import 'package:metadata_god/metadata_god.dart';
import 'package:path/path.dart';
import 'package:spotube/provider/user_preferences/user_preferences_provider.dart';
import 'package:spotube/services/download_manager/download_manager.dart';
import 'package:spotube/services/sourced_track/sourced_track.dart';
import 'package:spotube/utils/primitive_utils.dart';
import 'package:spotube/services/logger/logger.dart';
import 'package:spotube/utils/service_utils.dart';
class DownloadManagerProvider extends ChangeNotifier {
DownloadManagerProvider({required this.ref})
: $history = <SourcedTrack>{},
dl = DownloadManager() {
dl.statusStream.listen((event) async {
try {
final (:request, :status) = event;
final sourcedTrack = $history.firstWhereOrNull(
(element) =>
element.getUrlOfQuality(
downloadContainer,
downloadQualityIndex,
) ==
request.url,
);
if (sourcedTrack == null) return;
final savePath = getTrackFileUrl(sourcedTrack);
// related to onFileExists
final oldFile = File("$savePath.old");
// if download failed and old file exists, rename it back
if ((status == DownloadStatus.failed ||
status == DownloadStatus.canceled) &&
await oldFile.exists()) {
await oldFile.rename(savePath);
enum DownloadStatus {
queued,
downloading,
completed,
failed,
canceled,
}
if (status != DownloadStatus.completed ||
//? WebA audiotagging is not supported yet
//? Although in future by converting weba to opus & then tagging it
//? is possible using vorbis comments
downloadContainer.getFileExtension() == "weba") {
class DownloadTask {
final SpotubeFullTrackObject track;
final DownloadStatus status;
final CancelToken cancelToken;
final int? totalSizeBytes;
final StreamController<int> _downloadedBytesStreamController;
Stream<int> get downloadedBytesStream =>
_downloadedBytesStreamController.stream;
DownloadTask({
required this.track,
required this.status,
required this.cancelToken,
this.totalSizeBytes,
StreamController<int>? downloadedBytesStreamController,
}) : _downloadedBytesStreamController =
downloadedBytesStreamController ?? StreamController.broadcast();
DownloadTask copyWith({
SpotubeFullTrackObject? track,
DownloadStatus? status,
CancelToken? cancelToken,
int? totalSizeBytes,
StreamController<int>? downloadedBytesStreamController,
}) {
return DownloadTask(
track: track ?? this.track,
status: status ?? this.status,
cancelToken: cancelToken ?? this.cancelToken,
totalSizeBytes: totalSizeBytes ?? this.totalSizeBytes,
downloadedBytesStreamController:
downloadedBytesStreamController ?? _downloadedBytesStreamController,
);
}
}
class DownloadManagerNotifier extends Notifier<List<DownloadTask>> {
final Dio dio;
DownloadManagerNotifier()
: dio = Dio(),
super();
@override
build() {
ref.onDispose(() {
for (final task in state) {
if (task.status == DownloadStatus.downloading) {
task.cancelToken.cancel();
}
task._downloadedBytesStreamController.close();
}
});
return [];
}
DownloadTask? getTaskByTrackId(String trackId) {
return state.firstWhereOrNull((element) => element.track.id == trackId);
}
void addToQueue(SpotubeFullTrackObject track) {
if (state.any((element) => element.track.id == track.id)) return;
state = [
...state,
DownloadTask(
track: track,
status: DownloadStatus.queued,
cancelToken: CancelToken(),
),
];
ref.read(sourcedTrackProvider(track));
_startDownloading(); // No await should be invoked to avoid stuck UI
}
void addAllToQueue(List<SpotubeFullTrackObject> tracks) {
state = [
...state,
...tracks.map((e) => DownloadTask(
track: e,
status: DownloadStatus.queued,
cancelToken: CancelToken(),
)),
];
ref.read(sourcedTrackProvider(tracks.first));
_startDownloading(); // No await should be invoked to avoid stuck UI
}
void retry(SpotubeFullTrackObject track) {
if (state.firstWhereOrNull((e) => e.track.id == track.id)?.status
case DownloadStatus.canceled || DownloadStatus.failed) {
_setStatus(track, DownloadStatus.queued);
_startDownloading(); // No await should be invoked to avoid stuck UI
}
}
void cancel(SpotubeFullTrackObject track) {
if (state.firstWhereOrNull((e) => e.track.id == track.id)?.status ==
DownloadStatus.failed) {
return;
}
_setStatus(track, DownloadStatus.canceled);
}
void clearAll() {
for (final task in state) {
if (task.status == DownloadStatus.downloading) {
task.cancelToken.cancel();
}
}
state = [];
}
void _setStatus(SpotubeFullTrackObject track, DownloadStatus status) {
state = state.map((e) {
if (e.track.id == track.id) {
if ((status == DownloadStatus.canceled) && e.cancelToken.isCancelled) {
e.cancelToken.cancel();
}
return e.copyWith(status: status);
}
return e;
}).toList();
}
bool _isShowingDialog = false;
Future<bool> _shouldReplaceFileOnExist(DownloadTask task) async {
if (rootNavigatorKey.currentContext == null || _isShowingDialog) {
return false;
}
final replaceAll = ref.read(replaceDownloadedFileState);
if (replaceAll != null) return replaceAll;
_isShowingDialog = true;
try {
return await showDialog<bool>(
context: rootNavigatorKey.currentContext!,
builder: (context) => ReplaceDownloadedDialog(
track: task.track,
),
) ??
false;
} finally {
_isShowingDialog = false;
}
}
Future<void> _downloadTrack(DownloadTask task) async {
try {
_setStatus(task.track, DownloadStatus.downloading);
final track = await ref.read(sourcedTrackProvider(task.track).future);
if (task.cancelToken.isCancelled) {
_setStatus(task.track, DownloadStatus.canceled);
}
final presets = ref.read(audioSourcePresetsProvider);
final container =
presets.presets[presets.selectedDownloadingContainerIndex];
final downloadLocation = ref.read(
userPreferencesProvider.select((value) => value.downloadLocation));
final url = track.getUrlOfQuality(
container,
presets.selectedDownloadingQualityIndex,
);
if (url == null) {
throw Exception("No download URL found for selected codec");
}
final savePath = join(
downloadLocation,
ServiceUtils.sanitizeFilename(
"${track.query.name} - ${track.query.artists.map((e) => e.name).join(", ")}.${container.getFileExtension()}",
),
);
final savePathFile = File(savePath);
if (await savePathFile.exists()) {
// dio automatically replaces the file if it exists so no deletion required
if (!await _shouldReplaceFileOnExist(task)) {
_setStatus(track.query, DownloadStatus.completed);
return;
}
}
final response = await dio.chunkDownload(
url,
savePath,
cancelToken: task.cancelToken,
onReceiveProgress: (count, total) {
if (task.totalSizeBytes == null) {
state = state.map((e) {
if (e.track.id == track.query.id) {
return e.copyWith(totalSizeBytes: total);
}
return e;
}).toList();
}
task._downloadedBytesStreamController.add(count);
},
deleteOnError: true,
fileAccessMode: FileAccessMode.write,
);
if (response.statusCode != null && response.statusCode! < 400) {
_setStatus(track.query, DownloadStatus.completed);
} else {
_setStatus(track.query, DownloadStatus.failed);
return;
}
final file = File(request.path);
if (await oldFile.exists()) {
await oldFile.delete();
}
if (container.getFileExtension() == "weba") return;
final imageBytes = await ServiceUtils.downloadImage(
(sourcedTrack.query.album.images).asUrlString(
(task.track.album.images).asUrlString(
placeholder: ImagePlaceholder.albumArt,
index: 1,
),
);
final metadata = sourcedTrack.query.toMetadata(
fileLength: await file.length(),
imageBytes: imageBytes,
);
await MetadataGod.writeMetadata(
file: file.path,
metadata: metadata,
file: savePath,
metadata: task.track.toMetadata(
fileLength: await savePathFile.length(),
imageBytes: imageBytes,
),
);
} catch (e, stack) {
if (e is! DioException || e.type != DioExceptionType.cancel) {
_setStatus(task.track, DownloadStatus.failed);
AppLogger.reportError(e, stack);
}
});
}
}
Future<bool> Function(SpotubeFullTrackObject track) onFileExists =
(SpotubeFullTrackObject track) async => true;
Future<void> _startDownloading() async {
for (final task in state) {
if (task.status == DownloadStatus.downloading) return;
final Ref<DownloadManagerProvider> ref;
String get downloadDirectory =>
ref.read(userPreferencesProvider.select((s) => s.downloadLocation));
SpotubeAudioSourceContainerPreset get downloadContainer => ref.read(
audioSourcePresetsProvider
.select((s) => s.presets[s.selectedDownloadingContainerIndex]),
);
int get downloadQualityIndex => ref.read(audioSourcePresetsProvider
.select((s) => s.selectedDownloadingQualityIndex));
int get $downloadCount => dl
.getAllDownloads()
.where(
(download) =>
download.status.value == DownloadStatus.downloading ||
download.status.value == DownloadStatus.paused ||
download.status.value == DownloadStatus.queued,
)
.length;
final Set<SourcedTrack> $history;
// these are the tracks which metadata hasn't been fetched yet
final DownloadManager dl;
String getTrackFileUrl(SourcedTrack track) {
final name =
"${track.query.name} - ${track.query.artists.map((e) => e.name).join(", ")}.${downloadContainer.getFileExtension()}";
return join(downloadDirectory, PrimitiveUtils.toSafeFileName(name));
}
bool isActive(SpotubeFullTrackObject track) {
if ($history.any((e) => e.query.id == track.id)) return true;
final sourcedTrack = $history.firstWhereOrNull(
(element) => element.query.id == track.id,
);
if (sourcedTrack == null) return false;
return dl
.getAllDownloads()
.where(
(download) =>
download.status.value == DownloadStatus.downloading ||
download.status.value == DownloadStatus.paused ||
download.status.value == DownloadStatus.queued,
)
.map((e) => e.request.url)
.contains(sourcedTrack.getUrlOfQuality(
downloadContainer,
downloadQualityIndex,
)!);
}
/// For singular downloads
Future<void> addToQueue(SpotubeFullTrackObject track) async {
final sourcedTrack = await ref.read(sourcedTrackProvider(track).future);
final savePath = getTrackFileUrl(sourcedTrack);
final oldFile = File(savePath);
if (await oldFile.exists() && !await onFileExists(track)) {
return;
}
if (await oldFile.exists()) {
await oldFile.rename("$savePath.old");
}
final downloadTask = await dl.addDownload(
sourcedTrack.getUrlOfQuality(downloadContainer, downloadQualityIndex)!,
savePath,
);
if (downloadTask != null) {
$history.add(sourcedTrack);
}
notifyListeners();
}
Future<void> batchAddToQueue(List<SpotubeFullTrackObject> tracks) async {
notifyListeners();
for (final track in tracks) {
if (task.status == DownloadStatus.queued) {
try {
if (track == tracks.first) {
await addToQueue(track);
} else {
await Future.delayed(
const Duration(seconds: 1),
() => addToQueue(track),
);
}
} catch (e) {
AppLogger.reportError(e, StackTrace.current);
continue;
}
}
}
Future<void> removeFromQueue(SpotubeFullTrackObject track) async {
final sourcedTrack = await mapToSourcedTrack(track);
await dl.removeDownload(
sourcedTrack.getUrlOfQuality(downloadContainer, downloadQualityIndex)!);
$history.remove(sourcedTrack);
}
Future<void> pause(SpotubeFullTrackObject track) async {
final sourcedTrack = await mapToSourcedTrack(track);
return dl.pauseDownload(
sourcedTrack.getUrlOfQuality(downloadContainer, downloadQualityIndex)!);
}
Future<void> resume(SpotubeFullTrackObject track) async {
final sourcedTrack = await mapToSourcedTrack(track);
return dl.resumeDownload(
sourcedTrack.getUrlOfQuality(downloadContainer, downloadQualityIndex)!);
}
Future<void> retry(SpotubeFullTrackObject track) {
return addToQueue(track);
}
void cancel(SpotubeFullTrackObject track) async {
final sourcedTrack = await mapToSourcedTrack(track);
return dl.cancelDownload(
sourcedTrack.getUrlOfQuality(downloadContainer, downloadQualityIndex)!);
}
void cancelAll() {
for (final download in dl.getAllDownloads()) {
if (download.status.value == DownloadStatus.completed) continue;
dl.cancelDownload(download.request.url);
}
}
Future<SourcedTrack> mapToSourcedTrack(SpotubeFullTrackObject track) async {
final historicTrack =
$history.firstWhereOrNull((element) => element.query.id == track.id);
if (historicTrack != null) {
return historicTrack;
}
final sourcedTrack = await ref.read(sourcedTrackProvider(track).future);
return sourcedTrack;
}
ValueNotifier<DownloadStatus>? getStatusNotifier(
SpotubeFullTrackObject track,
) {
final sourcedTrack = $history.firstWhereOrNull(
(element) => element.query.id == track.id,
);
if (sourcedTrack == null) {
return null;
}
return dl
.getDownload(sourcedTrack.getUrlOfQuality(
downloadContainer, downloadQualityIndex)!)
?.status;
}
ValueNotifier<double>? getProgressNotifier(SpotubeFullTrackObject track) {
final sourcedTrack = $history.firstWhereOrNull(
(element) => element.query.id == track.id,
);
if (sourcedTrack == null) {
return null;
}
return dl
.getDownload(sourcedTrack.getUrlOfQuality(
downloadContainer, downloadQualityIndex)!)
?.progress;
}
}
final downloadManagerProvider = ChangeNotifierProvider<DownloadManagerProvider>(
(ref) => DownloadManagerProvider(ref: ref),
await _downloadTrack(task);
} finally {
// After completion, check for more queued tasks
// Ignore errors of the prior task to allow next task to complete
await _startDownloading();
}
}
}
}
}
final downloadManagerProvider =
NotifierProvider<DownloadManagerNotifier, List<DownloadTask>>(
DownloadManagerNotifier.new,
);

View File

@ -53,7 +53,7 @@ class PlaybackHistorySummaryNotifier
database.historyTable.itemId.count(distinct: true);
final itemIdCountingCol = database.historyTable.itemId.count();
final durationSumJsonColumn =
database.historyTable.data.jsonExtract<int>(r"$.duration_ms").sum();
database.historyTable.data.jsonExtract<int>(r"$.durationMs").sum();
final artistCountingCol =
database.historyTable.data.jsonExtract<String>(r"$.artists");

View File

@ -12,7 +12,7 @@ final metadataPluginAlbumProvider =
final metadataPlugin = await ref.watch(metadataPluginProvider.future);
if (metadataPlugin == null) {
throw MetadataPluginException.noDefaultPlugin();
throw MetadataPluginException.noDefaultMetadataPlugin();
}
return metadataPlugin.album.getAlbum(id);

View File

@ -12,7 +12,7 @@ final metadataPluginArtistProvider =
final metadataPlugin = await ref.watch(metadataPluginProvider.future);
if (metadataPlugin == null) {
throw MetadataPluginException.noDefaultPlugin();
throw MetadataPluginException.noDefaultMetadataPlugin();
}
return metadataPlugin.artist.getArtist(artistId);

Some files were not shown because too many files have changed in this diff Show More