mirror of
https://github.com/zarzet/SpotiFLAC-Mobile.git
synced 2026-07-04 11:48:00 +02:00
Compare commits
119 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 203e6bc4eb | |||
| 5f1ffbee4e | |||
| b29dc63337 | |||
| 29699117dc | |||
| 3c75f9ecc6 | |||
| 79340703c1 | |||
| df23e3f96c | |||
| d9f788ddeb | |||
| 62afbdcaaa | |||
| 6c578cfd78 | |||
| a17abec799 | |||
| 2a71b70a34 | |||
| 03f77daf19 | |||
| 270b0c1af6 | |||
| 317bb523a4 | |||
| 2c8ad87b7e | |||
| 5e06729029 | |||
| 21bcfe1157 | |||
| 3aeaaaf4f2 | |||
| 3a9d1395db | |||
| 90c46d99d4 | |||
| 96f44fefd4 | |||
| 38a0a76b69 | |||
| 7fc73b6038 | |||
| 6b61dbc2da | |||
| fd3158fd15 | |||
| ff7135bf2c | |||
| 74bac570c7 | |||
| 5f999035c3 | |||
| fa7b5a3559 | |||
| 187821b2ae | |||
| 1435ba9658 | |||
| 62e2e1703c | |||
| 21a732379b | |||
| 8ac035d146 | |||
| d7e7fb065e | |||
| 11d3b8ab3b | |||
| 566e5996bc | |||
| 51618c7dbd | |||
| bdff3a6135 | |||
| ef7cd4ff5d | |||
| 431e437dee | |||
| cebd43e75a | |||
| 17bfbf95f2 | |||
| dad525be40 | |||
| 7dd0dbd594 | |||
| a0bf423a50 | |||
| 288b060983 | |||
| 5ba60d4fd0 | |||
| 07dae97fe6 | |||
| b210f67728 | |||
| 728d1d58c2 | |||
| 6b9650d451 | |||
| 72ae9072bf | |||
| e82263dc14 | |||
| f03b218775 | |||
| c840b59ae1 | |||
| 1213fc449a | |||
| ca21bb0f0c | |||
| 00555b2df6 | |||
| efca120470 | |||
| a178c3943a | |||
| 01ed1f20ad | |||
| e2bd67083e | |||
| 31fb0a87c9 | |||
| ac4d9fc602 | |||
| 8b1b581dbe | |||
| ebdaa24cfc | |||
| 5633e3adf8 | |||
| fcae5e066d | |||
| c312aea75f | |||
| 1e6e19ecd2 | |||
| 0866b04766 | |||
| 78cef8d58e | |||
| ce84aee8da | |||
| 1ba1665215 | |||
| 60fb18c8e2 | |||
| c042b490b8 | |||
| f544b46d97 | |||
| 70759724fe | |||
| fbfe252df6 | |||
| 2c3def8c7b | |||
| 47e67e8299 | |||
| ec15516230 | |||
| 462013bc2a | |||
| 6b5e53864d | |||
| a8a47589c8 | |||
| b9d567d421 | |||
| 81c77af558 | |||
| 1121680da6 | |||
| d31f2e8894 | |||
| 5895a59cb2 | |||
| 3e5e8d7a42 | |||
| 518a7fd2cf | |||
| 6c832d1754 | |||
| d898b5f23e | |||
| c38a1428f1 | |||
| 759eeccc1f | |||
| d0bc3b203c | |||
| 831b68b6cc | |||
| a06111f445 | |||
| 31fdd30c13 | |||
| 867ec4d125 | |||
| 164467f3a2 | |||
| 543cb45c11 | |||
| 80707fc438 | |||
| 3f42128cb9 | |||
| 591a597333 | |||
| ac5f74a48f | |||
| 2d22d85c49 | |||
| 3edfe8e8bb | |||
| 6f9722e05b | |||
| 066d35967e | |||
| 2b932cff70 | |||
| 556c0e1db2 | |||
| 9897d3102e | |||
| 88dfb88bcc | |||
| 75bfe9b3bf | |||
| f4fe74f972 |
@@ -71,7 +71,7 @@ jobs:
|
|||||||
- name: Setup Go
|
- name: Setup Go
|
||||||
uses: actions/setup-go@v5
|
uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
go-version: "1.21"
|
go-version: "1.25"
|
||||||
cache-dependency-path: go_backend/go.sum
|
cache-dependency-path: go_backend/go.sum
|
||||||
|
|
||||||
# Cache Gradle for faster builds
|
# Cache Gradle for faster builds
|
||||||
@@ -174,7 +174,7 @@ jobs:
|
|||||||
- name: Setup Go
|
- name: Setup Go
|
||||||
uses: actions/setup-go@v5
|
uses: actions/setup-go@v5
|
||||||
with:
|
with:
|
||||||
go-version: "1.21"
|
go-version: "1.25"
|
||||||
cache-dependency-path: go_backend/go.sum
|
cache-dependency-path: go_backend/go.sum
|
||||||
|
|
||||||
# Cache CocoaPods
|
# Cache CocoaPods
|
||||||
@@ -194,7 +194,7 @@ jobs:
|
|||||||
working-directory: go_backend
|
working-directory: go_backend
|
||||||
run: |
|
run: |
|
||||||
mkdir -p ../ios/Frameworks
|
mkdir -p ../ios/Frameworks
|
||||||
gomobile bind -target=ios -o ../ios/Frameworks/Gobackend.xcframework .
|
gomobile bind -target=ios -tags ios -o ../ios/Frameworks/Gobackend.xcframework .
|
||||||
env:
|
env:
|
||||||
CGO_ENABLED: 1
|
CGO_ENABLED: 1
|
||||||
|
|
||||||
@@ -249,23 +249,6 @@ jobs:
|
|||||||
channel: "stable"
|
channel: "stable"
|
||||||
cache: true
|
cache: true
|
||||||
|
|
||||||
# Swap pubspec for iOS build (includes ffmpeg_kit_flutter)
|
|
||||||
- name: Use iOS pubspec with FFmpeg plugin
|
|
||||||
run: |
|
|
||||||
cp pubspec.yaml pubspec_android_backup.yaml
|
|
||||||
cp pubspec_ios.yaml pubspec.yaml
|
|
||||||
echo "Swapped to iOS pubspec with ffmpeg_kit_flutter"
|
|
||||||
|
|
||||||
# Swap FFmpeg service for iOS
|
|
||||||
- name: Use iOS FFmpeg service
|
|
||||||
run: |
|
|
||||||
cp lib/services/ffmpeg_service.dart lib/services/ffmpeg_service_android.dart
|
|
||||||
cp build_assets/ffmpeg_service_ios.dart lib/services/ffmpeg_service.dart
|
|
||||||
# Update class name in the swapped file
|
|
||||||
sed -i '' 's/FFmpegServiceIOS/FFmpegService/g' lib/services/ffmpeg_service.dart
|
|
||||||
sed -i '' 's/FFmpegResultIOS/FFmpegResult/g' lib/services/ffmpeg_service.dart
|
|
||||||
echo "Swapped to iOS FFmpeg service"
|
|
||||||
|
|
||||||
- name: Get Flutter dependencies
|
- name: Get Flutter dependencies
|
||||||
run: flutter pub get
|
run: flutter pub get
|
||||||
|
|
||||||
@@ -507,11 +490,13 @@ jobs:
|
|||||||
MESSAGE=$(cat /tmp/telegram_message.txt)
|
MESSAGE=$(cat /tmp/telegram_message.txt)
|
||||||
|
|
||||||
# Send message first (using HTML parse mode)
|
# Send message first (using HTML parse mode)
|
||||||
|
# Use --data-urlencode for proper encoding of special chars (+, &, etc.)
|
||||||
|
# Use || true to ensure file uploads continue even if message fails
|
||||||
curl -s -X POST "https://api.telegram.org/bot${TELEGRAM_BOT_TOKEN}/sendMessage" \
|
curl -s -X POST "https://api.telegram.org/bot${TELEGRAM_BOT_TOKEN}/sendMessage" \
|
||||||
-d chat_id="${TELEGRAM_CHANNEL_ID}" \
|
--data-urlencode "chat_id=${TELEGRAM_CHANNEL_ID}" \
|
||||||
-d text="${MESSAGE}" \
|
--data-urlencode "text=${MESSAGE}" \
|
||||||
-d parse_mode="HTML" \
|
--data-urlencode "parse_mode=HTML" \
|
||||||
-d disable_web_page_preview="true"
|
--data-urlencode "disable_web_page_preview=true" || true
|
||||||
|
|
||||||
# Upload arm64 APK to channel
|
# Upload arm64 APK to channel
|
||||||
if [ -f "$ARM64_APK" ]; then
|
if [ -f "$ARM64_APK" ]; then
|
||||||
|
|||||||
@@ -72,3 +72,4 @@ flutter_*.log
|
|||||||
|
|
||||||
# Development tools
|
# Development tools
|
||||||
tool/
|
tool/
|
||||||
|
.claude/settings.local.json
|
||||||
|
|||||||
+28
-2
@@ -1,8 +1,34 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
## [3.2.1] - 2026-01-22
|
## [3.3.1] - 2026-02-01
|
||||||
|
|
||||||
> **Note:** Next release will use `year.month.day` format (e.g., 26.2.1) and is scheduled for early February. Developer is taking a short break!
|
### Added
|
||||||
|
|
||||||
|
- **Clear All Queue Button**: Cancel all queued downloads with one tap ([#96](https://github.com/zarzet/SpotiFLAC-Mobile/issues/96))
|
||||||
|
- **IDHS Fallback**: Fallback link resolver when SongLink fails (rate limited 8 req/min)
|
||||||
|
- **Lossy Bitrate Options**: MP3 (320/256/192/128kbps), Opus (128/96/64kbps)
|
||||||
|
- **Search Filters**: Filter results by type (Tracks, Artists, Albums, Playlists)
|
||||||
|
- **Album/Playlist Search**: Deezer search now includes albums and playlists
|
||||||
|
- **New Languages**: Turkish (Kaan, BedirhanGltkn), Japanese (Re*Index.(ot_inc))
|
||||||
|
- **Optional All Files Access**: Android 13+ no longer requires full storage access; enable in Settings if needed
|
||||||
|
- **Improved VPN Compatibility**: Better HTTP/2 support for users behind VPN or restricted networks
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
|
||||||
|
- **Amazon Download API**: Switched to AfkarXYZ API
|
||||||
|
- **Qobuz Download API**: Added Jumo API as fallback
|
||||||
|
- **Search Results**: Reduced artist limit from 5 to 2
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- **MP3 Download Error 403**: Fixed 403 Forbidden error when downloading MP3 files ([#108](https://github.com/zarzet/SpotiFLAC-Mobile/issues/108))
|
||||||
|
- **Opus Cover Art**: Implemented METADATA_BLOCK_PICTURE for proper cover embedding
|
||||||
|
- **Deezer Pagination**: Fixed >25 tracks only showing first 25 ([#112](https://github.com/zarzet/SpotiFLAC-Mobile/issues/112))
|
||||||
|
- **Duplicate Embed Lyrics Setting**: Removed from Options page ([#110](https://github.com/zarzet/SpotiFLAC-Mobile/issues/110))
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## [3.2.1] - 2026-01-22
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
[](https://github.com/zarzet/SpotiFLAC-Mobile/releases)
|
[](https://github.com/zarzet/SpotiFLAC-Mobile/releases)
|
||||||
[](https://www.virustotal.com/gui/file/3257155286587a3596ad5d4380d4576a684aa3d37a5b19a615914a845fbe57f3)
|
[](https://www.virustotal.com/gui/file/516142f029a4f3642a899832a6f600acf07040170a98c106cd03222cf584d9a3)
|
||||||
[](https://crowdin.com/project/spotiflac-mobile)
|
[](https://crowdin.com/project/spotiflac-mobile)
|
||||||
|
|
||||||
<div align="center">
|
<div align="center">
|
||||||
@@ -52,8 +52,6 @@ Want to create your own extension? Check out the [Extension Development Guide](h
|
|||||||
### [SpotiFLAC (Desktop)](https://github.com/afkarxyz/SpotiFLAC)
|
### [SpotiFLAC (Desktop)](https://github.com/afkarxyz/SpotiFLAC)
|
||||||
Download music in true lossless FLAC from Tidal, Qobuz & Amazon Music for Windows, macOS & Linux
|
Download music in true lossless FLAC from Tidal, Qobuz & Amazon Music for Windows, macOS & Linux
|
||||||
|
|
||||||
> **Note:** Currently unavailable because the GitHub account is suspended. Alternatively, use [SpotiFLAC-Next](https://github.com/spotiverse/SpotiFLAC-Next) until the original is restored.
|
|
||||||
|
|
||||||
## Telegram
|
## Telegram
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
@@ -61,7 +59,7 @@ Download music in true lossless FLAC from Tidal, Qobuz & Amazon Music for Window
|
|||||||
<img src="https://img.shields.io/badge/Telegram-Channel-2CA5E0?style=for-the-badge&logo=telegram&logoColor=white" alt="Telegram Channel">
|
<img src="https://img.shields.io/badge/Telegram-Channel-2CA5E0?style=for-the-badge&logo=telegram&logoColor=white" alt="Telegram Channel">
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
<a href="https://t.me/spotiflacchat">
|
<a href="https://t.me/spotiflac_chat">
|
||||||
<img src="https://img.shields.io/badge/Telegram-Community-2CA5E0?style=for-the-badge&logo=telegram&logoColor=white" alt="Telegram Community">
|
<img src="https://img.shields.io/badge/Telegram-Community-2CA5E0?style=for-the-badge&logo=telegram&logoColor=white" alt="Telegram Community">
|
||||||
</a>
|
</a>
|
||||||
</p>
|
</p>
|
||||||
@@ -86,6 +84,14 @@ A: Yes, the app is open source and you can verify the code yourself. Each releas
|
|||||||
**Q: Why is download not working in my country?**
|
**Q: Why is download not working in my country?**
|
||||||
A: Some countries have restricted access to certain streaming service APIs. If downloads are failing, try using a VPN to connect through a different region.
|
A: Some countries have restricted access to certain streaming service APIs. If downloads are failing, try using a VPN to connect through a different region.
|
||||||
|
|
||||||
|
|
||||||
|
### Want to support SpotiFLAC-Mobile?
|
||||||
|
|
||||||
|
_If this software is useful and brings you value, consider supporting the project by buying me a coffee. Your support helps keep development going._
|
||||||
|
|
||||||
|
[](https://ko-fi.com/zarzet) <a href="https://www.buymeacoffee.com/zarzet" target="_blank"><img src="https://cdn.buymeacoffee.com/buttons/v2/default-yellow.png" alt="Buy Me A Coffee" style="height: 40px !important;width: 150px !important;" ></a>
|
||||||
|
|
||||||
|
|
||||||
## Disclaimer
|
## Disclaimer
|
||||||
|
|
||||||
This project is for **educational and private use only**. The developer does not condone or encourage copyright infringement.
|
This project is for **educational and private use only**. The developer does not condone or encourage copyright infringement.
|
||||||
@@ -100,3 +106,8 @@ You are solely responsible for:
|
|||||||
3. Any legal consequences resulting from the misuse of this tool.
|
3. Any legal consequences resulting from the misuse of this tool.
|
||||||
|
|
||||||
The software is provided "as is", without warranty of any kind. The author assumes no liability for any bans, damages, or legal issues arising from its use.
|
The software is provided "as is", without warranty of any kind. The author assumes no liability for any bans, damages, or legal issues arising from its use.
|
||||||
|
|
||||||
|
|
||||||
|
> [!TIP]
|
||||||
|
>
|
||||||
|
> **Star Us**, You will receive all release notifications from GitHub without any delay ~
|
||||||
|
|||||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Vendored
+71
-2
@@ -5,6 +5,7 @@
|
|||||||
-keep class io.flutter.view.** { *; }
|
-keep class io.flutter.view.** { *; }
|
||||||
-keep class io.flutter.** { *; }
|
-keep class io.flutter.** { *; }
|
||||||
-keep class io.flutter.plugins.** { *; }
|
-keep class io.flutter.plugins.** { *; }
|
||||||
|
-keep class io.flutter.embedding.** { *; }
|
||||||
|
|
||||||
# Ignore missing Play Core classes (not used, but referenced by Flutter)
|
# Ignore missing Play Core classes (not used, but referenced by Flutter)
|
||||||
-dontwarn com.google.android.play.core.splitcompat.**
|
-dontwarn com.google.android.play.core.splitcompat.**
|
||||||
@@ -14,9 +15,15 @@
|
|||||||
# Ignore missing javax.xml.stream (not used on Android)
|
# Ignore missing javax.xml.stream (not used on Android)
|
||||||
-dontwarn javax.xml.stream.**
|
-dontwarn javax.xml.stream.**
|
||||||
|
|
||||||
# Go backend (gobackend.aar)
|
# Go backend (gobackend.aar) - CRITICAL for release builds
|
||||||
-keep class gobackend.** { *; }
|
-keep class gobackend.** { *; }
|
||||||
-keep class go.** { *; }
|
-keep class go.** { *; }
|
||||||
|
-keep interface gobackend.** { *; }
|
||||||
|
-keepclassmembers class gobackend.** { *; }
|
||||||
|
|
||||||
|
# Go mobile binding internals
|
||||||
|
-keep class org.golang.** { *; }
|
||||||
|
-dontwarn org.golang.**
|
||||||
|
|
||||||
# FFmpeg Kit
|
# FFmpeg Kit
|
||||||
-keep class com.arthenica.ffmpegkit.** { *; }
|
-keep class com.arthenica.ffmpegkit.** { *; }
|
||||||
@@ -30,15 +37,77 @@
|
|||||||
native <methods>;
|
native <methods>;
|
||||||
}
|
}
|
||||||
|
|
||||||
# Kotlin coroutines
|
# Kotlin coroutines - expanded rules
|
||||||
-keepnames class kotlinx.coroutines.internal.MainDispatcherFactory {}
|
-keepnames class kotlinx.coroutines.internal.MainDispatcherFactory {}
|
||||||
-keepnames class kotlinx.coroutines.CoroutineExceptionHandler {}
|
-keepnames class kotlinx.coroutines.CoroutineExceptionHandler {}
|
||||||
-keepclassmembers class kotlinx.coroutines.** {
|
-keepclassmembers class kotlinx.coroutines.** {
|
||||||
volatile <fields>;
|
volatile <fields>;
|
||||||
}
|
}
|
||||||
|
-keepclassmembernames class kotlinx.** {
|
||||||
|
volatile <fields>;
|
||||||
|
}
|
||||||
|
-dontwarn kotlinx.coroutines.**
|
||||||
|
|
||||||
|
# Kotlin serialization
|
||||||
|
-keepattributes RuntimeVisibleAnnotations,AnnotationDefault
|
||||||
|
-dontwarn kotlin.**
|
||||||
|
-keep class kotlin.** { *; }
|
||||||
|
-keep class kotlin.Metadata { *; }
|
||||||
|
|
||||||
|
# Keep MainActivity and related classes
|
||||||
|
-keep class com.zarz.spotiflac.** { *; }
|
||||||
|
|
||||||
# Prevent R8 from removing metadata
|
# Prevent R8 from removing metadata
|
||||||
-keepattributes *Annotation*
|
-keepattributes *Annotation*
|
||||||
-keepattributes SourceFile,LineNumberTable
|
-keepattributes SourceFile,LineNumberTable
|
||||||
-keepattributes Signature
|
-keepattributes Signature
|
||||||
-keepattributes Exceptions
|
-keepattributes Exceptions
|
||||||
|
-keepattributes InnerClasses
|
||||||
|
-keepattributes EnclosingMethod
|
||||||
|
|
||||||
|
# JSON parsing (used by Go backend responses)
|
||||||
|
-keep class org.json.** { *; }
|
||||||
|
|
||||||
|
# Shared Preferences
|
||||||
|
-keep class androidx.datastore.** { *; }
|
||||||
|
-dontwarn androidx.datastore.**
|
||||||
|
|
||||||
|
# Flutter Plugins - CRITICAL: Prevent R8 from removing plugin implementations
|
||||||
|
# Path Provider
|
||||||
|
-keep class io.flutter.plugins.pathprovider.** { *; }
|
||||||
|
-keep class dev.flutter.pigeon.** { *; }
|
||||||
|
|
||||||
|
# Local Notifications
|
||||||
|
-keep class com.dexterous.** { *; }
|
||||||
|
-keep class com.dexterous.flutterlocalnotifications.** { *; }
|
||||||
|
|
||||||
|
# Receive Sharing Intent
|
||||||
|
-keep class com.kasem.receive_sharing_intent.** { *; }
|
||||||
|
|
||||||
|
# Permission Handler
|
||||||
|
-keep class com.baseflow.permissionhandler.** { *; }
|
||||||
|
|
||||||
|
# File Picker
|
||||||
|
-keep class com.mr.flutter.plugin.filepicker.** { *; }
|
||||||
|
|
||||||
|
# URL Launcher
|
||||||
|
-keep class io.flutter.plugins.urllauncher.** { *; }
|
||||||
|
|
||||||
|
# Share Plus
|
||||||
|
-keep class dev.fluttercommunity.plus.share.** { *; }
|
||||||
|
|
||||||
|
# Device Info Plus
|
||||||
|
-keep class dev.fluttercommunity.plus.device_info.** { *; }
|
||||||
|
|
||||||
|
# Open File
|
||||||
|
-keep class com.crazecoder.openfile.** { *; }
|
||||||
|
|
||||||
|
# Sqflite
|
||||||
|
-keep class com.tekartik.sqflite.** { *; }
|
||||||
|
|
||||||
|
# Dynamic Color
|
||||||
|
-keep class io.material.** { *; }
|
||||||
|
|
||||||
|
# Keep all Flutter plugin registrants
|
||||||
|
-keep class io.flutter.plugins.GeneratedPluginRegistrant { *; }
|
||||||
|
-keep class ** extends io.flutter.embedding.engine.plugins.FlutterPlugin { *; }
|
||||||
|
|||||||
@@ -1,23 +1,155 @@
|
|||||||
package com.zarz.spotiflac
|
package com.zarz.spotiflac
|
||||||
|
|
||||||
import android.content.Intent
|
import android.content.Intent
|
||||||
|
import android.os.Build
|
||||||
import io.flutter.embedding.android.FlutterActivity
|
import io.flutter.embedding.android.FlutterActivity
|
||||||
import io.flutter.embedding.engine.FlutterEngine
|
import io.flutter.embedding.engine.FlutterEngine
|
||||||
|
import io.flutter.embedding.engine.FlutterShellArgs
|
||||||
import io.flutter.plugin.common.MethodChannel
|
import io.flutter.plugin.common.MethodChannel
|
||||||
|
import io.flutter.plugins.GeneratedPluginRegistrant
|
||||||
import gobackend.Gobackend
|
import gobackend.Gobackend
|
||||||
import com.arthenica.ffmpegkit.FFmpegKit
|
|
||||||
import com.arthenica.ffmpegkit.ReturnCode
|
|
||||||
import kotlinx.coroutines.CoroutineScope
|
import kotlinx.coroutines.CoroutineScope
|
||||||
import kotlinx.coroutines.Dispatchers
|
import kotlinx.coroutines.Dispatchers
|
||||||
import kotlinx.coroutines.SupervisorJob
|
import kotlinx.coroutines.SupervisorJob
|
||||||
import kotlinx.coroutines.launch
|
import kotlinx.coroutines.launch
|
||||||
import kotlinx.coroutines.withContext
|
import kotlinx.coroutines.withContext
|
||||||
|
import java.util.Locale
|
||||||
|
|
||||||
class MainActivity: FlutterActivity() {
|
class MainActivity: FlutterActivity() {
|
||||||
private val CHANNEL = "com.zarz.spotiflac/backend"
|
private val CHANNEL = "com.zarz.spotiflac/backend"
|
||||||
private val FFMPEG_CHANNEL = "com.zarz.spotiflac/ffmpeg"
|
|
||||||
private val scope = CoroutineScope(SupervisorJob() + Dispatchers.Main)
|
private val scope = CoroutineScope(SupervisorJob() + Dispatchers.Main)
|
||||||
|
|
||||||
|
companion object {
|
||||||
|
// Minimum API level we consider "safe" for Impeller (Android 10+)
|
||||||
|
private const val SAFE_API_FOR_IMPELLER = 29
|
||||||
|
|
||||||
|
// Known problematic GPU patterns (lowercase)
|
||||||
|
private val PROBLEMATIC_GPU_PATTERNS = listOf(
|
||||||
|
"adreno (tm) 3", // Adreno 300 series (305, 320, 330, etc.) - old Qualcomm
|
||||||
|
"adreno (tm) 4", // Adreno 400 series - some have issues
|
||||||
|
"mali-4", // Mali-400 series - old ARM GPUs
|
||||||
|
"mali-t6", // Mali-T600 series
|
||||||
|
"mali-t7", // Mali-T700 series (some)
|
||||||
|
"powervr sgx", // PowerVR SGX series - old Imagination GPUs
|
||||||
|
"powervr ge8320", // PowerVR GE8320 - known issues
|
||||||
|
"gc1000", // Vivante GC1000
|
||||||
|
"gc2000", // Vivante GC2000
|
||||||
|
)
|
||||||
|
|
||||||
|
// Known problematic chipsets/hardware (lowercase)
|
||||||
|
private val PROBLEMATIC_CHIPSETS = listOf(
|
||||||
|
"mt6762", // MediaTek Helio P22 with PowerVR GE8320
|
||||||
|
"mt6765", // MediaTek Helio P35 with PowerVR GE8320
|
||||||
|
"mt8768", // MediaTek tablet chip
|
||||||
|
"mp0873", // MediaTek variant
|
||||||
|
"msm8974", // Snapdragon 800/801 with Adreno 330
|
||||||
|
"msm8226", // Snapdragon 400 with Adreno 305
|
||||||
|
"msm8926", // Snapdragon 400 with Adreno 305
|
||||||
|
"apq8084", // Snapdragon 805 (some issues)
|
||||||
|
)
|
||||||
|
|
||||||
|
// Known problematic device models (lowercase)
|
||||||
|
private val PROBLEMATIC_MODELS = listOf(
|
||||||
|
"sm-t220", // Samsung Tab A7 Lite
|
||||||
|
"sm-t225", // Samsung Tab A7 Lite LTE
|
||||||
|
"hammerhead", // Nexus 5 (Adreno 330)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Override Flutter shell args to disable Impeller on problematic devices.
|
||||||
|
* This is called before the Flutter engine starts.
|
||||||
|
*/
|
||||||
|
override fun getFlutterShellArgs(): FlutterShellArgs {
|
||||||
|
val args = super.getFlutterShellArgs()
|
||||||
|
|
||||||
|
if (shouldDisableImpeller()) {
|
||||||
|
// Log for debugging
|
||||||
|
android.util.Log.i("SpotiFLAC", "Legacy/problematic GPU detected: Disabling Impeller for ${Build.MODEL}")
|
||||||
|
android.util.Log.i("SpotiFLAC", "Device: ${Build.MANUFACTURER} ${Build.MODEL}, SDK: ${Build.VERSION.SDK_INT}")
|
||||||
|
android.util.Log.i("SpotiFLAC", "Hardware: ${Build.HARDWARE}, Board: ${Build.BOARD}")
|
||||||
|
|
||||||
|
// Disable Impeller, forcing Skia renderer
|
||||||
|
args.add("--enable-impeller=false")
|
||||||
|
} else {
|
||||||
|
android.util.Log.i("SpotiFLAC", "Using Impeller renderer for ${Build.MODEL}")
|
||||||
|
}
|
||||||
|
|
||||||
|
return args
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if device should use Skia instead of Impeller.
|
||||||
|
* Returns true for devices with old/problematic GPUs or old Android versions.
|
||||||
|
*/
|
||||||
|
private fun shouldDisableImpeller(): Boolean {
|
||||||
|
val hardware = Build.HARDWARE.lowercase(Locale.ROOT)
|
||||||
|
val board = Build.BOARD.lowercase(Locale.ROOT)
|
||||||
|
val model = Build.MODEL.lowercase(Locale.ROOT)
|
||||||
|
val device = Build.DEVICE.lowercase(Locale.ROOT)
|
||||||
|
|
||||||
|
// 1. Check for explicitly problematic device models
|
||||||
|
for (problematicModel in PROBLEMATIC_MODELS) {
|
||||||
|
if (model.contains(problematicModel) || device.contains(problematicModel)) {
|
||||||
|
android.util.Log.i("SpotiFLAC", "Matched problematic model: $problematicModel")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. Check for problematic chipsets
|
||||||
|
for (chipset in PROBLEMATIC_CHIPSETS) {
|
||||||
|
if (hardware.contains(chipset) || board.contains(chipset)) {
|
||||||
|
android.util.Log.i("SpotiFLAC", "Matched problematic chipset: $chipset")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. For Android < 10 (API 29), be more aggressive about disabling Impeller
|
||||||
|
if (Build.VERSION.SDK_INT < SAFE_API_FOR_IMPELLER) {
|
||||||
|
// For older Android, check GPU renderer if available
|
||||||
|
val gpuRenderer = getGpuRenderer().lowercase(Locale.ROOT)
|
||||||
|
|
||||||
|
// Check for known problematic GPUs
|
||||||
|
for (pattern in PROBLEMATIC_GPU_PATTERNS) {
|
||||||
|
if (gpuRenderer.contains(pattern)) {
|
||||||
|
android.util.Log.i("SpotiFLAC", "Matched problematic GPU on old Android: $pattern")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// For very old Android (< 8.0), always use Skia as Vulkan support is spotty
|
||||||
|
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O) {
|
||||||
|
android.util.Log.i("SpotiFLAC", "Android < 8.0, using Skia for safety")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 4. For Android 10+, still check for known problematic GPUs
|
||||||
|
val gpuRenderer = getGpuRenderer().lowercase(Locale.ROOT)
|
||||||
|
for (pattern in PROBLEMATIC_GPU_PATTERNS) {
|
||||||
|
if (gpuRenderer.contains(pattern)) {
|
||||||
|
android.util.Log.i("SpotiFLAC", "Matched problematic GPU: $pattern")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Try to get GPU renderer string.
|
||||||
|
* Note: This may return empty on some devices before OpenGL context is created.
|
||||||
|
*/
|
||||||
|
private fun getGpuRenderer(): String {
|
||||||
|
return try {
|
||||||
|
// This might not work before GL context is created,
|
||||||
|
// but worth trying for additional detection
|
||||||
|
android.opengl.GLES20.glGetString(android.opengl.GLES20.GL_RENDERER) ?: ""
|
||||||
|
} catch (e: Exception) {
|
||||||
|
""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
override fun onNewIntent(intent: Intent) {
|
override fun onNewIntent(intent: Intent) {
|
||||||
super.onNewIntent(intent)
|
super.onNewIntent(intent)
|
||||||
// Update the intent so receive_sharing_intent can access the new data
|
// Update the intent so receive_sharing_intent can access the new data
|
||||||
@@ -26,6 +158,7 @@ class MainActivity: FlutterActivity() {
|
|||||||
|
|
||||||
override fun configureFlutterEngine(flutterEngine: FlutterEngine) {
|
override fun configureFlutterEngine(flutterEngine: FlutterEngine) {
|
||||||
super.configureFlutterEngine(flutterEngine)
|
super.configureFlutterEngine(flutterEngine)
|
||||||
|
GeneratedPluginRegistrant.registerWith(flutterEngine)
|
||||||
|
|
||||||
MethodChannel(flutterEngine.dartExecutor.binaryMessenger, CHANNEL).setMethodCallHandler { call, result ->
|
MethodChannel(flutterEngine.dartExecutor.binaryMessenger, CHANNEL).setMethodCallHandler { call, result ->
|
||||||
scope.launch {
|
scope.launch {
|
||||||
@@ -278,9 +411,10 @@ class MainActivity: FlutterActivity() {
|
|||||||
"searchDeezerAll" -> {
|
"searchDeezerAll" -> {
|
||||||
val query = call.argument<String>("query") ?: ""
|
val query = call.argument<String>("query") ?: ""
|
||||||
val trackLimit = call.argument<Int>("track_limit") ?: 15
|
val trackLimit = call.argument<Int>("track_limit") ?: 15
|
||||||
val artistLimit = call.argument<Int>("artist_limit") ?: 3
|
val artistLimit = call.argument<Int>("artist_limit") ?: 2
|
||||||
|
val filter = call.argument<String>("filter") ?: ""
|
||||||
val response = withContext(Dispatchers.IO) {
|
val response = withContext(Dispatchers.IO) {
|
||||||
Gobackend.searchDeezerAll(query, trackLimit.toLong(), artistLimit.toLong())
|
Gobackend.searchDeezerAll(query, trackLimit.toLong(), artistLimit.toLong(), filter)
|
||||||
}
|
}
|
||||||
result.success(response)
|
result.success(response)
|
||||||
}
|
}
|
||||||
@@ -767,37 +901,5 @@ class MainActivity: FlutterActivity() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// FFmpeg method channel
|
|
||||||
MethodChannel(flutterEngine.dartExecutor.binaryMessenger, FFMPEG_CHANNEL).setMethodCallHandler { call, result ->
|
|
||||||
scope.launch {
|
|
||||||
try {
|
|
||||||
when (call.method) {
|
|
||||||
"execute" -> {
|
|
||||||
val command = call.argument<String>("command") ?: ""
|
|
||||||
val session = withContext(Dispatchers.IO) {
|
|
||||||
FFmpegKit.execute(command)
|
|
||||||
}
|
|
||||||
val returnCode = session.returnCode
|
|
||||||
val output = session.output ?: ""
|
|
||||||
result.success(mapOf(
|
|
||||||
"success" to ReturnCode.isSuccess(returnCode),
|
|
||||||
"returnCode" to (returnCode?.value ?: -1),
|
|
||||||
"output" to output
|
|
||||||
))
|
|
||||||
}
|
|
||||||
"getVersion" -> {
|
|
||||||
val session = withContext(Dispatchers.IO) {
|
|
||||||
FFmpegKit.execute("-version")
|
|
||||||
}
|
|
||||||
result.success(session.output ?: "unknown")
|
|
||||||
}
|
|
||||||
else -> result.notImplemented()
|
|
||||||
}
|
|
||||||
} catch (e: Exception) {
|
|
||||||
result.error("FFMPEG_ERROR", e.message, null)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,335 +0,0 @@
|
|||||||
import 'dart:io';
|
|
||||||
import 'package:ffmpeg_kit_flutter_new_audio/ffmpeg_kit.dart';
|
|
||||||
import 'package:ffmpeg_kit_flutter_new_audio/return_code.dart';
|
|
||||||
import 'package:spotiflac_android/utils/logger.dart';
|
|
||||||
|
|
||||||
final _log = AppLogger('FFmpeg');
|
|
||||||
|
|
||||||
/// FFmpeg service for iOS using ffmpeg_kit_flutter plugin
|
|
||||||
class FFmpegServiceIOS {
|
|
||||||
/// Execute FFmpeg command and return result
|
|
||||||
static Future<FFmpegResultIOS> _execute(String command) async {
|
|
||||||
try {
|
|
||||||
final session = await FFmpegKit.execute(command);
|
|
||||||
final returnCode = await session.getReturnCode();
|
|
||||||
final output = await session.getOutput() ?? '';
|
|
||||||
return FFmpegResultIOS(
|
|
||||||
success: ReturnCode.isSuccess(returnCode),
|
|
||||||
returnCode: returnCode?.getValue() ?? -1,
|
|
||||||
output: output,
|
|
||||||
);
|
|
||||||
} catch (e) {
|
|
||||||
_log.e('FFmpeg execute error: $e');
|
|
||||||
return FFmpegResultIOS(success: false, returnCode: -1, output: e.toString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Convert M4A (DASH segments) to FLAC
|
|
||||||
static Future<String?> convertM4aToFlac(String inputPath) async {
|
|
||||||
final outputPath = inputPath.replaceAll('.m4a', '.flac');
|
|
||||||
final command = '-i "$inputPath" -c:a flac -compression_level 8 "$outputPath" -y';
|
|
||||||
final result = await _execute(command);
|
|
||||||
|
|
||||||
if (result.success) {
|
|
||||||
try {
|
|
||||||
await File(inputPath).delete();
|
|
||||||
} catch (_) {}
|
|
||||||
return outputPath;
|
|
||||||
}
|
|
||||||
|
|
||||||
_log.e('M4A to FLAC conversion failed: ${result.output}');
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Convert FLAC to MP3
|
|
||||||
/// If deleteOriginal is true, deletes the FLAC file after conversion
|
|
||||||
static Future<String?> convertFlacToMp3(
|
|
||||||
String inputPath, {
|
|
||||||
String bitrate = '320k',
|
|
||||||
bool deleteOriginal = true,
|
|
||||||
}) async {
|
|
||||||
// Convert in same folder, just change extension
|
|
||||||
final outputPath = inputPath.replaceAll('.flac', '.mp3');
|
|
||||||
|
|
||||||
final command = '-i "$inputPath" -codec:a libmp3lame -b:a $bitrate -map 0:a -map_metadata 0 -id3v2_version 3 "$outputPath" -y';
|
|
||||||
final result = await _execute(command);
|
|
||||||
|
|
||||||
if (result.success) {
|
|
||||||
// Delete original FLAC if requested
|
|
||||||
if (deleteOriginal) {
|
|
||||||
try {
|
|
||||||
await File(inputPath).delete();
|
|
||||||
} catch (_) {}
|
|
||||||
}
|
|
||||||
return outputPath;
|
|
||||||
}
|
|
||||||
_log.e('FLAC to MP3 conversion failed: ${result.output}');
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Convert FLAC to M4A
|
|
||||||
static Future<String?> convertFlacToM4a(String inputPath, {String codec = 'aac', String bitrate = '256k'}) async {
|
|
||||||
final dir = File(inputPath).parent.path;
|
|
||||||
final baseName = inputPath.split(Platform.pathSeparator).last.replaceAll('.flac', '');
|
|
||||||
final outputDir = '$dir${Platform.pathSeparator}M4A';
|
|
||||||
await Directory(outputDir).create(recursive: true);
|
|
||||||
final outputPath = '$outputDir${Platform.pathSeparator}$baseName.m4a';
|
|
||||||
|
|
||||||
String command;
|
|
||||||
if (codec == 'alac') {
|
|
||||||
command = '-i "$inputPath" -codec:a alac -map 0:a -map_metadata 0 "$outputPath" -y';
|
|
||||||
} else {
|
|
||||||
command = '-i "$inputPath" -codec:a aac -b:a $bitrate -map 0:a -map_metadata 0 "$outputPath" -y';
|
|
||||||
}
|
|
||||||
|
|
||||||
final result = await _execute(command);
|
|
||||||
if (result.success) return outputPath;
|
|
||||||
_log.e('FLAC to M4A conversion failed: ${result.output}');
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Embed cover art to FLAC file
|
|
||||||
static Future<String?> embedCover(String flacPath, String coverPath) async {
|
|
||||||
final tempOutput = '$flacPath.tmp';
|
|
||||||
final command = '-i "$flacPath" -i "$coverPath" -map 0:a -map 1:0 -c copy -metadata:s:v title="Album cover" -metadata:s:v comment="Cover (front)" -disposition:v attached_pic "$tempOutput" -y';
|
|
||||||
|
|
||||||
final result = await _execute(command);
|
|
||||||
|
|
||||||
if (result.success) {
|
|
||||||
try {
|
|
||||||
await File(flacPath).delete();
|
|
||||||
await File(tempOutput).rename(flacPath);
|
|
||||||
return flacPath;
|
|
||||||
} catch (e) {
|
|
||||||
_log.e('Failed to replace file after cover embed: $e');
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
final tempFile = File(tempOutput);
|
|
||||||
if (await tempFile.exists()) await tempFile.delete();
|
|
||||||
} catch (_) {}
|
|
||||||
|
|
||||||
_log.e('Cover embed failed: ${result.output}');
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Embed metadata and cover art to FLAC file
|
|
||||||
/// Returns the file path on success, null on failure
|
|
||||||
static Future<String?> embedMetadata({
|
|
||||||
required String flacPath,
|
|
||||||
String? coverPath,
|
|
||||||
Map<String, String>? metadata,
|
|
||||||
}) async {
|
|
||||||
final tempOutput = '$flacPath.tmp';
|
|
||||||
|
|
||||||
// Construct command
|
|
||||||
final StringBuffer cmdBuffer = StringBuffer();
|
|
||||||
cmdBuffer.write('-i "$flacPath" ');
|
|
||||||
|
|
||||||
// Add cover input if available
|
|
||||||
if (coverPath != null) {
|
|
||||||
cmdBuffer.write('-i "$coverPath" ');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Map audio stream
|
|
||||||
cmdBuffer.write('-map 0:a ');
|
|
||||||
|
|
||||||
// Map cover stream if available
|
|
||||||
if (coverPath != null) {
|
|
||||||
cmdBuffer.write('-map 1:0 ');
|
|
||||||
cmdBuffer.write('-c:v copy ');
|
|
||||||
cmdBuffer.write('-disposition:v attached_pic ');
|
|
||||||
cmdBuffer.write('-metadata:s:v title="Album cover" ');
|
|
||||||
cmdBuffer.write('-metadata:s:v comment="Cover (front)" ');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Copy audio codec (don't re-encode)
|
|
||||||
cmdBuffer.write('-c:a copy ');
|
|
||||||
|
|
||||||
// Add text metadata
|
|
||||||
if (metadata != null) {
|
|
||||||
metadata.forEach((key, value) {
|
|
||||||
// Sanitize value: escape double quotes
|
|
||||||
final sanitizedValue = value.replaceAll('"', '\\"');
|
|
||||||
cmdBuffer.write('-metadata $key="$sanitizedValue" ');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
cmdBuffer.write('"$tempOutput" -y');
|
|
||||||
|
|
||||||
final command = cmdBuffer.toString();
|
|
||||||
_log.d('Executing FFmpeg command: $command');
|
|
||||||
|
|
||||||
final result = await _execute(command);
|
|
||||||
|
|
||||||
if (result.success) {
|
|
||||||
try {
|
|
||||||
await File(flacPath).delete();
|
|
||||||
await File(tempOutput).rename(flacPath);
|
|
||||||
return flacPath;
|
|
||||||
} catch (e) {
|
|
||||||
_log.e('Failed to replace file after metadata embed: $e');
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Clean up temp file if exists
|
|
||||||
try {
|
|
||||||
final tempFile = File(tempOutput);
|
|
||||||
if (await tempFile.exists()) {
|
|
||||||
await tempFile.delete();
|
|
||||||
}
|
|
||||||
} catch (_) {}
|
|
||||||
|
|
||||||
_log.e('Metadata/Cover embed failed: ${result.output}');
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Embed metadata and cover art to MP3 file using ID3v2 tags
|
|
||||||
/// Returns the file path on success, null on failure
|
|
||||||
static Future<String?> embedMetadataToMp3({
|
|
||||||
required String mp3Path,
|
|
||||||
String? coverPath,
|
|
||||||
Map<String, String>? metadata,
|
|
||||||
}) async {
|
|
||||||
final tempOutput = '$mp3Path.tmp';
|
|
||||||
|
|
||||||
final StringBuffer cmdBuffer = StringBuffer();
|
|
||||||
cmdBuffer.write('-i "$mp3Path" ');
|
|
||||||
|
|
||||||
if (coverPath != null) {
|
|
||||||
cmdBuffer.write('-i "$coverPath" ');
|
|
||||||
}
|
|
||||||
|
|
||||||
cmdBuffer.write('-map 0:a ');
|
|
||||||
|
|
||||||
if (coverPath != null) {
|
|
||||||
cmdBuffer.write('-map 1:0 ');
|
|
||||||
cmdBuffer.write('-c:v:0 copy ');
|
|
||||||
cmdBuffer.write('-id3v2_version 3 ');
|
|
||||||
cmdBuffer.write('-metadata:s:v title="Album cover" ');
|
|
||||||
cmdBuffer.write('-metadata:s:v comment="Cover (front)" ');
|
|
||||||
}
|
|
||||||
|
|
||||||
cmdBuffer.write('-c:a copy ');
|
|
||||||
|
|
||||||
if (metadata != null) {
|
|
||||||
// Convert FLAC/Vorbis tags to ID3v2 tags for MP3
|
|
||||||
final id3Metadata = _convertToId3Tags(metadata);
|
|
||||||
id3Metadata.forEach((key, value) {
|
|
||||||
final sanitizedValue = value.replaceAll('"', '\\"');
|
|
||||||
cmdBuffer.write('-metadata $key="$sanitizedValue" ');
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
cmdBuffer.write('-id3v2_version 3 "$tempOutput" -y');
|
|
||||||
|
|
||||||
final command = cmdBuffer.toString();
|
|
||||||
_log.d('Executing FFmpeg MP3 embed command: $command');
|
|
||||||
|
|
||||||
final result = await _execute(command);
|
|
||||||
|
|
||||||
if (result.success) {
|
|
||||||
try {
|
|
||||||
await File(mp3Path).delete();
|
|
||||||
await File(tempOutput).rename(mp3Path);
|
|
||||||
_log.d('MP3 metadata embedded successfully');
|
|
||||||
return mp3Path;
|
|
||||||
} catch (e) {
|
|
||||||
_log.e('Failed to replace MP3 file after metadata embed: $e');
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
final tempFile = File(tempOutput);
|
|
||||||
if (await tempFile.exists()) {
|
|
||||||
await tempFile.delete();
|
|
||||||
}
|
|
||||||
} catch (_) {}
|
|
||||||
|
|
||||||
_log.e('MP3 Metadata/Cover embed failed: ${result.output}');
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Convert FLAC/Vorbis comment tags to ID3v2 compatible tags
|
|
||||||
static Map<String, String> _convertToId3Tags(Map<String, String> vorbisMetadata) {
|
|
||||||
final id3Map = <String, String>{};
|
|
||||||
|
|
||||||
for (final entry in vorbisMetadata.entries) {
|
|
||||||
final key = entry.key.toUpperCase();
|
|
||||||
final value = entry.value;
|
|
||||||
|
|
||||||
// Map Vorbis comments to ID3v2 frame names
|
|
||||||
switch (key) {
|
|
||||||
case 'TITLE':
|
|
||||||
id3Map['title'] = value;
|
|
||||||
break;
|
|
||||||
case 'ARTIST':
|
|
||||||
id3Map['artist'] = value;
|
|
||||||
break;
|
|
||||||
case 'ALBUM':
|
|
||||||
id3Map['album'] = value;
|
|
||||||
break;
|
|
||||||
case 'ALBUMARTIST':
|
|
||||||
id3Map['album_artist'] = value;
|
|
||||||
break;
|
|
||||||
case 'TRACKNUMBER':
|
|
||||||
case 'TRACK':
|
|
||||||
id3Map['track'] = value;
|
|
||||||
break;
|
|
||||||
case 'DISCNUMBER':
|
|
||||||
case 'DISC':
|
|
||||||
id3Map['disc'] = value;
|
|
||||||
break;
|
|
||||||
case 'DATE':
|
|
||||||
case 'YEAR':
|
|
||||||
id3Map['date'] = value;
|
|
||||||
break;
|
|
||||||
case 'ISRC':
|
|
||||||
id3Map['TSRC'] = value; // ID3v2 ISRC frame
|
|
||||||
break;
|
|
||||||
case 'LYRICS':
|
|
||||||
case 'UNSYNCEDLYRICS':
|
|
||||||
id3Map['lyrics'] = value;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
// Pass through other tags as-is
|
|
||||||
id3Map[key.toLowerCase()] = value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return id3Map;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Check if FFmpeg is available
|
|
||||||
static Future<bool> isAvailable() async {
|
|
||||||
try {
|
|
||||||
final session = await FFmpegKit.execute('-version');
|
|
||||||
final returnCode = await session.getReturnCode();
|
|
||||||
return ReturnCode.isSuccess(returnCode);
|
|
||||||
} catch (e) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Get FFmpeg version info
|
|
||||||
static Future<String?> getVersion() async {
|
|
||||||
try {
|
|
||||||
final session = await FFmpegKit.execute('-version');
|
|
||||||
return await session.getOutput();
|
|
||||||
} catch (e) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class FFmpegResultIOS {
|
|
||||||
final bool success;
|
|
||||||
final int returnCode;
|
|
||||||
final String output;
|
|
||||||
|
|
||||||
FFmpegResultIOS({required this.success, required this.returnCode, required this.output});
|
|
||||||
}
|
|
||||||
+68
-297
@@ -3,7 +3,6 @@ package gobackend
|
|||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
"context"
|
"context"
|
||||||
"encoding/base64"
|
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
@@ -12,79 +11,29 @@ import (
|
|||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
type AmazonDownloader struct {
|
type AmazonDownloader struct {
|
||||||
client *http.Client
|
client *http.Client
|
||||||
regions []string
|
|
||||||
lastAPICallTime time.Time
|
|
||||||
apiCallCount int
|
|
||||||
apiCallResetTime time.Time
|
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
globalAmazonDownloader *AmazonDownloader
|
globalAmazonDownloader *AmazonDownloader
|
||||||
amazonDownloaderOnce sync.Once
|
amazonDownloaderOnce sync.Once
|
||||||
amazonRateLimitMu sync.Mutex
|
|
||||||
)
|
)
|
||||||
|
|
||||||
// DoubleDoubleSubmitResponse is the response from DoubleDouble submit endpoint
|
// AfkarXYZResponse is the response from AfkarXYZ API
|
||||||
type DoubleDoubleSubmitResponse struct {
|
type AfkarXYZResponse struct {
|
||||||
Success bool `json:"success"`
|
Success bool `json:"success"`
|
||||||
ID string `json:"id"`
|
Data struct {
|
||||||
}
|
DirectLink string `json:"direct_link"`
|
||||||
|
FileName string `json:"file_name"`
|
||||||
type DoubleDoubleStatusResponse struct {
|
FileSize int64 `json:"file_size"`
|
||||||
Status string `json:"status"`
|
} `json:"data"`
|
||||||
FriendlyStatus string `json:"friendlyStatus"`
|
|
||||||
URL string `json:"url"`
|
|
||||||
Current struct {
|
|
||||||
Name string `json:"name"`
|
|
||||||
Artist string `json:"artist"`
|
|
||||||
} `json:"current"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func amazonArtistsMatch(expectedArtist, foundArtist string) bool {
|
|
||||||
normExpected := strings.ToLower(strings.TrimSpace(expectedArtist))
|
|
||||||
normFound := strings.ToLower(strings.TrimSpace(foundArtist))
|
|
||||||
|
|
||||||
if normExpected == normFound {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
if strings.Contains(normExpected, normFound) || strings.Contains(normFound, normExpected) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
expectedFirst := strings.Split(normExpected, ",")[0]
|
|
||||||
expectedFirst = strings.Split(expectedFirst, " feat")[0]
|
|
||||||
expectedFirst = strings.Split(expectedFirst, " ft.")[0]
|
|
||||||
expectedFirst = strings.TrimSpace(expectedFirst)
|
|
||||||
|
|
||||||
foundFirst := strings.Split(normFound, ",")[0]
|
|
||||||
foundFirst = strings.Split(foundFirst, " feat")[0]
|
|
||||||
foundFirst = strings.Split(foundFirst, " ft.")[0]
|
|
||||||
foundFirst = strings.TrimSpace(foundFirst)
|
|
||||||
|
|
||||||
if expectedFirst == foundFirst {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
if strings.Contains(expectedFirst, foundFirst) || strings.Contains(foundFirst, expectedFirst) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
expectedASCII := amazonIsASCIIString(expectedArtist)
|
|
||||||
foundASCII := amazonIsASCIIString(foundArtist)
|
|
||||||
if expectedASCII != foundASCII {
|
|
||||||
GoLog("[Amazon] Artist names in different scripts, assuming match: '%s' vs '%s'\n", expectedArtist, foundArtist)
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
return false
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func amazonIsASCIIString(s string) bool {
|
func amazonIsASCIIString(s string) bool {
|
||||||
@@ -99,228 +48,63 @@ func amazonIsASCIIString(s string) bool {
|
|||||||
func NewAmazonDownloader() *AmazonDownloader {
|
func NewAmazonDownloader() *AmazonDownloader {
|
||||||
amazonDownloaderOnce.Do(func() {
|
amazonDownloaderOnce.Do(func() {
|
||||||
globalAmazonDownloader = &AmazonDownloader{
|
globalAmazonDownloader = &AmazonDownloader{
|
||||||
client: NewHTTPClientWithTimeout(120 * time.Second), // 120s timeout like PC
|
client: NewHTTPClientWithTimeout(120 * time.Second),
|
||||||
regions: []string{"us", "eu"}, // Same regions as PC
|
|
||||||
apiCallResetTime: time.Now(),
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
return globalAmazonDownloader
|
return globalAmazonDownloader
|
||||||
}
|
}
|
||||||
|
|
||||||
// waitForRateLimit implements rate limiting similar to PC version
|
// downloadFromAfkarXYZ downloads a track using AfkarXYZ API
|
||||||
func (a *AmazonDownloader) waitForRateLimit() {
|
// Returns: downloadURL, fileName, error
|
||||||
amazonRateLimitMu.Lock()
|
func (a *AmazonDownloader) downloadFromAfkarXYZ(amazonURL string) (string, string, error) {
|
||||||
defer amazonRateLimitMu.Unlock()
|
// AfkarXYZ API endpoint
|
||||||
|
apiURL := "https://amazon.afkarxyz.fun/convert?url=" + url.QueryEscape(amazonURL)
|
||||||
|
|
||||||
now := time.Now()
|
GoLog("[Amazon] Fetching from AfkarXYZ API...\n")
|
||||||
|
|
||||||
if now.Sub(a.apiCallResetTime) >= time.Minute {
|
req, err := http.NewRequest("GET", apiURL, nil)
|
||||||
a.apiCallCount = 0
|
if err != nil {
|
||||||
a.apiCallResetTime = now
|
return "", "", fmt.Errorf("failed to create request: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if a.apiCallCount >= 9 {
|
req.Header.Set("User-Agent", getRandomUserAgent())
|
||||||
waitTime := time.Minute - now.Sub(a.apiCallResetTime)
|
|
||||||
if waitTime > 0 {
|
resp, err := a.client.Do(req)
|
||||||
GoLog("[Amazon] Rate limit reached, waiting %v...\n", waitTime.Round(time.Second))
|
if err != nil {
|
||||||
time.Sleep(waitTime)
|
return "", "", fmt.Errorf("failed to call AfkarXYZ API: %w", err)
|
||||||
a.apiCallCount = 0
|
}
|
||||||
a.apiCallResetTime = time.Now()
|
defer resp.Body.Close()
|
||||||
}
|
|
||||||
|
if resp.StatusCode != 200 {
|
||||||
|
return "", "", fmt.Errorf("AfkarXYZ API returned status %d", resp.StatusCode)
|
||||||
}
|
}
|
||||||
|
|
||||||
if !a.lastAPICallTime.IsZero() {
|
body, err := io.ReadAll(resp.Body)
|
||||||
timeSinceLastCall := now.Sub(a.lastAPICallTime)
|
if err != nil {
|
||||||
minDelay := 7 * time.Second
|
return "", "", fmt.Errorf("failed to read response: %w", err)
|
||||||
if timeSinceLastCall < minDelay {
|
|
||||||
waitTime := minDelay - timeSinceLastCall
|
|
||||||
GoLog("[Amazon] Rate limiting: waiting %v...\n", waitTime.Round(time.Second))
|
|
||||||
time.Sleep(waitTime)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
a.lastAPICallTime = time.Now()
|
var apiResp AfkarXYZResponse
|
||||||
a.apiCallCount++
|
if err := json.Unmarshal(body, &apiResp); err != nil {
|
||||||
}
|
return "", "", fmt.Errorf("failed to decode response: %w", err)
|
||||||
|
|
||||||
// Uses same service as PC version (doubledouble.top)
|
|
||||||
func (a *AmazonDownloader) GetAvailableAPIs() []string {
|
|
||||||
// DoubleDouble service regions (same as PC)
|
|
||||||
// Format: https://{region}.doubledouble.top
|
|
||||||
var apis []string
|
|
||||||
for _, region := range a.regions {
|
|
||||||
apis = append(apis, fmt.Sprintf("https://%s.doubledouble.top", region))
|
|
||||||
}
|
|
||||||
return apis
|
|
||||||
}
|
|
||||||
|
|
||||||
// downloadFromDoubleDoubleService downloads a track using DoubleDouble service (same as PC)
|
|
||||||
// This uses submit → poll → download mechanism
|
|
||||||
// Internal function - not exported to gomobile
|
|
||||||
func (a *AmazonDownloader) downloadFromDoubleDoubleService(amazonURL, _ string) (string, string, string, error) {
|
|
||||||
var lastError error
|
|
||||||
|
|
||||||
for _, region := range a.regions {
|
|
||||||
GoLog("[Amazon] Trying region: %s...\n", region)
|
|
||||||
|
|
||||||
serviceBase, _ := base64.StdEncoding.DecodeString("aHR0cHM6Ly8=") // https://
|
|
||||||
serviceDomain, _ := base64.StdEncoding.DecodeString("LmRvdWJsZWRvdWJsZS50b3A=") // .doubledouble.top
|
|
||||||
baseURL := fmt.Sprintf("%s%s%s", string(serviceBase), region, string(serviceDomain))
|
|
||||||
|
|
||||||
encodedURL := url.QueryEscape(amazonURL)
|
|
||||||
submitURL := fmt.Sprintf("%s/dl?url=%s", baseURL, encodedURL)
|
|
||||||
|
|
||||||
a.waitForRateLimit()
|
|
||||||
|
|
||||||
req, err := http.NewRequest("GET", submitURL, nil)
|
|
||||||
if err != nil {
|
|
||||||
lastError = fmt.Errorf("failed to create request: %w", err)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
req.Header.Set("User-Agent", getRandomUserAgent())
|
|
||||||
|
|
||||||
fmt.Println("[Amazon] Submitting download request...")
|
|
||||||
|
|
||||||
// Retry logic for 429 errors (like PC version: 3 retries with 15s wait)
|
|
||||||
var resp *http.Response
|
|
||||||
maxRetries := 3
|
|
||||||
for retry := 0; retry < maxRetries; retry++ {
|
|
||||||
resp, err = a.client.Do(req)
|
|
||||||
if err != nil {
|
|
||||||
lastError = fmt.Errorf("failed to submit request: %w", err)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
if resp.StatusCode == 429 { // Too Many Requests
|
|
||||||
resp.Body.Close()
|
|
||||||
if retry < maxRetries-1 {
|
|
||||||
waitTime := 15 * time.Second
|
|
||||||
GoLog("[Amazon] Rate limited (429), waiting %v before retry %d/%d...\n", waitTime, retry+2, maxRetries)
|
|
||||||
time.Sleep(waitTime)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
lastError = fmt.Errorf("API rate limit exceeded after %d retries", maxRetries)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
if resp.StatusCode != 200 {
|
|
||||||
resp.Body.Close()
|
|
||||||
lastError = fmt.Errorf("submit failed with status %d", resp.StatusCode)
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
// Success - break retry loop
|
|
||||||
break
|
|
||||||
}
|
|
||||||
|
|
||||||
if err != nil || lastError != nil {
|
|
||||||
if resp != nil {
|
|
||||||
resp.Body.Close()
|
|
||||||
}
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
var submitResp DoubleDoubleSubmitResponse
|
|
||||||
if err := json.NewDecoder(resp.Body).Decode(&submitResp); err != nil {
|
|
||||||
resp.Body.Close()
|
|
||||||
lastError = fmt.Errorf("failed to decode submit response: %w", err)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
resp.Body.Close()
|
|
||||||
|
|
||||||
if !submitResp.Success || submitResp.ID == "" {
|
|
||||||
lastError = fmt.Errorf("submit request failed")
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
downloadID := submitResp.ID
|
|
||||||
GoLog("[Amazon] Download ID: %s\n", downloadID)
|
|
||||||
|
|
||||||
// Step 2: Poll for completion
|
|
||||||
statusURL := fmt.Sprintf("%s/dl/%s", baseURL, downloadID)
|
|
||||||
fmt.Println("[Amazon] Waiting for download to complete...")
|
|
||||||
|
|
||||||
maxWait := 300 * time.Second // 5 minutes max wait
|
|
||||||
elapsed := time.Duration(0)
|
|
||||||
pollInterval := 3 * time.Second
|
|
||||||
|
|
||||||
for elapsed < maxWait {
|
|
||||||
time.Sleep(pollInterval)
|
|
||||||
elapsed += pollInterval
|
|
||||||
|
|
||||||
statusReq, err := http.NewRequest("GET", statusURL, nil)
|
|
||||||
if err != nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
statusReq.Header.Set("User-Agent", getRandomUserAgent())
|
|
||||||
|
|
||||||
statusResp, err := a.client.Do(statusReq)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Printf("\r[Amazon] Status check failed, retrying...")
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if statusResp.StatusCode != 200 {
|
|
||||||
statusResp.Body.Close()
|
|
||||||
fmt.Printf("\r[Amazon] Status check failed (status %d), retrying...", statusResp.StatusCode)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
var status DoubleDoubleStatusResponse
|
|
||||||
if err := json.NewDecoder(statusResp.Body).Decode(&status); err != nil {
|
|
||||||
statusResp.Body.Close()
|
|
||||||
fmt.Printf("\r[Amazon] Invalid JSON response, retrying...")
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
statusResp.Body.Close()
|
|
||||||
|
|
||||||
if status.Status == "done" {
|
|
||||||
fmt.Println("\n[Amazon] Download ready!")
|
|
||||||
|
|
||||||
fileURL := status.URL
|
|
||||||
if strings.HasPrefix(fileURL, "./") {
|
|
||||||
fileURL = fmt.Sprintf("%s/%s", baseURL, fileURL[2:])
|
|
||||||
} else if strings.HasPrefix(fileURL, "/") {
|
|
||||||
fileURL = fmt.Sprintf("%s%s", baseURL, fileURL)
|
|
||||||
}
|
|
||||||
|
|
||||||
trackName := status.Current.Name
|
|
||||||
artist := status.Current.Artist
|
|
||||||
|
|
||||||
GoLog("[Amazon] Downloading: %s - %s\n", artist, trackName)
|
|
||||||
return fileURL, trackName, artist, nil
|
|
||||||
|
|
||||||
} else if status.Status == "error" {
|
|
||||||
errorMsg := status.FriendlyStatus
|
|
||||||
if errorMsg == "" {
|
|
||||||
errorMsg = "Unknown error"
|
|
||||||
}
|
|
||||||
lastError = fmt.Errorf("processing failed: %s", errorMsg)
|
|
||||||
break
|
|
||||||
} else {
|
|
||||||
// Still processing
|
|
||||||
friendlyStatus := status.FriendlyStatus
|
|
||||||
if friendlyStatus == "" {
|
|
||||||
friendlyStatus = status.Status
|
|
||||||
}
|
|
||||||
fmt.Printf("\r[Amazon] %s...", friendlyStatus)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if elapsed >= maxWait {
|
|
||||||
lastError = fmt.Errorf("download timeout")
|
|
||||||
fmt.Printf("\n[Amazon] Error with %s region: %v\n", region, lastError)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
if lastError != nil {
|
|
||||||
fmt.Printf("\n[Amazon] Error with %s region: %v\n", region, lastError)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return "", "", "", fmt.Errorf("all regions failed. Last error: %v", lastError)
|
if !apiResp.Success || apiResp.Data.DirectLink == "" {
|
||||||
|
return "", "", fmt.Errorf("AfkarXYZ API failed or no download link found")
|
||||||
|
}
|
||||||
|
|
||||||
|
fileName := apiResp.Data.FileName
|
||||||
|
if fileName == "" {
|
||||||
|
fileName = "track.flac"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sanitize filename
|
||||||
|
reg := regexp.MustCompile(`[<>:"/\\|?*]`)
|
||||||
|
fileName = reg.ReplaceAllString(fileName, "")
|
||||||
|
|
||||||
|
GoLog("[Amazon] AfkarXYZ returned: %s (%.2f MB)\n", fileName, float64(apiResp.Data.FileSize)/(1024*1024))
|
||||||
|
|
||||||
|
return apiResp.Data.DirectLink, fileName, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (a *AmazonDownloader) DownloadFile(downloadURL, outputPath, itemID string) error {
|
func (a *AmazonDownloader) DownloadFile(downloadURL, outputPath, itemID string) error {
|
||||||
@@ -404,7 +188,7 @@ func (a *AmazonDownloader) DownloadFile(downloadURL, outputPath, itemID string)
|
|||||||
return fmt.Errorf("incomplete download: expected %d bytes, got %d bytes", expectedSize, written)
|
return fmt.Errorf("incomplete download: expected %d bytes, got %d bytes", expectedSize, written)
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Printf("\r[Amazon] Downloaded: %.2f MB (Complete)\n", float64(written)/(1024*1024))
|
GoLog("[Amazon] Downloaded: %.2f MB (Complete)\n", float64(written)/(1024*1024))
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -422,7 +206,7 @@ type AmazonDownloadResult struct {
|
|||||||
ISRC string
|
ISRC string
|
||||||
}
|
}
|
||||||
|
|
||||||
// Uses DoubleDouble service (same as PC version)
|
// downloadFromAmazon uses AfkarXYZ API to download from Amazon Music
|
||||||
func downloadFromAmazon(req DownloadRequest) (AmazonDownloadResult, error) {
|
func downloadFromAmazon(req DownloadRequest) (AmazonDownloadResult, error) {
|
||||||
downloader := NewAmazonDownloader()
|
downloader := NewAmazonDownloader()
|
||||||
|
|
||||||
@@ -434,8 +218,7 @@ func downloadFromAmazon(req DownloadRequest) (AmazonDownloadResult, error) {
|
|||||||
var availability *TrackAvailability
|
var availability *TrackAvailability
|
||||||
var err error
|
var err error
|
||||||
|
|
||||||
if strings.HasPrefix(req.SpotifyID, "deezer:") {
|
if deezerID, found := strings.CutPrefix(req.SpotifyID, "deezer:"); found {
|
||||||
deezerID := strings.TrimPrefix(req.SpotifyID, "deezer:")
|
|
||||||
GoLog("[Amazon] Using Deezer ID for SongLink lookup: %s\n", deezerID)
|
GoLog("[Amazon] Using Deezer ID for SongLink lookup: %s\n", deezerID)
|
||||||
availability, err = songlink.CheckAvailabilityFromDeezer(deezerID)
|
availability, err = songlink.CheckAvailabilityFromDeezer(deezerID)
|
||||||
} else if req.SpotifyID != "" {
|
} else if req.SpotifyID != "" {
|
||||||
@@ -458,21 +241,15 @@ func downloadFromAmazon(req DownloadRequest) (AmazonDownloadResult, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Download using DoubleDouble service (same as PC)
|
// Download using AfkarXYZ API
|
||||||
downloadURL, trackName, artistName, err := downloader.downloadFromDoubleDoubleService(availability.AmazonURL, req.OutputDir)
|
downloadURL, _, err := downloader.downloadFromAfkarXYZ(availability.AmazonURL)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return AmazonDownloadResult{}, fmt.Errorf("failed to get download URL: %w", err)
|
return AmazonDownloadResult{}, fmt.Errorf("failed to get download URL from AfkarXYZ: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Verify artist matches
|
GoLog("[Amazon] Match found: '%s' by '%s'\n", req.TrackName, req.ArtistName)
|
||||||
if artistName != "" && !amazonArtistsMatch(req.ArtistName, artistName) {
|
|
||||||
GoLog("[Amazon] Artist mismatch: expected '%s', got '%s'. Rejecting.\n", req.ArtistName, artistName)
|
|
||||||
return AmazonDownloadResult{}, fmt.Errorf("artist mismatch: expected '%s', got '%s'", req.ArtistName, artistName)
|
|
||||||
}
|
|
||||||
|
|
||||||
GoLog("[Amazon] Match found: '%s' by '%s'\n", trackName, artistName)
|
filename := buildFilenameFromTemplate(req.FilenameFormat, map[string]any{
|
||||||
|
|
||||||
filename := buildFilenameFromTemplate(req.FilenameFormat, map[string]interface{}{
|
|
||||||
"title": req.TrackName,
|
"title": req.TrackName,
|
||||||
"artist": req.ArtistName,
|
"artist": req.ArtistName,
|
||||||
"album": req.AlbumName,
|
"album": req.AlbumName,
|
||||||
@@ -519,11 +296,6 @@ func downloadFromAmazon(req DownloadRequest) (AmazonDownloadResult, error) {
|
|||||||
SetItemFinalizing(req.ItemID)
|
SetItemFinalizing(req.ItemID)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Log track info from DoubleDouble (for debugging)
|
|
||||||
if trackName != "" && artistName != "" {
|
|
||||||
GoLog("[Amazon] DoubleDouble returned: %s - %s\n", artistName, trackName)
|
|
||||||
}
|
|
||||||
|
|
||||||
existingMeta, metaErr := ReadMetadata(outputPath)
|
existingMeta, metaErr := ReadMetadata(outputPath)
|
||||||
actualTrackNum := req.TrackNumber
|
actualTrackNum := req.TrackNumber
|
||||||
actualDiscNum := req.DiscNumber
|
actualDiscNum := req.DiscNumber
|
||||||
@@ -539,8 +311,7 @@ func downloadFromAmazon(req DownloadRequest) (AmazonDownloadResult, error) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Embed metadata using Spotify data (more accurate than DoubleDouble)
|
// Embed metadata using Spotify data
|
||||||
// But preserve track/disc numbers from file if they were better
|
|
||||||
metadata := Metadata{
|
metadata := Metadata{
|
||||||
Title: req.TrackName,
|
Title: req.TrackName,
|
||||||
Artist: req.ArtistName,
|
Artist: req.ArtistName,
|
||||||
@@ -551,9 +322,9 @@ func downloadFromAmazon(req DownloadRequest) (AmazonDownloadResult, error) {
|
|||||||
TotalTracks: req.TotalTracks,
|
TotalTracks: req.TotalTracks,
|
||||||
DiscNumber: actualDiscNum,
|
DiscNumber: actualDiscNum,
|
||||||
ISRC: req.ISRC,
|
ISRC: req.ISRC,
|
||||||
Genre: req.Genre, // From Deezer album metadata
|
Genre: req.Genre,
|
||||||
Label: req.Label, // From Deezer album metadata
|
Label: req.Label,
|
||||||
Copyright: req.Copyright, // From Deezer album metadata
|
Copyright: req.Copyright,
|
||||||
}
|
}
|
||||||
|
|
||||||
// Use cover data from parallel fetch
|
// Use cover data from parallel fetch
|
||||||
@@ -564,7 +335,7 @@ func downloadFromAmazon(req DownloadRequest) (AmazonDownloadResult, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err := EmbedMetadataWithCoverData(outputPath, metadata, coverData); err != nil {
|
if err := EmbedMetadataWithCoverData(outputPath, metadata, coverData); err != nil {
|
||||||
fmt.Printf("Warning: failed to embed metadata: %v\n", err)
|
GoLog("[Amazon] Warning: failed to embed metadata: %v\n", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if req.EmbedLyrics && parallelResult != nil && parallelResult.LyricsLRC != "" {
|
if req.EmbedLyrics && parallelResult != nil && parallelResult.LyricsLRC != "" {
|
||||||
@@ -587,14 +358,14 @@ func downloadFromAmazon(req DownloadRequest) (AmazonDownloadResult, error) {
|
|||||||
if embedErr := EmbedLyrics(outputPath, parallelResult.LyricsLRC); embedErr != nil {
|
if embedErr := EmbedLyrics(outputPath, parallelResult.LyricsLRC); embedErr != nil {
|
||||||
GoLog("[Amazon] Warning: failed to embed lyrics: %v\n", embedErr)
|
GoLog("[Amazon] Warning: failed to embed lyrics: %v\n", embedErr)
|
||||||
} else {
|
} else {
|
||||||
fmt.Println("[Amazon] Lyrics embedded successfully")
|
GoLog("[Amazon] Lyrics embedded successfully\n")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if req.EmbedLyrics {
|
} else if req.EmbedLyrics {
|
||||||
fmt.Println("[Amazon] No lyrics available from parallel fetch")
|
GoLog("[Amazon] No lyrics available from parallel fetch\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
fmt.Println("[Amazon] ✓ Downloaded successfully from Amazon Music")
|
GoLog("[Amazon] Downloaded successfully from Amazon Music\n")
|
||||||
|
|
||||||
quality, err := GetAudioQuality(outputPath)
|
quality, err := GetAudioQuality(outputPath)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
+290
-69
@@ -183,10 +183,40 @@ type deezerPlaylistFull struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// NOTE: ISRC is NOT fetched during search for performance - use GetTrackISRC when needed for download
|
// NOTE: ISRC is NOT fetched during search for performance - use GetTrackISRC when needed for download
|
||||||
func (c *DeezerClient) SearchAll(ctx context.Context, query string, trackLimit, artistLimit int) (*SearchAllResult, error) {
|
// filter can be: "" (all), "track", "artist", "album", "playlist"
|
||||||
GoLog("[Deezer] SearchAll: query=%q, trackLimit=%d, artistLimit=%d\n", query, trackLimit, artistLimit)
|
func (c *DeezerClient) SearchAll(ctx context.Context, query string, trackLimit, artistLimit int, filter string) (*SearchAllResult, error) {
|
||||||
|
GoLog("[Deezer] SearchAll: query=%q, trackLimit=%d, artistLimit=%d, filter=%q\n", query, trackLimit, artistLimit, filter)
|
||||||
|
|
||||||
cacheKey := fmt.Sprintf("deezer:all:%s:%d:%d", query, trackLimit, artistLimit)
|
albumLimit := 5 // Same as artistLimit for consistency
|
||||||
|
playlistLimit := 5
|
||||||
|
|
||||||
|
// When filter is specified, increase limits for that type only
|
||||||
|
if filter != "" {
|
||||||
|
switch filter {
|
||||||
|
case "track":
|
||||||
|
trackLimit = 50
|
||||||
|
artistLimit = 0
|
||||||
|
albumLimit = 0
|
||||||
|
playlistLimit = 0
|
||||||
|
case "artist":
|
||||||
|
trackLimit = 0
|
||||||
|
artistLimit = 20
|
||||||
|
albumLimit = 0
|
||||||
|
playlistLimit = 0
|
||||||
|
case "album":
|
||||||
|
trackLimit = 0
|
||||||
|
artistLimit = 0
|
||||||
|
albumLimit = 20
|
||||||
|
playlistLimit = 0
|
||||||
|
case "playlist":
|
||||||
|
trackLimit = 0
|
||||||
|
artistLimit = 0
|
||||||
|
albumLimit = 0
|
||||||
|
playlistLimit = 20
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cacheKey := fmt.Sprintf("deezer:all:%s:%d:%d:%d:%d:%s", query, trackLimit, artistLimit, albumLimit, playlistLimit, filter)
|
||||||
|
|
||||||
c.cacheMu.RLock()
|
c.cacheMu.RLock()
|
||||||
if entry, ok := c.searchCache[cacheKey]; ok && !entry.isExpired() {
|
if entry, ok := c.searchCache[cacheKey]; ok && !entry.isExpired() {
|
||||||
@@ -197,69 +227,193 @@ func (c *DeezerClient) SearchAll(ctx context.Context, query string, trackLimit,
|
|||||||
c.cacheMu.RUnlock()
|
c.cacheMu.RUnlock()
|
||||||
|
|
||||||
result := &SearchAllResult{
|
result := &SearchAllResult{
|
||||||
Tracks: make([]TrackMetadata, 0, trackLimit),
|
Tracks: make([]TrackMetadata, 0, trackLimit),
|
||||||
Artists: make([]SearchArtistResult, 0, artistLimit),
|
Artists: make([]SearchArtistResult, 0, artistLimit),
|
||||||
|
Albums: make([]SearchAlbumResult, 0, albumLimit),
|
||||||
|
Playlists: make([]SearchPlaylistResult, 0, playlistLimit),
|
||||||
}
|
}
|
||||||
|
|
||||||
// Search tracks - NO ISRC fetch for performance
|
// Search tracks - NO ISRC fetch for performance
|
||||||
trackURL := fmt.Sprintf("%s/track?q=%s&limit=%d", deezerSearchURL, url.QueryEscape(query), trackLimit)
|
if trackLimit > 0 {
|
||||||
GoLog("[Deezer] Fetching tracks from: %s\n", trackURL)
|
trackURL := fmt.Sprintf("%s/track?q=%s&limit=%d", deezerSearchURL, url.QueryEscape(query), trackLimit)
|
||||||
|
GoLog("[Deezer] Fetching tracks from: %s\n", trackURL)
|
||||||
|
|
||||||
var trackResp struct {
|
var trackResp struct {
|
||||||
Data []deezerTrack `json:"data"`
|
Data []deezerTrack `json:"data"`
|
||||||
Error *struct {
|
Error *struct {
|
||||||
Type string `json:"type"`
|
Type string `json:"type"`
|
||||||
Message string `json:"message"`
|
Message string `json:"message"`
|
||||||
Code int `json:"code"`
|
Code int `json:"code"`
|
||||||
} `json:"error"`
|
} `json:"error"`
|
||||||
}
|
}
|
||||||
if err := c.getJSON(ctx, trackURL, &trackResp); err != nil {
|
if err := c.getJSON(ctx, trackURL, &trackResp); err != nil {
|
||||||
GoLog("[Deezer] Track search failed: %v\n", err)
|
GoLog("[Deezer] Track search failed: %v\n", err)
|
||||||
return nil, fmt.Errorf("deezer track search failed: %w", err)
|
return nil, fmt.Errorf("deezer track search failed: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if trackResp.Error != nil {
|
if trackResp.Error != nil {
|
||||||
GoLog("[Deezer] API error: type=%s, code=%d, message=%s\n", trackResp.Error.Type, trackResp.Error.Code, trackResp.Error.Message)
|
GoLog("[Deezer] API error: type=%s, code=%d, message=%s\n", trackResp.Error.Type, trackResp.Error.Code, trackResp.Error.Message)
|
||||||
return nil, fmt.Errorf("deezer API error: %s (code %d)", trackResp.Error.Message, trackResp.Error.Code)
|
return nil, fmt.Errorf("deezer API error: %s (code %d)", trackResp.Error.Message, trackResp.Error.Code)
|
||||||
}
|
}
|
||||||
|
|
||||||
GoLog("[Deezer] Got %d tracks from API\n", len(trackResp.Data))
|
GoLog("[Deezer] Got %d tracks from API\n", len(trackResp.Data))
|
||||||
|
|
||||||
for _, track := range trackResp.Data {
|
for _, track := range trackResp.Data {
|
||||||
result.Tracks = append(result.Tracks, c.convertTrack(track))
|
result.Tracks = append(result.Tracks, c.convertTrack(track))
|
||||||
}
|
|
||||||
|
|
||||||
artistURL := fmt.Sprintf("%s/artist?q=%s&limit=%d", deezerSearchURL, url.QueryEscape(query), artistLimit)
|
|
||||||
GoLog("[Deezer] Fetching artists from: %s\n", artistURL)
|
|
||||||
|
|
||||||
var artistResp struct {
|
|
||||||
Data []deezerArtist `json:"data"`
|
|
||||||
Error *struct {
|
|
||||||
Type string `json:"type"`
|
|
||||||
Message string `json:"message"`
|
|
||||||
Code int `json:"code"`
|
|
||||||
} `json:"error"`
|
|
||||||
}
|
|
||||||
if err := c.getJSON(ctx, artistURL, &artistResp); err == nil {
|
|
||||||
if artistResp.Error != nil {
|
|
||||||
GoLog("[Deezer] Artist API error: type=%s, code=%d, message=%s\n", artistResp.Error.Type, artistResp.Error.Code, artistResp.Error.Message)
|
|
||||||
} else {
|
|
||||||
GoLog("[Deezer] Got %d artists from API\n", len(artistResp.Data))
|
|
||||||
for _, artist := range artistResp.Data {
|
|
||||||
result.Artists = append(result.Artists, SearchArtistResult{
|
|
||||||
ID: fmt.Sprintf("deezer:%d", artist.ID),
|
|
||||||
Name: artist.Name,
|
|
||||||
Images: c.getBestArtistImage(artist),
|
|
||||||
Followers: artist.NbFan,
|
|
||||||
Popularity: 0,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
GoLog("[Deezer] Artist search failed: %v\n", err)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
GoLog("[Deezer] SearchAll complete: %d tracks, %d artists\n", len(result.Tracks), len(result.Artists))
|
// Search artists
|
||||||
|
if artistLimit > 0 {
|
||||||
|
artistURL := fmt.Sprintf("%s/artist?q=%s&limit=%d", deezerSearchURL, url.QueryEscape(query), artistLimit)
|
||||||
|
GoLog("[Deezer] Fetching artists from: %s\n", artistURL)
|
||||||
|
|
||||||
|
var artistResp struct {
|
||||||
|
Data []deezerArtist `json:"data"`
|
||||||
|
Error *struct {
|
||||||
|
Type string `json:"type"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Code int `json:"code"`
|
||||||
|
} `json:"error"`
|
||||||
|
}
|
||||||
|
if err := c.getJSON(ctx, artistURL, &artistResp); err == nil {
|
||||||
|
if artistResp.Error != nil {
|
||||||
|
GoLog("[Deezer] Artist API error: type=%s, code=%d, message=%s\n", artistResp.Error.Type, artistResp.Error.Code, artistResp.Error.Message)
|
||||||
|
} else {
|
||||||
|
GoLog("[Deezer] Got %d artists from API\n", len(artistResp.Data))
|
||||||
|
for _, artist := range artistResp.Data {
|
||||||
|
result.Artists = append(result.Artists, SearchArtistResult{
|
||||||
|
ID: fmt.Sprintf("deezer:%d", artist.ID),
|
||||||
|
Name: artist.Name,
|
||||||
|
Images: c.getBestArtistImage(artist),
|
||||||
|
Followers: artist.NbFan,
|
||||||
|
Popularity: 0,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
GoLog("[Deezer] Artist search failed: %v\n", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Search albums
|
||||||
|
if albumLimit > 0 {
|
||||||
|
albumURL := fmt.Sprintf("%s/album?q=%s&limit=%d", deezerSearchURL, url.QueryEscape(query), albumLimit)
|
||||||
|
GoLog("[Deezer] Fetching albums from: %s\n", albumURL)
|
||||||
|
|
||||||
|
var albumResp struct {
|
||||||
|
Data []struct {
|
||||||
|
ID int64 `json:"id"`
|
||||||
|
Title string `json:"title"`
|
||||||
|
Cover string `json:"cover"`
|
||||||
|
CoverMedium string `json:"cover_medium"`
|
||||||
|
CoverBig string `json:"cover_big"`
|
||||||
|
CoverXL string `json:"cover_xl"`
|
||||||
|
NbTracks int `json:"nb_tracks"`
|
||||||
|
ReleaseDate string `json:"release_date"`
|
||||||
|
RecordType string `json:"record_type"`
|
||||||
|
Artist deezerArtist `json:"artist"`
|
||||||
|
} `json:"data"`
|
||||||
|
Error *struct {
|
||||||
|
Type string `json:"type"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Code int `json:"code"`
|
||||||
|
} `json:"error"`
|
||||||
|
}
|
||||||
|
if err := c.getJSON(ctx, albumURL, &albumResp); err == nil {
|
||||||
|
if albumResp.Error != nil {
|
||||||
|
GoLog("[Deezer] Album API error: type=%s, code=%d, message=%s\n", albumResp.Error.Type, albumResp.Error.Code, albumResp.Error.Message)
|
||||||
|
} else {
|
||||||
|
GoLog("[Deezer] Got %d albums from API\n", len(albumResp.Data))
|
||||||
|
for _, album := range albumResp.Data {
|
||||||
|
coverURL := album.CoverXL
|
||||||
|
if coverURL == "" {
|
||||||
|
coverURL = album.CoverBig
|
||||||
|
}
|
||||||
|
if coverURL == "" {
|
||||||
|
coverURL = album.CoverMedium
|
||||||
|
}
|
||||||
|
if coverURL == "" {
|
||||||
|
coverURL = album.Cover
|
||||||
|
}
|
||||||
|
|
||||||
|
albumType := album.RecordType
|
||||||
|
if albumType == "compile" {
|
||||||
|
albumType = "compilation"
|
||||||
|
}
|
||||||
|
|
||||||
|
result.Albums = append(result.Albums, SearchAlbumResult{
|
||||||
|
ID: fmt.Sprintf("deezer:%d", album.ID),
|
||||||
|
Name: album.Title,
|
||||||
|
Artists: album.Artist.Name,
|
||||||
|
Images: coverURL,
|
||||||
|
ReleaseDate: album.ReleaseDate,
|
||||||
|
TotalTracks: album.NbTracks,
|
||||||
|
AlbumType: albumType,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
GoLog("[Deezer] Album search failed: %v\n", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Search playlists
|
||||||
|
if playlistLimit > 0 {
|
||||||
|
playlistURL := fmt.Sprintf("%s/playlist?q=%s&limit=%d", deezerSearchURL, url.QueryEscape(query), playlistLimit)
|
||||||
|
GoLog("[Deezer] Fetching playlists from: %s\n", playlistURL)
|
||||||
|
|
||||||
|
var playlistResp struct {
|
||||||
|
Data []struct {
|
||||||
|
ID int64 `json:"id"`
|
||||||
|
Title string `json:"title"`
|
||||||
|
Picture string `json:"picture"`
|
||||||
|
PictureMedium string `json:"picture_medium"`
|
||||||
|
PictureBig string `json:"picture_big"`
|
||||||
|
PictureXL string `json:"picture_xl"`
|
||||||
|
NbTracks int `json:"nb_tracks"`
|
||||||
|
User struct {
|
||||||
|
Name string `json:"name"`
|
||||||
|
} `json:"user"`
|
||||||
|
} `json:"data"`
|
||||||
|
Error *struct {
|
||||||
|
Type string `json:"type"`
|
||||||
|
Message string `json:"message"`
|
||||||
|
Code int `json:"code"`
|
||||||
|
} `json:"error"`
|
||||||
|
}
|
||||||
|
if err := c.getJSON(ctx, playlistURL, &playlistResp); err == nil {
|
||||||
|
if playlistResp.Error != nil {
|
||||||
|
GoLog("[Deezer] Playlist API error: type=%s, code=%d, message=%s\n", playlistResp.Error.Type, playlistResp.Error.Code, playlistResp.Error.Message)
|
||||||
|
} else {
|
||||||
|
GoLog("[Deezer] Got %d playlists from API\n", len(playlistResp.Data))
|
||||||
|
for _, playlist := range playlistResp.Data {
|
||||||
|
pictureURL := playlist.PictureXL
|
||||||
|
if pictureURL == "" {
|
||||||
|
pictureURL = playlist.PictureBig
|
||||||
|
}
|
||||||
|
if pictureURL == "" {
|
||||||
|
pictureURL = playlist.PictureMedium
|
||||||
|
}
|
||||||
|
if pictureURL == "" {
|
||||||
|
pictureURL = playlist.Picture
|
||||||
|
}
|
||||||
|
|
||||||
|
result.Playlists = append(result.Playlists, SearchPlaylistResult{
|
||||||
|
ID: fmt.Sprintf("deezer:%d", playlist.ID),
|
||||||
|
Name: playlist.Title,
|
||||||
|
Owner: playlist.User.Name,
|
||||||
|
Images: pictureURL,
|
||||||
|
TotalTracks: playlist.NbTracks,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
GoLog("[Deezer] Playlist search failed: %v\n", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
GoLog("[Deezer] SearchAll complete: %d tracks, %d artists, %d albums, %d playlists\n", len(result.Tracks), len(result.Artists), len(result.Albums), len(result.Playlists))
|
||||||
|
|
||||||
c.cacheMu.Lock()
|
c.cacheMu.Lock()
|
||||||
c.searchCache[cacheKey] = &cacheEntry{
|
c.searchCache[cacheKey] = &cacheEntry{
|
||||||
@@ -331,16 +485,51 @@ func (c *DeezerClient) GetAlbum(ctx context.Context, albumID string) (*AlbumResp
|
|||||||
Label: album.Label, // From Deezer album
|
Label: album.Label, // From Deezer album
|
||||||
}
|
}
|
||||||
|
|
||||||
isrcMap := c.fetchISRCsParallel(ctx, album.Tracks.Data)
|
// Fetch all tracks with pagination (Deezer default limit is 25)
|
||||||
|
allTracks := album.Tracks.Data
|
||||||
|
|
||||||
tracks := make([]AlbumTrackMetadata, 0, len(album.Tracks.Data))
|
// If album has more tracks than returned, fetch remaining pages
|
||||||
|
if album.NbTracks > len(allTracks) {
|
||||||
|
GoLog("[Deezer] Album has %d tracks but only got %d, fetching remaining...", album.NbTracks, len(allTracks))
|
||||||
|
|
||||||
|
tracksURL := fmt.Sprintf("%s/tracks?limit=100&index=%d", fmt.Sprintf(deezerAlbumURL, albumID), len(allTracks))
|
||||||
|
|
||||||
|
for len(allTracks) < album.NbTracks {
|
||||||
|
var tracksResp struct {
|
||||||
|
Data []deezerTrack `json:"data"`
|
||||||
|
Next string `json:"next"`
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := c.getJSON(ctx, tracksURL, &tracksResp); err != nil {
|
||||||
|
GoLog("[Deezer] Warning: failed to fetch album tracks page: %v", err)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(tracksResp.Data) == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
allTracks = append(allTracks, tracksResp.Data...)
|
||||||
|
|
||||||
|
if tracksResp.Next == "" {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
tracksURL = tracksResp.Next
|
||||||
|
}
|
||||||
|
|
||||||
|
GoLog("[Deezer] Fetched total %d tracks for album", len(allTracks))
|
||||||
|
}
|
||||||
|
|
||||||
|
isrcMap := c.fetchISRCsParallel(ctx, allTracks)
|
||||||
|
|
||||||
|
tracks := make([]AlbumTrackMetadata, 0, len(allTracks))
|
||||||
// Normalize record_type (Deezer uses "compile" instead of "compilation")
|
// Normalize record_type (Deezer uses "compile" instead of "compilation")
|
||||||
albumType := album.RecordType
|
albumType := album.RecordType
|
||||||
if albumType == "compile" {
|
if albumType == "compile" {
|
||||||
albumType = "compilation"
|
albumType = "compilation"
|
||||||
}
|
}
|
||||||
|
|
||||||
for i, track := range album.Tracks.Data {
|
for i, track := range allTracks {
|
||||||
trackIDStr := fmt.Sprintf("%d", track.ID)
|
trackIDStr := fmt.Sprintf("%d", track.ID)
|
||||||
isrc := isrcMap[trackIDStr]
|
isrc := isrcMap[trackIDStr]
|
||||||
|
|
||||||
@@ -491,10 +680,45 @@ func (c *DeezerClient) GetPlaylist(ctx context.Context, playlistID string) (*Pla
|
|||||||
info.Owner.Name = playlist.Title
|
info.Owner.Name = playlist.Title
|
||||||
info.Owner.Images = playlistImage
|
info.Owner.Images = playlistImage
|
||||||
|
|
||||||
isrcMap := c.fetchISRCsParallel(ctx, playlist.Tracks.Data)
|
// Fetch all tracks with pagination (Deezer default limit is 25)
|
||||||
|
allTracks := playlist.Tracks.Data
|
||||||
|
|
||||||
tracks := make([]AlbumTrackMetadata, 0, len(playlist.Tracks.Data))
|
// If playlist has more tracks than returned, fetch remaining pages
|
||||||
for _, track := range playlist.Tracks.Data {
|
if playlist.NbTracks > len(allTracks) {
|
||||||
|
GoLog("[Deezer] Playlist has %d tracks but only got %d, fetching remaining...", playlist.NbTracks, len(allTracks))
|
||||||
|
|
||||||
|
tracksURL := fmt.Sprintf("%s/tracks?limit=100&index=%d", fmt.Sprintf(deezerPlaylistURL, playlistID), len(allTracks))
|
||||||
|
|
||||||
|
for len(allTracks) < playlist.NbTracks {
|
||||||
|
var tracksResp struct {
|
||||||
|
Data []deezerTrack `json:"data"`
|
||||||
|
Next string `json:"next"`
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := c.getJSON(ctx, tracksURL, &tracksResp); err != nil {
|
||||||
|
GoLog("[Deezer] Warning: failed to fetch playlist tracks page: %v", err)
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(tracksResp.Data) == 0 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
allTracks = append(allTracks, tracksResp.Data...)
|
||||||
|
|
||||||
|
if tracksResp.Next == "" {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
tracksURL = tracksResp.Next
|
||||||
|
}
|
||||||
|
|
||||||
|
GoLog("[Deezer] Fetched total %d tracks for playlist", len(allTracks))
|
||||||
|
}
|
||||||
|
|
||||||
|
isrcMap := c.fetchISRCsParallel(ctx, allTracks)
|
||||||
|
|
||||||
|
tracks := make([]AlbumTrackMetadata, 0, len(allTracks))
|
||||||
|
for _, track := range allTracks {
|
||||||
albumImage := track.Album.CoverXL
|
albumImage := track.Album.CoverXL
|
||||||
if albumImage == "" {
|
if albumImage == "" {
|
||||||
albumImage = track.Album.CoverBig
|
albumImage = track.Album.CoverBig
|
||||||
@@ -786,10 +1010,7 @@ func (c *DeezerClient) GetExtendedMetadataByISRC(ctx context.Context, isrc strin
|
|||||||
}
|
}
|
||||||
|
|
||||||
// SpotifyID contains "deezer:123" format, extract the ID
|
// SpotifyID contains "deezer:123" format, extract the ID
|
||||||
deezerID := track.SpotifyID
|
deezerID := strings.TrimPrefix(track.SpotifyID, "deezer:")
|
||||||
if strings.HasPrefix(deezerID, "deezer:") {
|
|
||||||
deezerID = strings.TrimPrefix(deezerID, "deezer:")
|
|
||||||
}
|
|
||||||
|
|
||||||
if deezerID == "" {
|
if deezerID == "" {
|
||||||
return nil, fmt.Errorf("track found but no Deezer ID")
|
return nil, fmt.Errorf("track found but no Deezer ID")
|
||||||
|
|||||||
@@ -716,12 +716,12 @@ func ClearTrackIDCache() {
|
|||||||
ClearTrackCache()
|
ClearTrackCache()
|
||||||
}
|
}
|
||||||
|
|
||||||
func SearchDeezerAll(query string, trackLimit, artistLimit int) (string, error) {
|
func SearchDeezerAll(query string, trackLimit, artistLimit int, filter string) (string, error) {
|
||||||
ctx, cancel := context.WithTimeout(context.Background(), 15*time.Second)
|
ctx, cancel := context.WithTimeout(context.Background(), 15*time.Second)
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
client := GetDeezerClient()
|
client := GetDeezerClient()
|
||||||
results, err := client.SearchAll(ctx, query, trackLimit, artistLimit)
|
results, err := client.SearchAll(ctx, query, trackLimit, artistLimit, filter)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -66,15 +66,23 @@ type QualitySpecificSetting struct {
|
|||||||
Options []string `json:"options,omitempty"` // For select type
|
Options []string `json:"options,omitempty"` // For select type
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SearchFilter defines a filter option for search
|
||||||
|
type SearchFilter struct {
|
||||||
|
ID string `json:"id"` // Filter identifier (e.g., "track", "album", "artist", "playlist")
|
||||||
|
Label string `json:"label,omitempty"` // Display label (e.g., "Songs", "Albums", "Artists", "Playlists")
|
||||||
|
Icon string `json:"icon,omitempty"` // Optional icon name
|
||||||
|
}
|
||||||
|
|
||||||
// SearchBehaviorConfig defines custom search behavior for an extension
|
// SearchBehaviorConfig defines custom search behavior for an extension
|
||||||
type SearchBehaviorConfig struct {
|
type SearchBehaviorConfig struct {
|
||||||
Enabled bool `json:"enabled"` // Whether extension provides custom search
|
Enabled bool `json:"enabled"` // Whether extension provides custom search
|
||||||
Placeholder string `json:"placeholder,omitempty"` // Placeholder text for search box
|
Placeholder string `json:"placeholder,omitempty"` // Placeholder text for search box
|
||||||
Primary bool `json:"primary,omitempty"` // If true, show as primary search tab
|
Primary bool `json:"primary,omitempty"` // If true, show as primary search tab
|
||||||
Icon string `json:"icon,omitempty"` // Icon for search tab
|
Icon string `json:"icon,omitempty"` // Icon for search tab
|
||||||
ThumbnailRatio string `json:"thumbnailRatio,omitempty"` // Thumbnail aspect ratio: "square" (1:1), "wide" (16:9), "portrait" (2:3)
|
ThumbnailRatio string `json:"thumbnailRatio,omitempty"` // Thumbnail aspect ratio: "square" (1:1), "wide" (16:9), "portrait" (2:3)
|
||||||
ThumbnailWidth int `json:"thumbnailWidth,omitempty"` // Custom thumbnail width in pixels
|
ThumbnailWidth int `json:"thumbnailWidth,omitempty"` // Custom thumbnail width in pixels
|
||||||
ThumbnailHeight int `json:"thumbnailHeight,omitempty"` // Custom thumbnail height in pixels
|
ThumbnailHeight int `json:"thumbnailHeight,omitempty"` // Custom thumbnail height in pixels
|
||||||
|
Filters []SearchFilter `json:"filters,omitempty"` // Available search filters (e.g., track, album, artist, playlist)
|
||||||
}
|
}
|
||||||
|
|
||||||
// URLHandlerConfig defines custom URL handling for an extension
|
// URLHandlerConfig defines custom URL handling for an extension
|
||||||
|
|||||||
+12
-6
@@ -1,23 +1,29 @@
|
|||||||
module github.com/zarz/spotiflac_android/go_backend
|
module github.com/zarz/spotiflac_android/go_backend
|
||||||
|
|
||||||
go 1.24.0
|
go 1.25.0
|
||||||
|
|
||||||
toolchain go1.24.5
|
toolchain go1.25.6
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/dop251/goja v0.0.0-20260106131823-651366fbe6e3
|
github.com/dop251/goja v0.0.0-20260106131823-651366fbe6e3
|
||||||
github.com/go-flac/flacpicture v0.3.0
|
github.com/go-flac/flacpicture v0.3.0
|
||||||
github.com/go-flac/flacvorbis v0.2.0
|
github.com/go-flac/flacvorbis v0.2.0
|
||||||
github.com/go-flac/go-flac v1.0.0
|
github.com/go-flac/go-flac v1.0.0
|
||||||
|
github.com/refraction-networking/utls v1.8.2
|
||||||
|
golang.org/x/mobile v0.0.0-20260120165949-40bd9ace6ce4
|
||||||
|
golang.org/x/net v0.49.0
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
github.com/andybalholm/brotli v1.0.6 // indirect
|
||||||
github.com/dlclark/regexp2 v1.11.4 // indirect
|
github.com/dlclark/regexp2 v1.11.4 // indirect
|
||||||
github.com/go-sourcemap/sourcemap v2.1.3+incompatible // indirect
|
github.com/go-sourcemap/sourcemap v2.1.3+incompatible // indirect
|
||||||
github.com/google/pprof v0.0.0-20230207041349-798e818bf904 // indirect
|
github.com/google/pprof v0.0.0-20230207041349-798e818bf904 // indirect
|
||||||
golang.org/x/mobile v0.0.0-20251209145715-2553ed8ce294 // indirect
|
github.com/klauspost/compress v1.17.4 // indirect
|
||||||
golang.org/x/mod v0.31.0 // indirect
|
golang.org/x/crypto v0.47.0 // indirect
|
||||||
|
golang.org/x/mod v0.32.0 // indirect
|
||||||
golang.org/x/sync v0.19.0 // indirect
|
golang.org/x/sync v0.19.0 // indirect
|
||||||
golang.org/x/text v0.3.8 // indirect
|
golang.org/x/sys v0.40.0 // indirect
|
||||||
golang.org/x/tools v0.40.0 // indirect
|
golang.org/x/text v0.33.0 // indirect
|
||||||
|
golang.org/x/tools v0.41.0 // indirect
|
||||||
)
|
)
|
||||||
|
|||||||
+22
-8
@@ -1,5 +1,7 @@
|
|||||||
github.com/Masterminds/semver/v3 v3.2.1 h1:RN9w6+7QoMeJVGyfmbcgs28Br8cvmnucEXnY0rYXWg0=
|
github.com/Masterminds/semver/v3 v3.2.1 h1:RN9w6+7QoMeJVGyfmbcgs28Br8cvmnucEXnY0rYXWg0=
|
||||||
github.com/Masterminds/semver/v3 v3.2.1/go.mod h1:qvl/7zhW3nngYb5+80sSMF+FG2BjYrf8m9wsX0PNOMQ=
|
github.com/Masterminds/semver/v3 v3.2.1/go.mod h1:qvl/7zhW3nngYb5+80sSMF+FG2BjYrf8m9wsX0PNOMQ=
|
||||||
|
github.com/andybalholm/brotli v1.0.6 h1:Yf9fFpf49Zrxb9NlQaluyE92/+X7UVHlhMNJN2sxfOI=
|
||||||
|
github.com/andybalholm/brotli v1.0.6/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig=
|
||||||
github.com/dlclark/regexp2 v1.11.4 h1:rPYF9/LECdNymJufQKmri9gV604RvvABwgOA8un7yAo=
|
github.com/dlclark/regexp2 v1.11.4 h1:rPYF9/LECdNymJufQKmri9gV604RvvABwgOA8un7yAo=
|
||||||
github.com/dlclark/regexp2 v1.11.4/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8=
|
github.com/dlclark/regexp2 v1.11.4/go.mod h1:DHkYz0B9wPfa6wondMfaivmHpzrQ3v9q8cnmRbL6yW8=
|
||||||
github.com/dop251/goja v0.0.0-20260106131823-651366fbe6e3 h1:bVp3yUzvSAJzu9GqID+Z96P+eu5TKnIMJSV4QaZMauM=
|
github.com/dop251/goja v0.0.0-20260106131823-651366fbe6e3 h1:bVp3yUzvSAJzu9GqID+Z96P+eu5TKnIMJSV4QaZMauM=
|
||||||
@@ -12,17 +14,29 @@ github.com/go-flac/go-flac v1.0.0 h1:6qI9XOVLcO50xpzm3nXvO31BgDgHhnr/p/rER/K/doY
|
|||||||
github.com/go-flac/go-flac v1.0.0/go.mod h1:WnZhcpmq4u1UdZMNn9LYSoASpWOCMOoxXxcWEHSzkW8=
|
github.com/go-flac/go-flac v1.0.0/go.mod h1:WnZhcpmq4u1UdZMNn9LYSoASpWOCMOoxXxcWEHSzkW8=
|
||||||
github.com/go-sourcemap/sourcemap v2.1.3+incompatible h1:W1iEw64niKVGogNgBN3ePyLFfuisuzeidWPMPWmECqU=
|
github.com/go-sourcemap/sourcemap v2.1.3+incompatible h1:W1iEw64niKVGogNgBN3ePyLFfuisuzeidWPMPWmECqU=
|
||||||
github.com/go-sourcemap/sourcemap v2.1.3+incompatible/go.mod h1:F8jJfvm2KbVjc5NqelyYJmf/v5J0dwNLS2mL4sNA1Jg=
|
github.com/go-sourcemap/sourcemap v2.1.3+incompatible/go.mod h1:F8jJfvm2KbVjc5NqelyYJmf/v5J0dwNLS2mL4sNA1Jg=
|
||||||
|
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
|
||||||
|
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
|
||||||
github.com/google/pprof v0.0.0-20230207041349-798e818bf904 h1:4/hN5RUoecvl+RmJRE2YxKWtnnQls6rQjjW5oV7qg2U=
|
github.com/google/pprof v0.0.0-20230207041349-798e818bf904 h1:4/hN5RUoecvl+RmJRE2YxKWtnnQls6rQjjW5oV7qg2U=
|
||||||
github.com/google/pprof v0.0.0-20230207041349-798e818bf904/go.mod h1:uglQLonpP8qtYCYyzA+8c/9qtqgA3qsXGYqCPKARAFg=
|
github.com/google/pprof v0.0.0-20230207041349-798e818bf904/go.mod h1:uglQLonpP8qtYCYyzA+8c/9qtqgA3qsXGYqCPKARAFg=
|
||||||
golang.org/x/mobile v0.0.0-20251209145715-2553ed8ce294 h1:Cr6kbEvA6nqvdHynE4CtVKlqpZB9dS1Jva/6IsHA19g=
|
github.com/klauspost/compress v1.17.4 h1:Ej5ixsIri7BrIjBkRZLTo6ghwrEtHFk7ijlczPW4fZ4=
|
||||||
golang.org/x/mobile v0.0.0-20251209145715-2553ed8ce294/go.mod h1:RdZ+3sb4CVgpCFnzv+I4haEpwqFfsfzlLHs3L7ok+e0=
|
github.com/klauspost/compress v1.17.4/go.mod h1:/dCuZOvVtNoHsyb+cuJD3itjs3NbnF6KH9zAO4BDxPM=
|
||||||
golang.org/x/mod v0.31.0 h1:HaW9xtz0+kOcWKwli0ZXy79Ix+UW/vOfmWI5QVd2tgI=
|
github.com/refraction-networking/utls v1.8.2 h1:j4Q1gJj0xngdeH+Ox/qND11aEfhpgoEvV+S9iJ2IdQo=
|
||||||
golang.org/x/mod v0.31.0/go.mod h1:43JraMp9cGx1Rx3AqioxrbrhNsLl2l/iNAvuBkrezpg=
|
github.com/refraction-networking/utls v1.8.2/go.mod h1:jkSOEkLqn+S/jtpEHPOsVv/4V4EVnelwbMQl4vCWXAM=
|
||||||
|
golang.org/x/crypto v0.47.0 h1:V6e3FRj+n4dbpw86FJ8Fv7XVOql7TEwpHapKoMJ/GO8=
|
||||||
|
golang.org/x/crypto v0.47.0/go.mod h1:ff3Y9VzzKbwSSEzWqJsJVBnWmRwRSHt/6Op5n9bQc4A=
|
||||||
|
golang.org/x/mobile v0.0.0-20260120165949-40bd9ace6ce4 h1:C3JuLOLhdaE75vk5m7u18NvZciRk+lnO34xcXl3NPTU=
|
||||||
|
golang.org/x/mobile v0.0.0-20260120165949-40bd9ace6ce4/go.mod h1:yHJY0EGzMJ0i5ONrrhdpDSSnoyres5LO7D2hSIbJJ5I=
|
||||||
|
golang.org/x/mod v0.32.0 h1:9F4d3PHLljb6x//jOyokMv3eX+YDeepZSEo3mFJy93c=
|
||||||
|
golang.org/x/mod v0.32.0/go.mod h1:SgipZ/3h2Ci89DlEtEXWUk/HteuRin+HHhN+WbNhguU=
|
||||||
|
golang.org/x/net v0.49.0 h1:eeHFmOGUTtaaPSGNmjBKpbng9MulQsJURQUAfUwY++o=
|
||||||
|
golang.org/x/net v0.49.0/go.mod h1:/ysNB2EvaqvesRkuLAyjI1ycPZlQHM3q01F02UY/MV8=
|
||||||
golang.org/x/sync v0.19.0 h1:vV+1eWNmZ5geRlYjzm2adRgW2/mcpevXNg50YZtPCE4=
|
golang.org/x/sync v0.19.0 h1:vV+1eWNmZ5geRlYjzm2adRgW2/mcpevXNg50YZtPCE4=
|
||||||
golang.org/x/sync v0.19.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
|
golang.org/x/sync v0.19.0/go.mod h1:9KTHXmSnoGruLpwFjVSX0lNNA75CykiMECbovNTZqGI=
|
||||||
golang.org/x/text v0.3.8 h1:nAL+RVCQ9uMn3vJZbV+MRnydTJFPf8qqY42YiA6MrqY=
|
golang.org/x/sys v0.40.0 h1:DBZZqJ2Rkml6QMQsZywtnjnnGvHza6BTfYFWY9kjEWQ=
|
||||||
golang.org/x/text v0.3.8/go.mod h1:E6s5w1FMmriuDzIBO73fBruAKo1PCIq6d2Q6DHfQ8WQ=
|
golang.org/x/sys v0.40.0/go.mod h1:OgkHotnGiDImocRcuBABYBEXf8A9a87e/uXjp9XT3ks=
|
||||||
golang.org/x/tools v0.40.0 h1:yLkxfA+Qnul4cs9QA3KnlFu0lVmd8JJfoq+E41uSutA=
|
golang.org/x/text v0.33.0 h1:B3njUFyqtHDUI5jMn1YIr5B0IE2U0qck04r6d4KPAxE=
|
||||||
golang.org/x/tools v0.40.0/go.mod h1:Ik/tzLRlbscWpqqMRjyWYDisX8bG13FrdXp3o4Sr9lc=
|
golang.org/x/text v0.33.0/go.mod h1:LuMebE6+rBincTi9+xWTY8TztLzKHc/9C1uBCG27+q8=
|
||||||
|
golang.org/x/tools v0.41.0 h1:a9b8iMweWG+S0OBnlU36rzLp20z1Rp10w+IY2czHTQc=
|
||||||
|
golang.org/x/tools v0.41.0/go.mod h1:XSY6eDqxVNiYgezAVqqCeihT4j1U2CCsqvH3WhQpnlg=
|
||||||
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
|
gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY=
|
||||||
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
|
gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ=
|
||||||
|
|||||||
@@ -38,6 +38,7 @@ const (
|
|||||||
SongLinkTimeout = 30 * time.Second
|
SongLinkTimeout = 30 * time.Second
|
||||||
DefaultMaxRetries = 3
|
DefaultMaxRetries = 3
|
||||||
DefaultRetryDelay = 1 * time.Second
|
DefaultRetryDelay = 1 * time.Second
|
||||||
|
Second = time.Second // Exported for use in other files
|
||||||
)
|
)
|
||||||
|
|
||||||
// Shared transport with connection pooling to prevent TCP exhaustion
|
// Shared transport with connection pooling to prevent TCP exhaustion
|
||||||
|
|||||||
@@ -0,0 +1,27 @@
|
|||||||
|
//go:build ios
|
||||||
|
|
||||||
|
package gobackend
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
)
|
||||||
|
|
||||||
|
// iOS version: uTLS is not supported on iOS due to cgo DNS resolver issues
|
||||||
|
// Fall back to standard HTTP client
|
||||||
|
|
||||||
|
// GetCloudflareBypassClient returns the standard HTTP client on iOS
|
||||||
|
// uTLS is not available on iOS due to cgo DNS resolver compatibility issues
|
||||||
|
func GetCloudflareBypassClient() *http.Client {
|
||||||
|
return sharedClient
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoRequestWithCloudflareBypass on iOS just uses the standard client
|
||||||
|
// uTLS Chrome fingerprint bypass is not available on iOS
|
||||||
|
func DoRequestWithCloudflareBypass(req *http.Request) (*http.Response, error) {
|
||||||
|
req.Header.Set("User-Agent", getRandomUserAgent())
|
||||||
|
resp, err := sharedClient.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
CheckAndLogISPBlocking(err, req.URL.String(), "HTTP")
|
||||||
|
}
|
||||||
|
return resp, err
|
||||||
|
}
|
||||||
@@ -0,0 +1,188 @@
|
|||||||
|
//go:build !ios
|
||||||
|
|
||||||
|
package gobackend
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"crypto/tls"
|
||||||
|
"io"
|
||||||
|
"net"
|
||||||
|
"net/http"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
|
||||||
|
utls "github.com/refraction-networking/utls"
|
||||||
|
"golang.org/x/net/http2"
|
||||||
|
)
|
||||||
|
|
||||||
|
// uTLS transport that mimics Chrome's TLS fingerprint to bypass Cloudflare
|
||||||
|
// Uses HTTP/2 for optimal performance as uTLS works best with HTTP/2
|
||||||
|
type utlsTransport struct {
|
||||||
|
dialer *net.Dialer
|
||||||
|
mu sync.Mutex
|
||||||
|
h2Transports map[string]*http2.Transport
|
||||||
|
}
|
||||||
|
|
||||||
|
func newUTLSTransport() *utlsTransport {
|
||||||
|
return &utlsTransport{
|
||||||
|
dialer: &net.Dialer{
|
||||||
|
Timeout: 30 * Second,
|
||||||
|
KeepAlive: 30 * Second,
|
||||||
|
},
|
||||||
|
h2Transports: make(map[string]*http2.Transport),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *utlsTransport) RoundTrip(req *http.Request) (*http.Response, error) {
|
||||||
|
// For non-HTTPS, use standard transport
|
||||||
|
if req.URL.Scheme != "https" {
|
||||||
|
return sharedTransport.RoundTrip(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
host := req.URL.Hostname()
|
||||||
|
port := t.getPort(req.URL)
|
||||||
|
addr := net.JoinHostPort(host, port)
|
||||||
|
|
||||||
|
// Dial TCP connection
|
||||||
|
conn, err := t.dialer.DialContext(req.Context(), "tcp", addr)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create uTLS connection with Chrome fingerprint (supports HTTP/2 ALPN)
|
||||||
|
tlsConn := utls.UClient(conn, &utls.Config{
|
||||||
|
ServerName: host,
|
||||||
|
NextProtos: []string{"h2", "http/1.1"}, // Prefer HTTP/2
|
||||||
|
}, utls.HelloChrome_Auto)
|
||||||
|
|
||||||
|
// Perform TLS handshake
|
||||||
|
if err := tlsConn.Handshake(); err != nil {
|
||||||
|
conn.Close()
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if server supports HTTP/2
|
||||||
|
negotiatedProto := tlsConn.ConnectionState().NegotiatedProtocol
|
||||||
|
|
||||||
|
if negotiatedProto == "h2" {
|
||||||
|
// Use HTTP/2 transport
|
||||||
|
h2Transport := &http2.Transport{
|
||||||
|
DialTLSContext: func(ctx context.Context, network, addr string, cfg *tls.Config) (net.Conn, error) {
|
||||||
|
return tlsConn, nil
|
||||||
|
},
|
||||||
|
AllowHTTP: false,
|
||||||
|
DisableCompression: false,
|
||||||
|
}
|
||||||
|
return h2Transport.RoundTrip(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback to HTTP/1.1
|
||||||
|
transport := &http.Transport{
|
||||||
|
DialTLSContext: func(ctx context.Context, network, addr string) (net.Conn, error) {
|
||||||
|
return tlsConn, nil
|
||||||
|
},
|
||||||
|
DisableKeepAlives: true,
|
||||||
|
}
|
||||||
|
|
||||||
|
return transport.RoundTrip(req)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (t *utlsTransport) getPort(u *url.URL) string {
|
||||||
|
if u.Port() != "" {
|
||||||
|
return u.Port()
|
||||||
|
}
|
||||||
|
if u.Scheme == "https" {
|
||||||
|
return "443"
|
||||||
|
}
|
||||||
|
return "80"
|
||||||
|
}
|
||||||
|
|
||||||
|
// Cloudflare bypass client using uTLS Chrome fingerprint
|
||||||
|
var cloudflareBypassTransport = newUTLSTransport()
|
||||||
|
|
||||||
|
var cloudflareBypassClient = &http.Client{
|
||||||
|
Transport: cloudflareBypassTransport,
|
||||||
|
Timeout: DefaultTimeout,
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetCloudflareBypassClient returns an HTTP client that mimics Chrome's TLS fingerprint
|
||||||
|
// Use this when requests are blocked by Cloudflare (common when using VPN)
|
||||||
|
func GetCloudflareBypassClient() *http.Client {
|
||||||
|
return cloudflareBypassClient
|
||||||
|
}
|
||||||
|
|
||||||
|
// DoRequestWithCloudflareBypass attempts request with standard client first,
|
||||||
|
// then retries with uTLS Chrome fingerprint if Cloudflare blocks it.
|
||||||
|
// This is useful when using VPN as Cloudflare detects Go's default TLS fingerprint.
|
||||||
|
func DoRequestWithCloudflareBypass(req *http.Request) (*http.Response, error) {
|
||||||
|
req.Header.Set("User-Agent", getRandomUserAgent())
|
||||||
|
|
||||||
|
// Try with standard client first
|
||||||
|
resp, err := sharedClient.Do(req)
|
||||||
|
if err == nil {
|
||||||
|
// Check for Cloudflare challenge page (403 with specific markers)
|
||||||
|
if resp.StatusCode == 403 || resp.StatusCode == 503 {
|
||||||
|
body, readErr := io.ReadAll(resp.Body)
|
||||||
|
resp.Body.Close()
|
||||||
|
|
||||||
|
if readErr == nil {
|
||||||
|
bodyStr := strings.ToLower(string(body))
|
||||||
|
cloudflareMarkers := []string{
|
||||||
|
"cloudflare", "cf-ray", "checking your browser",
|
||||||
|
"please wait", "ddos protection", "ray id",
|
||||||
|
"enable javascript", "challenge-platform",
|
||||||
|
}
|
||||||
|
|
||||||
|
isCloudflare := false
|
||||||
|
for _, marker := range cloudflareMarkers {
|
||||||
|
if strings.Contains(bodyStr, marker) {
|
||||||
|
isCloudflare = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if isCloudflare {
|
||||||
|
LogDebug("HTTP", "Cloudflare detected, retrying with Chrome TLS fingerprint...")
|
||||||
|
|
||||||
|
// Clone request for retry
|
||||||
|
reqCopy := req.Clone(req.Context())
|
||||||
|
reqCopy.Header.Set("User-Agent", getRandomUserAgent())
|
||||||
|
|
||||||
|
// Retry with uTLS Chrome fingerprint
|
||||||
|
return cloudflareBypassClient.Do(reqCopy)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Not Cloudflare, return original response (recreate body)
|
||||||
|
return &http.Response{
|
||||||
|
Status: resp.Status,
|
||||||
|
StatusCode: resp.StatusCode,
|
||||||
|
Header: resp.Header,
|
||||||
|
Body: io.NopCloser(strings.NewReader(string(body))),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
return resp, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if error might be TLS-related (Cloudflare blocking)
|
||||||
|
errStr := strings.ToLower(err.Error())
|
||||||
|
tlsRelated := strings.Contains(errStr, "tls") ||
|
||||||
|
strings.Contains(errStr, "handshake") ||
|
||||||
|
strings.Contains(errStr, "certificate") ||
|
||||||
|
strings.Contains(errStr, "connection reset")
|
||||||
|
|
||||||
|
if tlsRelated {
|
||||||
|
LogDebug("HTTP", "TLS error detected, retrying with Chrome TLS fingerprint: %v", err)
|
||||||
|
|
||||||
|
// Clone request for retry
|
||||||
|
reqCopy := req.Clone(req.Context())
|
||||||
|
reqCopy.Header.Set("User-Agent", getRandomUserAgent())
|
||||||
|
|
||||||
|
// Retry with uTLS Chrome fingerprint
|
||||||
|
return cloudflareBypassClient.Do(reqCopy)
|
||||||
|
}
|
||||||
|
|
||||||
|
CheckAndLogISPBlocking(err, req.URL.String(), "HTTP")
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
@@ -0,0 +1,189 @@
|
|||||||
|
package gobackend
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// IDHSClient is a client for I Don't Have Spotify API
|
||||||
|
// Used as fallback when SongLink fails or is rate limited
|
||||||
|
type IDHSClient struct {
|
||||||
|
client *http.Client
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
globalIDHSClient *IDHSClient
|
||||||
|
idhsClientOnce sync.Once
|
||||||
|
idhsRateLimiter = NewRateLimiter(8, time.Minute) // 8 req/min (below 10 limit)
|
||||||
|
)
|
||||||
|
|
||||||
|
// IDHSSearchRequest represents the request body for IDHS API
|
||||||
|
type IDHSSearchRequest struct {
|
||||||
|
Link string `json:"link"`
|
||||||
|
Adapters []string `json:"adapters,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// IDHSSearchResponse represents the response from IDHS API
|
||||||
|
type IDHSSearchResponse struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Type string `json:"type"` // song, album, artist, podcast, show
|
||||||
|
Title string `json:"title"`
|
||||||
|
Description string `json:"description"`
|
||||||
|
Image string `json:"image,omitempty"`
|
||||||
|
Audio string `json:"audio,omitempty"`
|
||||||
|
Source string `json:"source"`
|
||||||
|
UniversalLink string `json:"universalLink"`
|
||||||
|
Links []IDHSLink `json:"links"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// IDHSLink represents a link to a streaming platform
|
||||||
|
type IDHSLink struct {
|
||||||
|
Type string `json:"type"` // spotify, youTube, appleMusic, deezer, soundCloud, tidal
|
||||||
|
URL string `json:"url"`
|
||||||
|
IsVerified bool `json:"isVerified,omitempty"`
|
||||||
|
NotAvailable bool `json:"notAvailable,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewIDHSClient creates a new IDHS client
|
||||||
|
func NewIDHSClient() *IDHSClient {
|
||||||
|
idhsClientOnce.Do(func() {
|
||||||
|
globalIDHSClient = &IDHSClient{
|
||||||
|
client: NewHTTPClientWithTimeout(15 * time.Second),
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return globalIDHSClient
|
||||||
|
}
|
||||||
|
|
||||||
|
// Search converts a music link to links on other platforms
|
||||||
|
func (c *IDHSClient) Search(link string, adapters []string) (*IDHSSearchResponse, error) {
|
||||||
|
idhsRateLimiter.WaitForSlot()
|
||||||
|
|
||||||
|
reqBody := IDHSSearchRequest{
|
||||||
|
Link: link,
|
||||||
|
Adapters: adapters,
|
||||||
|
}
|
||||||
|
|
||||||
|
jsonBody, err := json.Marshal(reqBody)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to marshal request: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
req, err := http.NewRequest("POST", "https://idonthavespotify.sjdonado.com/api/search?v=1", bytes.NewBuffer(jsonBody))
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to create request: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
req.Header.Set("Content-Type", "application/json")
|
||||||
|
req.Header.Set("User-Agent", getRandomUserAgent())
|
||||||
|
|
||||||
|
resp, err := c.client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("request failed: %w", err)
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode == 400 {
|
||||||
|
return nil, fmt.Errorf("invalid link or missing parameters")
|
||||||
|
}
|
||||||
|
if resp.StatusCode == 429 {
|
||||||
|
return nil, fmt.Errorf("IDHS rate limit exceeded")
|
||||||
|
}
|
||||||
|
if resp.StatusCode == 500 {
|
||||||
|
return nil, fmt.Errorf("IDHS processing failed")
|
||||||
|
}
|
||||||
|
if resp.StatusCode != 200 {
|
||||||
|
return nil, fmt.Errorf("IDHS API returned status %d", resp.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := ReadResponseBody(resp)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to read response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
var result IDHSSearchResponse
|
||||||
|
if err := json.Unmarshal(body, &result); err != nil {
|
||||||
|
return nil, fmt.Errorf("failed to decode response: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return &result, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAvailabilityFromSpotify checks track availability using IDHS as fallback
|
||||||
|
func (c *IDHSClient) GetAvailabilityFromSpotify(spotifyTrackID string) (*TrackAvailability, error) {
|
||||||
|
spotifyURL := fmt.Sprintf("https://open.spotify.com/track/%s", spotifyTrackID)
|
||||||
|
|
||||||
|
// Request only the platforms we need
|
||||||
|
adapters := []string{"tidal", "deezer"}
|
||||||
|
|
||||||
|
result, err := c.Search(spotifyURL, adapters)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
availability := &TrackAvailability{
|
||||||
|
SpotifyID: spotifyTrackID,
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, link := range result.Links {
|
||||||
|
if link.NotAvailable {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
switch strings.ToLower(link.Type) {
|
||||||
|
case "tidal":
|
||||||
|
availability.Tidal = true
|
||||||
|
availability.TidalURL = link.URL
|
||||||
|
case "deezer":
|
||||||
|
availability.Deezer = true
|
||||||
|
availability.DeezerURL = link.URL
|
||||||
|
availability.DeezerID = extractDeezerIDFromURL(link.URL)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
LogDebug("IDHS", "Availability from Spotify %s: Tidal=%v, Deezer=%v",
|
||||||
|
spotifyTrackID, availability.Tidal, availability.Deezer)
|
||||||
|
|
||||||
|
return availability, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetAvailabilityFromDeezer checks track availability using IDHS
|
||||||
|
func (c *IDHSClient) GetAvailabilityFromDeezer(deezerTrackID string) (*TrackAvailability, error) {
|
||||||
|
deezerURL := fmt.Sprintf("https://www.deezer.com/track/%s", deezerTrackID)
|
||||||
|
|
||||||
|
// Request only the platforms we need
|
||||||
|
adapters := []string{"spotify", "tidal"}
|
||||||
|
|
||||||
|
result, err := c.Search(deezerURL, adapters)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
availability := &TrackAvailability{
|
||||||
|
Deezer: true,
|
||||||
|
DeezerID: deezerTrackID,
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, link := range result.Links {
|
||||||
|
if link.NotAvailable {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
switch strings.ToLower(link.Type) {
|
||||||
|
case "spotify":
|
||||||
|
availability.SpotifyID = extractSpotifyIDFromURL(link.URL)
|
||||||
|
case "tidal":
|
||||||
|
availability.Tidal = true
|
||||||
|
availability.TidalURL = link.URL
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
LogDebug("IDHS", "Availability from Deezer %s: Spotify=%s, Tidal=%v",
|
||||||
|
deezerTrackID, availability.SpotifyID, availability.Tidal)
|
||||||
|
|
||||||
|
return availability, nil
|
||||||
|
}
|
||||||
@@ -150,11 +150,11 @@ func GoLog(format string, args ...interface{}) {
|
|||||||
|
|
||||||
// Determine level from message content
|
// Determine level from message content
|
||||||
msgLower := strings.ToLower(message)
|
msgLower := strings.ToLower(message)
|
||||||
if strings.Contains(msgLower, "error") || strings.Contains(msgLower, "failed") || strings.HasPrefix(message, "✗") {
|
if strings.Contains(msgLower, "error") || strings.Contains(msgLower, "failed") {
|
||||||
level = "ERROR"
|
level = "ERROR"
|
||||||
} else if strings.Contains(msgLower, "warning") || strings.Contains(msgLower, "warn") {
|
} else if strings.Contains(msgLower, "warning") || strings.Contains(msgLower, "warn") {
|
||||||
level = "WARN"
|
level = "WARN"
|
||||||
} else if strings.HasPrefix(message, "✓") || strings.Contains(msgLower, "success") || strings.Contains(msgLower, "match found") {
|
} else if strings.Contains(msgLower, "success") || strings.Contains(msgLower, "match found") {
|
||||||
level = "INFO"
|
level = "INFO"
|
||||||
} else if strings.Contains(msgLower, "searching") || strings.Contains(msgLower, "trying") || strings.Contains(msgLower, "found") {
|
} else if strings.Contains(msgLower, "searching") || strings.Contains(msgLower, "trying") || strings.Contains(msgLower, "found") {
|
||||||
level = "DEBUG"
|
level = "DEBUG"
|
||||||
|
|||||||
@@ -682,21 +682,6 @@ func EmbedM4AMetadata(filePath string, metadata Metadata, coverData []byte) erro
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func findAtom(data []byte, name string, offset int) int {
|
|
||||||
for i := offset; i < len(data)-8; {
|
|
||||||
size := int(uint32(data[i])<<24 | uint32(data[i+1])<<16 | uint32(data[i+2])<<8 | uint32(data[i+3]))
|
|
||||||
if size < 8 {
|
|
||||||
break
|
|
||||||
}
|
|
||||||
atomName := string(data[i+4 : i+8])
|
|
||||||
if atomName == name {
|
|
||||||
return i
|
|
||||||
}
|
|
||||||
i += size
|
|
||||||
}
|
|
||||||
return -1
|
|
||||||
}
|
|
||||||
|
|
||||||
// buildMetaAtom builds a complete meta atom with ilst containing metadata
|
// buildMetaAtom builds a complete meta atom with ilst containing metadata
|
||||||
func buildMetaAtom(metadata Metadata, coverData []byte) []byte {
|
func buildMetaAtom(metadata Metadata, coverData []byte) []byte {
|
||||||
var ilst []byte
|
var ilst []byte
|
||||||
|
|||||||
@@ -0,0 +1,10 @@
|
|||||||
|
// mobile_deps.go
|
||||||
|
// This file ensures gomobile dependencies are not removed by go mod tidy.
|
||||||
|
// These packages are required by gomobile bind but not directly imported in code.
|
||||||
|
|
||||||
|
package gobackend
|
||||||
|
|
||||||
|
import (
|
||||||
|
// Required for gomobile bind to work
|
||||||
|
_ "golang.org/x/mobile/bind"
|
||||||
|
)
|
||||||
+52
-9
@@ -17,6 +17,9 @@ type TrackIDCache struct {
|
|||||||
cache map[string]*TrackIDCacheEntry
|
cache map[string]*TrackIDCacheEntry
|
||||||
mu sync.RWMutex
|
mu sync.RWMutex
|
||||||
ttl time.Duration
|
ttl time.Duration
|
||||||
|
// Cleanup is triggered on writes at a fixed interval to avoid unbounded growth.
|
||||||
|
lastCleanup time.Time
|
||||||
|
cleanupInterval time.Duration
|
||||||
}
|
}
|
||||||
|
|
||||||
var (
|
var (
|
||||||
@@ -27,8 +30,9 @@ var (
|
|||||||
func GetTrackIDCache() *TrackIDCache {
|
func GetTrackIDCache() *TrackIDCache {
|
||||||
trackIDCacheOnce.Do(func() {
|
trackIDCacheOnce.Do(func() {
|
||||||
globalTrackIDCache = &TrackIDCache{
|
globalTrackIDCache = &TrackIDCache{
|
||||||
cache: make(map[string]*TrackIDCacheEntry),
|
cache: make(map[string]*TrackIDCacheEntry),
|
||||||
ttl: 30 * time.Minute,
|
ttl: 30 * time.Minute,
|
||||||
|
cleanupInterval: 5 * time.Minute,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
return globalTrackIDCache
|
return globalTrackIDCache
|
||||||
@@ -36,13 +40,34 @@ func GetTrackIDCache() *TrackIDCache {
|
|||||||
|
|
||||||
func (c *TrackIDCache) Get(isrc string) *TrackIDCacheEntry {
|
func (c *TrackIDCache) Get(isrc string) *TrackIDCacheEntry {
|
||||||
c.mu.RLock()
|
c.mu.RLock()
|
||||||
defer c.mu.RUnlock()
|
|
||||||
|
|
||||||
entry, exists := c.cache[isrc]
|
entry, exists := c.cache[isrc]
|
||||||
if !exists || time.Now().After(entry.ExpiresAt) {
|
if !exists {
|
||||||
|
c.mu.RUnlock()
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
return entry
|
expired := time.Now().After(entry.ExpiresAt)
|
||||||
|
c.mu.RUnlock()
|
||||||
|
|
||||||
|
if !expired {
|
||||||
|
return entry
|
||||||
|
}
|
||||||
|
|
||||||
|
// Lazily delete expired entry.
|
||||||
|
c.mu.Lock()
|
||||||
|
entry, exists = c.cache[isrc]
|
||||||
|
if exists && time.Now().After(entry.ExpiresAt) {
|
||||||
|
delete(c.cache, isrc)
|
||||||
|
}
|
||||||
|
c.mu.Unlock()
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *TrackIDCache) pruneExpiredLocked(now time.Time) {
|
||||||
|
for key, entry := range c.cache {
|
||||||
|
if now.After(entry.ExpiresAt) {
|
||||||
|
delete(c.cache, key)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *TrackIDCache) SetTidal(isrc string, trackID int64) {
|
func (c *TrackIDCache) SetTidal(isrc string, trackID int64) {
|
||||||
@@ -55,7 +80,13 @@ func (c *TrackIDCache) SetTidal(isrc string, trackID int64) {
|
|||||||
c.cache[isrc] = entry
|
c.cache[isrc] = entry
|
||||||
}
|
}
|
||||||
entry.TidalTrackID = trackID
|
entry.TidalTrackID = trackID
|
||||||
entry.ExpiresAt = time.Now().Add(c.ttl)
|
now := time.Now()
|
||||||
|
entry.ExpiresAt = now.Add(c.ttl)
|
||||||
|
|
||||||
|
if c.cleanupInterval > 0 && (c.lastCleanup.IsZero() || now.Sub(c.lastCleanup) >= c.cleanupInterval) {
|
||||||
|
c.pruneExpiredLocked(now)
|
||||||
|
c.lastCleanup = now
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *TrackIDCache) SetQobuz(isrc string, trackID int64) {
|
func (c *TrackIDCache) SetQobuz(isrc string, trackID int64) {
|
||||||
@@ -68,7 +99,13 @@ func (c *TrackIDCache) SetQobuz(isrc string, trackID int64) {
|
|||||||
c.cache[isrc] = entry
|
c.cache[isrc] = entry
|
||||||
}
|
}
|
||||||
entry.QobuzTrackID = trackID
|
entry.QobuzTrackID = trackID
|
||||||
entry.ExpiresAt = time.Now().Add(c.ttl)
|
now := time.Now()
|
||||||
|
entry.ExpiresAt = now.Add(c.ttl)
|
||||||
|
|
||||||
|
if c.cleanupInterval > 0 && (c.lastCleanup.IsZero() || now.Sub(c.lastCleanup) >= c.cleanupInterval) {
|
||||||
|
c.pruneExpiredLocked(now)
|
||||||
|
c.lastCleanup = now
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *TrackIDCache) SetAmazon(isrc string, trackID string) {
|
func (c *TrackIDCache) SetAmazon(isrc string, trackID string) {
|
||||||
@@ -81,7 +118,13 @@ func (c *TrackIDCache) SetAmazon(isrc string, trackID string) {
|
|||||||
c.cache[isrc] = entry
|
c.cache[isrc] = entry
|
||||||
}
|
}
|
||||||
entry.AmazonTrackID = trackID
|
entry.AmazonTrackID = trackID
|
||||||
entry.ExpiresAt = time.Now().Add(c.ttl)
|
now := time.Now()
|
||||||
|
entry.ExpiresAt = now.Add(c.ttl)
|
||||||
|
|
||||||
|
if c.cleanupInterval > 0 && (c.lastCleanup.IsZero() || now.Sub(c.lastCleanup) >= c.cleanupInterval) {
|
||||||
|
c.pruneExpiredLocked(now)
|
||||||
|
c.lastCleanup = now
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (c *TrackIDCache) Clear() {
|
func (c *TrackIDCache) Clear() {
|
||||||
|
|||||||
+125
-11
@@ -375,10 +375,11 @@ func (q *QobuzDownloader) GetTrackByID(trackID int64) (*QobuzTrack, error) {
|
|||||||
// Uses same APIs as PC version for compatibility
|
// Uses same APIs as PC version for compatibility
|
||||||
func (q *QobuzDownloader) GetAvailableAPIs() []string {
|
func (q *QobuzDownloader) GetAvailableAPIs() []string {
|
||||||
// Same APIs as PC version (referensi/backend/qobuz.go)
|
// Same APIs as PC version (referensi/backend/qobuz.go)
|
||||||
// Primary: dab.yeet.su, Fallback: dabmusic.xyz
|
// Primary: dab.yeet.su, Fallback: dabmusic.xyz, qobuz.squid.wtf
|
||||||
encodedAPIs := []string{
|
encodedAPIs := []string{
|
||||||
"ZGFiLnllZXQuc3UvYXBpL3N0cmVhbT90cmFja0lkPQ==", // dab.yeet.su/api/stream?trackId= (PRIMARY - same as PC)
|
"ZGFiLnllZXQuc3UvYXBpL3N0cmVhbT90cmFja0lkPQ==", // dab.yeet.su/api/stream?trackId=
|
||||||
"ZGFibXVzaWMueHl6L2FwaS9zdHJlYW0/dHJhY2tJZD0=", // dabmusic.xyz/api/stream?trackId= (FALLBACK - same as PC)
|
"ZGFibXVzaWMueHl6L2FwaS9zdHJlYW0/dHJhY2tJZD0=", // dabmusic.xyz/api/stream?trackId=
|
||||||
|
"cW9idXouc3F1aWQud3RmL2FwaS9kb3dubG9hZC1tdXNpYz90cmFja19pZD0=", // qobuz.squid.wtf/api/download-music?track_id=
|
||||||
}
|
}
|
||||||
|
|
||||||
var apis []string
|
var apis []string
|
||||||
@@ -393,6 +394,95 @@ func (q *QobuzDownloader) GetAvailableAPIs() []string {
|
|||||||
return apis
|
return apis
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// mapJumoQuality maps Qobuz quality codes to Jumo format
|
||||||
|
func mapJumoQuality(quality string) int {
|
||||||
|
switch quality {
|
||||||
|
case "6":
|
||||||
|
return 6 // 16-bit FLAC
|
||||||
|
case "7":
|
||||||
|
return 7 // 24-bit 96kHz
|
||||||
|
case "27":
|
||||||
|
return 27 // 24-bit 192kHz
|
||||||
|
default:
|
||||||
|
return 6
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// decodeXOR decodes XOR-encoded response from Jumo API
|
||||||
|
func decodeXOR(data []byte) string {
|
||||||
|
text := string(data)
|
||||||
|
runes := []rune(text)
|
||||||
|
result := make([]rune, len(runes))
|
||||||
|
for i, char := range runes {
|
||||||
|
key := rune((i * 17) % 128)
|
||||||
|
result[i] = char ^ 253 ^ key
|
||||||
|
}
|
||||||
|
return string(result)
|
||||||
|
}
|
||||||
|
|
||||||
|
// downloadFromJumo gets download URL from Jumo API (fallback)
|
||||||
|
func (q *QobuzDownloader) downloadFromJumo(trackID int64, quality string) (string, error) {
|
||||||
|
formatID := mapJumoQuality(quality)
|
||||||
|
region := "US"
|
||||||
|
|
||||||
|
// Jumo API endpoint
|
||||||
|
jumoURL := fmt.Sprintf("https://jumo-dl.pages.dev/file?track_id=%d&format_id=%d®ion=%s", trackID, formatID, region)
|
||||||
|
|
||||||
|
GoLog("[Qobuz] Trying Jumo API fallback...\n")
|
||||||
|
|
||||||
|
client := NewHTTPClientWithTimeout(30 * time.Second)
|
||||||
|
req, err := http.NewRequest("GET", jumoURL, nil)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
resp, err := client.Do(req)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
defer resp.Body.Close()
|
||||||
|
|
||||||
|
if resp.StatusCode != 200 {
|
||||||
|
return "", fmt.Errorf("Jumo API returned HTTP %d", resp.StatusCode)
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := io.ReadAll(resp.Body)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
var result map[string]any
|
||||||
|
|
||||||
|
// Try parsing as plain JSON first
|
||||||
|
if err := json.Unmarshal(body, &result); err != nil {
|
||||||
|
// Try XOR decoding
|
||||||
|
decoded := decodeXOR(body)
|
||||||
|
if err := json.Unmarshal([]byte(decoded), &result); err != nil {
|
||||||
|
return "", fmt.Errorf("failed to parse Jumo response (plain or XOR): %w", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check for URL in various response formats
|
||||||
|
if urlVal, ok := result["url"].(string); ok && urlVal != "" {
|
||||||
|
GoLog("[Qobuz] Jumo API returned URL successfully\n")
|
||||||
|
return urlVal, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if data, ok := result["data"].(map[string]any); ok {
|
||||||
|
if urlVal, ok := data["url"].(string); ok && urlVal != "" {
|
||||||
|
GoLog("[Qobuz] Jumo API returned URL successfully (from data)\n")
|
||||||
|
return urlVal, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if linkVal, ok := result["link"].(string); ok && linkVal != "" {
|
||||||
|
GoLog("[Qobuz] Jumo API returned URL successfully (from link)\n")
|
||||||
|
return linkVal, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return "", fmt.Errorf("URL not found in Jumo response")
|
||||||
|
}
|
||||||
|
|
||||||
func (q *QobuzDownloader) SearchTrackByISRC(isrc string) (*QobuzTrack, error) {
|
func (q *QobuzDownloader) SearchTrackByISRC(isrc string) (*QobuzTrack, error) {
|
||||||
apiBase, _ := base64.StdEncoding.DecodeString("aHR0cHM6Ly93d3cucW9idXouY29tL2FwaS5qc29uLzAuMi90cmFjay9zZWFyY2g/cXVlcnk9")
|
apiBase, _ := base64.StdEncoding.DecodeString("aHR0cHM6Ly93d3cucW9idXouY29tL2FwaS5qc29uLzAuMi90cmFjay9zZWFyY2g/cXVlcnk9")
|
||||||
searchURL := fmt.Sprintf("%s%s&limit=50&app_id=%s", string(apiBase), url.QueryEscape(isrc), q.appID)
|
searchURL := fmt.Sprintf("%s%s&limit=50&app_id=%s", string(apiBase), url.QueryEscape(isrc), q.appID)
|
||||||
@@ -662,12 +752,12 @@ func (q *QobuzDownloader) SearchTrackByMetadataWithDuration(trackName, artistNam
|
|||||||
if len(durationMatches) > 0 {
|
if len(durationMatches) > 0 {
|
||||||
for _, track := range durationMatches {
|
for _, track := range durationMatches {
|
||||||
if track.MaximumBitDepth >= 24 {
|
if track.MaximumBitDepth >= 24 {
|
||||||
GoLog("[Qobuz] ✓ Match found: '%s' by '%s' (title+duration verified, hi-res)\n",
|
GoLog("[Qobuz] Match found: '%s' by '%s' (title+duration verified, hi-res)\n",
|
||||||
track.Title, track.Performer.Name)
|
track.Title, track.Performer.Name)
|
||||||
return track, nil
|
return track, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
GoLog("[Qobuz] ✓ Match found: '%s' by '%s' (title+duration verified)\n",
|
GoLog("[Qobuz] Match found: '%s' by '%s' (title+duration verified)\n",
|
||||||
durationMatches[0].Title, durationMatches[0].Performer.Name)
|
durationMatches[0].Title, durationMatches[0].Performer.Name)
|
||||||
return durationMatches[0], nil
|
return durationMatches[0], nil
|
||||||
}
|
}
|
||||||
@@ -678,14 +768,14 @@ func (q *QobuzDownloader) SearchTrackByMetadataWithDuration(trackName, artistNam
|
|||||||
// No duration verification, return best quality from title matches
|
// No duration verification, return best quality from title matches
|
||||||
for _, track := range tracksToCheck {
|
for _, track := range tracksToCheck {
|
||||||
if track.MaximumBitDepth >= 24 {
|
if track.MaximumBitDepth >= 24 {
|
||||||
GoLog("[Qobuz] ✓ Match found: '%s' by '%s' (title verified, hi-res)\n",
|
GoLog("[Qobuz] Match found: '%s' by '%s' (title verified, hi-res)\n",
|
||||||
track.Title, track.Performer.Name)
|
track.Title, track.Performer.Name)
|
||||||
return track, nil
|
return track, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(tracksToCheck) > 0 {
|
if len(tracksToCheck) > 0 {
|
||||||
GoLog("[Qobuz] ✓ Match found: '%s' by '%s' (title verified)\n",
|
GoLog("[Qobuz] Match found: '%s' by '%s' (title verified)\n",
|
||||||
tracksToCheck[0].Title, tracksToCheck[0].Performer.Name)
|
tracksToCheck[0].Title, tracksToCheck[0].Performer.Name)
|
||||||
return tracksToCheck[0], nil
|
return tracksToCheck[0], nil
|
||||||
}
|
}
|
||||||
@@ -782,7 +872,7 @@ func getQobuzDownloadURLParallel(apis []string, trackID int64, quality string) (
|
|||||||
for i := 0; i < len(apis); i++ {
|
for i := 0; i < len(apis); i++ {
|
||||||
result := <-resultChan
|
result := <-resultChan
|
||||||
if result.err == nil {
|
if result.err == nil {
|
||||||
GoLog("[Qobuz] [Parallel] ✓ Got response from %s in %v\n", result.apiURL, result.duration)
|
GoLog("[Qobuz] [Parallel] Got response from %s in %v\n", result.apiURL, result.duration)
|
||||||
|
|
||||||
// Drain remaining results to avoid goroutine leaks
|
// Drain remaining results to avoid goroutine leaks
|
||||||
go func(remaining int) {
|
go func(remaining int) {
|
||||||
@@ -812,11 +902,35 @@ func (q *QobuzDownloader) GetDownloadURL(trackID int64, quality string) (string,
|
|||||||
}
|
}
|
||||||
|
|
||||||
_, downloadURL, err := getQobuzDownloadURLParallel(apis, trackID, quality)
|
_, downloadURL, err := getQobuzDownloadURLParallel(apis, trackID, quality)
|
||||||
if err != nil {
|
if err == nil {
|
||||||
return "", err
|
return downloadURL, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return downloadURL, nil
|
// All standard APIs failed, try Jumo as fallback
|
||||||
|
GoLog("[Qobuz] Standard APIs failed, trying Jumo fallback...\n")
|
||||||
|
jumoURL, jumoErr := q.downloadFromJumo(trackID, quality)
|
||||||
|
if jumoErr == nil {
|
||||||
|
return jumoURL, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// If quality is 27 (hi-res), try fallback to lower quality
|
||||||
|
if quality == "27" {
|
||||||
|
GoLog("[Qobuz] Hi-res (27) failed, trying 24-bit (7)...\n")
|
||||||
|
jumoURL, jumoErr = q.downloadFromJumo(trackID, "7")
|
||||||
|
if jumoErr == nil {
|
||||||
|
return jumoURL, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if quality == "27" || quality == "7" {
|
||||||
|
GoLog("[Qobuz] 24-bit failed, trying 16-bit (6)...\n")
|
||||||
|
jumoURL, jumoErr = q.downloadFromJumo(trackID, "6")
|
||||||
|
if jumoErr == nil {
|
||||||
|
return jumoURL, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return "", fmt.Errorf("all Qobuz APIs and Jumo fallback failed: %w", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
// DownloadFile downloads a file from URL with User-Agent and progress tracking
|
// DownloadFile downloads a file from URL with User-Agent and progress tracking
|
||||||
|
|||||||
+54
-17
@@ -46,7 +46,30 @@ func (s *SongLinkClient) CheckTrackAvailability(spotifyTrackID string, isrc stri
|
|||||||
if spotifyTrackID == "" {
|
if spotifyTrackID == "" {
|
||||||
return nil, fmt.Errorf("spotify track ID is empty")
|
return nil, fmt.Errorf("spotify track ID is empty")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Try SongLink first
|
||||||
|
availability, err := s.checkTrackAvailabilitySongLink(spotifyTrackID)
|
||||||
|
if err != nil {
|
||||||
|
// Fallback to IDHS if SongLink fails
|
||||||
|
LogWarn("SongLink", "SongLink failed, trying IDHS fallback: %v", err)
|
||||||
|
idhsClient := NewIDHSClient()
|
||||||
|
availability, err = idhsClient.GetAvailabilityFromSpotify(spotifyTrackID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("both SongLink and IDHS failed: %w", err)
|
||||||
|
}
|
||||||
|
LogInfo("SongLink", "IDHS fallback successful for %s", spotifyTrackID)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check Qobuz availability separately via ISRC
|
||||||
|
if isrc != "" {
|
||||||
|
availability.Qobuz = checkQobuzAvailability(isrc)
|
||||||
|
}
|
||||||
|
|
||||||
|
return availability, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// checkTrackAvailabilitySongLink is the original SongLink implementation
|
||||||
|
func (s *SongLinkClient) checkTrackAvailabilitySongLink(spotifyTrackID string) (*TrackAvailability, error) {
|
||||||
songLinkRateLimiter.WaitForSlot()
|
songLinkRateLimiter.WaitForSlot()
|
||||||
|
|
||||||
spotifyBase, _ := base64.StdEncoding.DecodeString("aHR0cHM6Ly9vcGVuLnNwb3RpZnkuY29tL3RyYWNrLw==")
|
spotifyBase, _ := base64.StdEncoding.DecodeString("aHR0cHM6Ly9vcGVuLnNwb3RpZnkuY29tL3RyYWNrLw==")
|
||||||
@@ -115,10 +138,6 @@ func (s *SongLinkClient) CheckTrackAvailability(spotifyTrackID string, isrc stri
|
|||||||
availability.DeezerID = extractDeezerIDFromURL(deezerLink.URL)
|
availability.DeezerID = extractDeezerIDFromURL(deezerLink.URL)
|
||||||
}
|
}
|
||||||
|
|
||||||
if isrc != "" {
|
|
||||||
availability.Qobuz = checkQobuzAvailability(isrc)
|
|
||||||
}
|
|
||||||
|
|
||||||
return availability, nil
|
return availability, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -191,11 +210,11 @@ func (s *SongLinkClient) GetDeezerIDFromSpotify(spotifyTrackID string) (string,
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
if !availability.Deezer || availability.DeezerID == "" {
|
if !availability.Deezer || availability.DeezerID == "" {
|
||||||
return "", fmt.Errorf("track not found on Deezer")
|
return "", fmt.Errorf("track not found on Deezer")
|
||||||
}
|
}
|
||||||
|
|
||||||
return availability.DeezerID, nil
|
return availability.DeezerID, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -268,11 +287,11 @@ func (s *SongLinkClient) GetDeezerAlbumIDFromSpotify(spotifyAlbumID string) (str
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
if !availability.Deezer || availability.DeezerID == "" {
|
if !availability.Deezer || availability.DeezerID == "" {
|
||||||
return "", fmt.Errorf("album not found on Deezer")
|
return "", fmt.Errorf("album not found on Deezer")
|
||||||
}
|
}
|
||||||
|
|
||||||
return availability.DeezerID, nil
|
return availability.DeezerID, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -281,7 +300,25 @@ func (s *SongLinkClient) CheckAvailabilityFromDeezer(deezerTrackID string) (*Tra
|
|||||||
if deezerTrackID == "" {
|
if deezerTrackID == "" {
|
||||||
return nil, fmt.Errorf("deezer track ID is empty")
|
return nil, fmt.Errorf("deezer track ID is empty")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Try SongLink first
|
||||||
|
availability, err := s.checkAvailabilityFromDeezerSongLink(deezerTrackID)
|
||||||
|
if err != nil {
|
||||||
|
// Fallback to IDHS if SongLink fails
|
||||||
|
LogWarn("SongLink", "SongLink failed for Deezer, trying IDHS fallback: %v", err)
|
||||||
|
idhsClient := NewIDHSClient()
|
||||||
|
availability, err = idhsClient.GetAvailabilityFromDeezer(deezerTrackID)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("both SongLink and IDHS failed: %w", err)
|
||||||
|
}
|
||||||
|
LogInfo("SongLink", "IDHS fallback successful for Deezer %s", deezerTrackID)
|
||||||
|
}
|
||||||
|
|
||||||
|
return availability, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// checkAvailabilityFromDeezerSongLink is the original SongLink implementation for Deezer
|
||||||
|
func (s *SongLinkClient) checkAvailabilityFromDeezerSongLink(deezerTrackID string) (*TrackAvailability, error) {
|
||||||
songLinkRateLimiter.WaitForSlot()
|
songLinkRateLimiter.WaitForSlot()
|
||||||
|
|
||||||
deezerURL := fmt.Sprintf("https://www.deezer.com/track/%s", deezerTrackID)
|
deezerURL := fmt.Sprintf("https://www.deezer.com/track/%s", deezerTrackID)
|
||||||
@@ -369,7 +406,7 @@ func (s *SongLinkClient) CheckAvailabilityByPlatform(platform, entityType, entit
|
|||||||
if entityID == "" {
|
if entityID == "" {
|
||||||
return nil, fmt.Errorf("%s ID is empty", platform)
|
return nil, fmt.Errorf("%s ID is empty", platform)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Use global rate limiter
|
// Use global rate limiter
|
||||||
songLinkRateLimiter.WaitForSlot()
|
songLinkRateLimiter.WaitForSlot()
|
||||||
|
|
||||||
@@ -464,11 +501,11 @@ func (s *SongLinkClient) GetSpotifyIDFromDeezer(deezerTrackID string) (string, e
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
if availability.SpotifyID == "" {
|
if availability.SpotifyID == "" {
|
||||||
return "", fmt.Errorf("track not found on Spotify")
|
return "", fmt.Errorf("track not found on Spotify")
|
||||||
}
|
}
|
||||||
|
|
||||||
return availability.SpotifyID, nil
|
return availability.SpotifyID, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -478,11 +515,11 @@ func (s *SongLinkClient) GetTidalURLFromDeezer(deezerTrackID string) (string, er
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
if !availability.Tidal || availability.TidalURL == "" {
|
if !availability.Tidal || availability.TidalURL == "" {
|
||||||
return "", fmt.Errorf("track not found on Tidal")
|
return "", fmt.Errorf("track not found on Tidal")
|
||||||
}
|
}
|
||||||
|
|
||||||
return availability.TidalURL, nil
|
return availability.TidalURL, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -491,10 +528,10 @@ func (s *SongLinkClient) GetAmazonURLFromDeezer(deezerTrackID string) (string, e
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return "", err
|
return "", err
|
||||||
}
|
}
|
||||||
|
|
||||||
if !availability.Amazon || availability.AmazonURL == "" {
|
if !availability.Amazon || availability.AmazonURL == "" {
|
||||||
return "", fmt.Errorf("track not found on Amazon Music")
|
return "", fmt.Errorf("track not found on Amazon Music")
|
||||||
}
|
}
|
||||||
|
|
||||||
return availability.AmazonURL, nil
|
return availability.AmazonURL, nil
|
||||||
}
|
}
|
||||||
|
|||||||
+22
-2
@@ -238,9 +238,29 @@ type SearchArtistResult struct {
|
|||||||
Popularity int `json:"popularity"`
|
Popularity int `json:"popularity"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type SearchAlbumResult struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Artists string `json:"artists"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
ReleaseDate string `json:"release_date"`
|
||||||
|
TotalTracks int `json:"total_tracks"`
|
||||||
|
AlbumType string `json:"album_type"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type SearchPlaylistResult struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Owner string `json:"owner"`
|
||||||
|
Images string `json:"images"`
|
||||||
|
TotalTracks int `json:"total_tracks"`
|
||||||
|
}
|
||||||
|
|
||||||
type SearchAllResult struct {
|
type SearchAllResult struct {
|
||||||
Tracks []TrackMetadata `json:"tracks"`
|
Tracks []TrackMetadata `json:"tracks"`
|
||||||
Artists []SearchArtistResult `json:"artists"`
|
Artists []SearchArtistResult `json:"artists"`
|
||||||
|
Albums []SearchAlbumResult `json:"albums"`
|
||||||
|
Playlists []SearchPlaylistResult `json:"playlists"`
|
||||||
}
|
}
|
||||||
|
|
||||||
type spotifyURI struct {
|
type spotifyURI struct {
|
||||||
|
|||||||
+16
-14
@@ -122,14 +122,16 @@ func NewTidalDownloader() *TidalDownloader {
|
|||||||
// GetAvailableAPIs returns list of available Tidal APIs
|
// GetAvailableAPIs returns list of available Tidal APIs
|
||||||
func (t *TidalDownloader) GetAvailableAPIs() []string {
|
func (t *TidalDownloader) GetAvailableAPIs() []string {
|
||||||
encodedAPIs := []string{
|
encodedAPIs := []string{
|
||||||
"dGlkYWwua2lub3BsdXMub25saW5l",
|
"dGlkYWwua2lub3BsdXMub25saW5l", // tidal.kinoplus.online
|
||||||
"dGlkYWwtYXBpLmJpbmltdW0ub3Jn",
|
"dGlkYWwtYXBpLmJpbmltdW0ub3Jn", // tidal-api.binimum.org
|
||||||
"dHJpdG9uLnNxdWlkLnd0Zg==",
|
"dHJpdG9uLnNxdWlkLnd0Zg==", // triton.squid.wtf
|
||||||
"dm9nZWwucXFkbC5zaXRl",
|
"aGlmaS1vbmUuc3BvdGlzYXZlci5uZXQ=", // hifi-one.spotisaver.net
|
||||||
"bWF1cy5xcWRsLnNpdGU=",
|
"aGlmaS10d28uc3BvdGlzYXZlci5uZXQ=", // hifi-two.spotisaver.net
|
||||||
"aHVuZC5xcWRsLnNpdGU=",
|
"dm9nZWwucXFkbC5zaXRl", // vogel.qqdl.site
|
||||||
"a2F0emUucXFkbC5zaXRl",
|
"bWF1cy5xcWRsLnNpdGU=", // maus.qqdl.site
|
||||||
"d29sZi5xcWRsLnNpdGU=",
|
"aHVuZC5xcWRsLnNpdGU=", // hund.qqdl.site
|
||||||
|
"a2F0emUucXFkbC5zaXRl", // katze.qqdl.site
|
||||||
|
"d29sZi5xcWRsLnNpdGU=", // wolf.qqdl.site
|
||||||
}
|
}
|
||||||
|
|
||||||
var apis []string
|
var apis []string
|
||||||
@@ -442,13 +444,13 @@ func (t *TidalDownloader) SearchTrackByMetadataWithISRC(trackName, artistName, s
|
|||||||
durationDiff = -durationDiff
|
durationDiff = -durationDiff
|
||||||
}
|
}
|
||||||
if durationDiff <= 3 {
|
if durationDiff <= 3 {
|
||||||
GoLog("[Tidal] ✓ ISRC match: '%s' (duration verified)\n", track.Title)
|
GoLog("[Tidal] ISRC match: '%s' (duration verified)\n", track.Title)
|
||||||
return track, nil
|
return track, nil
|
||||||
}
|
}
|
||||||
GoLog("[Tidal] ISRC match but duration mismatch (expected %ds, got %ds), continuing...\n",
|
GoLog("[Tidal] ISRC match but duration mismatch (expected %ds, got %ds), continuing...\n",
|
||||||
expectedDuration, track.Duration)
|
expectedDuration, track.Duration)
|
||||||
} else {
|
} else {
|
||||||
GoLog("[Tidal] ✓ ISRC match: '%s'\n", track.Title)
|
GoLog("[Tidal] ISRC match: '%s'\n", track.Title)
|
||||||
return track, nil
|
return track, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -487,7 +489,7 @@ func (t *TidalDownloader) SearchTrackByMetadataWithISRC(trackName, artistName, s
|
|||||||
}
|
}
|
||||||
|
|
||||||
if len(durationVerifiedMatches) > 0 {
|
if len(durationVerifiedMatches) > 0 {
|
||||||
GoLog("[Tidal] ✓ ISRC match with duration verification: '%s' (expected %ds, found %ds)\n",
|
GoLog("[Tidal] ISRC match with duration verification: '%s' (expected %ds, found %ds)\n",
|
||||||
durationVerifiedMatches[0].Title, expectedDuration, durationVerifiedMatches[0].Duration)
|
durationVerifiedMatches[0].Title, expectedDuration, durationVerifiedMatches[0].Duration)
|
||||||
return durationVerifiedMatches[0], nil
|
return durationVerifiedMatches[0], nil
|
||||||
}
|
}
|
||||||
@@ -498,11 +500,11 @@ func (t *TidalDownloader) SearchTrackByMetadataWithISRC(trackName, artistName, s
|
|||||||
expectedDuration, isrcMatches[0].Duration)
|
expectedDuration, isrcMatches[0].Duration)
|
||||||
}
|
}
|
||||||
|
|
||||||
GoLog("[Tidal] ✓ ISRC match (no duration verification): '%s'\n", isrcMatches[0].Title)
|
GoLog("[Tidal] ISRC match (no duration verification): '%s'\n", isrcMatches[0].Title)
|
||||||
return isrcMatches[0], nil
|
return isrcMatches[0], nil
|
||||||
}
|
}
|
||||||
|
|
||||||
GoLog("[Tidal] ✗ No ISRC match found for: %s\n", spotifyISRC)
|
GoLog("[Tidal] No ISRC match found for: %s\n", spotifyISRC)
|
||||||
return nil, fmt.Errorf("ISRC mismatch: no track found with ISRC %s on Tidal", spotifyISRC)
|
return nil, fmt.Errorf("ISRC mismatch: no track found with ISRC %s on Tidal", spotifyISRC)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -669,7 +671,7 @@ func getDownloadURLParallel(apis []string, trackID int64, quality string) (strin
|
|||||||
for i := 0; i < len(apis); i++ {
|
for i := 0; i < len(apis); i++ {
|
||||||
result := <-resultChan
|
result := <-resultChan
|
||||||
if result.err == nil {
|
if result.err == nil {
|
||||||
GoLog("[Tidal] [Parallel] ✓ Got response from %s (%d-bit/%dHz) in %v\n",
|
GoLog("[Tidal] [Parallel] Got response from %s (%d-bit/%dHz) in %v\n",
|
||||||
result.apiURL, result.info.BitDepth, result.info.SampleRate, result.duration)
|
result.apiURL, result.info.BitDepth, result.info.SampleRate, result.duration)
|
||||||
|
|
||||||
go func(remaining int) {
|
go func(remaining int) {
|
||||||
|
|||||||
@@ -222,7 +222,8 @@ import Gobackend // Import Go framework
|
|||||||
let query = args["query"] as! String
|
let query = args["query"] as! String
|
||||||
let trackLimit = args["track_limit"] as? Int ?? 15
|
let trackLimit = args["track_limit"] as? Int ?? 15
|
||||||
let artistLimit = args["artist_limit"] as? Int ?? 3
|
let artistLimit = args["artist_limit"] as? Int ?? 3
|
||||||
let response = GobackendSearchDeezerAll(query, Int(trackLimit), Int(artistLimit), &error)
|
let filter = args["filter"] as? String ?? ""
|
||||||
|
let response = GobackendSearchDeezerAll(query, Int(trackLimit), Int(artistLimit), filter, &error)
|
||||||
if let error = error { throw error }
|
if let error = error { throw error }
|
||||||
return response
|
return response
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
/// App version and info constants
|
/// App version and info constants
|
||||||
/// Update version here only - all other files will reference this
|
/// Update version here only - all other files will reference this
|
||||||
class AppInfo {
|
class AppInfo {
|
||||||
static const String version = '3.2.1';
|
static const String version = '3.3.1';
|
||||||
static const String buildNumber = '64';
|
static const String buildNumber = '68';
|
||||||
static const String fullVersion = '$version+$buildNumber';
|
static const String fullVersion = '$version+$buildNumber';
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -952,6 +952,12 @@ abstract class AppLocalizations {
|
|||||||
/// **'The original HiFi project creator. The foundation of Tidal integration!'**
|
/// **'The original HiFi project creator. The foundation of Tidal integration!'**
|
||||||
String get aboutSachinsenalDesc;
|
String get aboutSachinsenalDesc;
|
||||||
|
|
||||||
|
/// Credit description for sjdonado
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!'**
|
||||||
|
String get aboutSjdonadoDesc;
|
||||||
|
|
||||||
/// Name of Amazon API service - DO NOT TRANSLATE
|
/// Name of Amazon API service - DO NOT TRANSLATE
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
@@ -3394,35 +3400,65 @@ abstract class AppLocalizations {
|
|||||||
/// **'24-bit / up to 192kHz'**
|
/// **'24-bit / up to 192kHz'**
|
||||||
String get qualityHiResFlacMaxSubtitle;
|
String get qualityHiResFlacMaxSubtitle;
|
||||||
|
|
||||||
/// Quality option - MP3 lossy format
|
/// Quality option - lossy format (MP3/Opus)
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
/// **'MP3'**
|
/// **'Lossy'**
|
||||||
String get qualityMp3;
|
String get qualityLossy;
|
||||||
|
|
||||||
/// Technical spec for MP3
|
/// Technical spec for lossy MP3
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
/// **'320kbps (converted from FLAC)'**
|
/// **'MP3 320kbps (converted from FLAC)'**
|
||||||
String get qualityMp3Subtitle;
|
String get qualityLossyMp3Subtitle;
|
||||||
|
|
||||||
/// Setting - enable MP3 quality option
|
/// Technical spec for lossy Opus
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
/// **'Enable MP3 Option'**
|
/// **'Opus 128kbps (converted from FLAC)'**
|
||||||
String get enableMp3Option;
|
String get qualityLossyOpusSubtitle;
|
||||||
|
|
||||||
/// Subtitle when MP3 is enabled
|
/// Setting - enable lossy quality option
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
/// **'MP3 quality option is available'**
|
/// **'Enable Lossy Option'**
|
||||||
String get enableMp3OptionSubtitleOn;
|
String get enableLossyOption;
|
||||||
|
|
||||||
/// Subtitle when MP3 is disabled
|
/// Subtitle when lossy is enabled
|
||||||
///
|
///
|
||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
/// **'Downloads FLAC then converts to 320kbps MP3'**
|
/// **'Lossy quality option is available'**
|
||||||
String get enableMp3OptionSubtitleOff;
|
String get enableLossyOptionSubtitleOn;
|
||||||
|
|
||||||
|
/// Subtitle when lossy is disabled
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Downloads FLAC then converts to lossy format'**
|
||||||
|
String get enableLossyOptionSubtitleOff;
|
||||||
|
|
||||||
|
/// Setting - choose lossy format
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Lossy Format'**
|
||||||
|
String get lossyFormat;
|
||||||
|
|
||||||
|
/// Description for lossy format picker
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Choose the lossy format for conversion'**
|
||||||
|
String get lossyFormatDescription;
|
||||||
|
|
||||||
|
/// MP3 format description
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'320kbps, best compatibility'**
|
||||||
|
String get lossyFormatMp3Subtitle;
|
||||||
|
|
||||||
|
/// Opus format description
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'128kbps, better quality at smaller size'**
|
||||||
|
String get lossyFormatOpusSubtitle;
|
||||||
|
|
||||||
/// Note about quality availability
|
/// Note about quality availability
|
||||||
///
|
///
|
||||||
@@ -3921,6 +3957,48 @@ abstract class AppLocalizations {
|
|||||||
/// In en, this message translates to:
|
/// In en, this message translates to:
|
||||||
/// **'Failed to fetch some albums'**
|
/// **'Failed to fetch some albums'**
|
||||||
String get discographyFailedToFetch;
|
String get discographyFailedToFetch;
|
||||||
|
|
||||||
|
/// Section header for storage access settings
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Storage Access'**
|
||||||
|
String get sectionStorageAccess;
|
||||||
|
|
||||||
|
/// Toggle for MANAGE_EXTERNAL_STORAGE permission
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'All Files Access'**
|
||||||
|
String get allFilesAccess;
|
||||||
|
|
||||||
|
/// Subtitle when all files access is enabled
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Can write to any folder'**
|
||||||
|
String get allFilesAccessEnabledSubtitle;
|
||||||
|
|
||||||
|
/// Subtitle when all files access is disabled
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Limited to media folders only'**
|
||||||
|
String get allFilesAccessDisabledSubtitle;
|
||||||
|
|
||||||
|
/// Description explaining when to enable all files access
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.'**
|
||||||
|
String get allFilesAccessDescription;
|
||||||
|
|
||||||
|
/// Message when permission is permanently denied
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'Permission was denied. Please enable \'All files access\' manually in system settings.'**
|
||||||
|
String get allFilesAccessDeniedMessage;
|
||||||
|
|
||||||
|
/// Snackbar message when user disables all files access
|
||||||
|
///
|
||||||
|
/// In en, this message translates to:
|
||||||
|
/// **'All Files Access disabled. The app will use limited storage access.'**
|
||||||
|
String get allFilesAccessDisabledMessage;
|
||||||
}
|
}
|
||||||
|
|
||||||
class _AppLocalizationsDelegate
|
class _AppLocalizationsDelegate
|
||||||
|
|||||||
@@ -112,7 +112,7 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
'Einzelne Titel-Downloads werden hier angezeigt';
|
'Einzelne Titel-Downloads werden hier angezeigt';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get historySearchHint => 'Search history...';
|
String get historySearchHint => 'Suchverlauf...';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get settingsTitle => 'Einstellungen';
|
String get settingsTitle => 'Einstellungen';
|
||||||
@@ -416,7 +416,7 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
'Der talentierte Künstler, der unser wunderschönes App-Logo entworfen hat!';
|
'Der talentierte Künstler, der unser wunderschönes App-Logo entworfen hat!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutTranslators => 'Translators';
|
String get aboutTranslators => 'Übersetzer';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutSpecialThanks => 'Besonderer Dank';
|
String get aboutSpecialThanks => 'Besonderer Dank';
|
||||||
@@ -445,19 +445,19 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
'Schlage neue Funktionen für die App vor';
|
'Schlage neue Funktionen für die App vor';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutTelegramChannel => 'Telegram Channel';
|
String get aboutTelegramChannel => 'Telegram Kanal';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutTelegramChannelSubtitle => 'Announcements and updates';
|
String get aboutTelegramChannelSubtitle => 'Ankündigungen und Updates';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutTelegramChat => 'Telegram Community';
|
String get aboutTelegramChat => 'Telegram Community';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutTelegramChatSubtitle => 'Chat with other users';
|
String get aboutTelegramChatSubtitle => 'Mit anderen Nutzern chatten';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutSocial => 'Social';
|
String get aboutSocial => 'Sozial';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutSupport => 'Support';
|
String get aboutSupport => 'Support';
|
||||||
@@ -483,6 +483,10 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
String get aboutSachinsenalDesc =>
|
String get aboutSachinsenalDesc =>
|
||||||
'Der ursprüngliche Entwickler des HiFi-Projekts. Die Grundlage der Tidal-Integration!';
|
'Der ursprüngliche Entwickler des HiFi-Projekts. Die Grundlage der Tidal-Integration!';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get aboutSjdonadoDesc =>
|
||||||
|
'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutDoubleDouble => 'DoubleDouble';
|
String get aboutDoubleDouble => 'DoubleDouble';
|
||||||
|
|
||||||
@@ -499,7 +503,7 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutAppDescription =>
|
String get aboutAppDescription =>
|
||||||
'Download Spotify tracks in lossless quality from Tidal, Qobuz, and Amazon Music.';
|
'Lade Spotify-Titel in verlustfreier Qualität von Tidal, Qobuz und Amazon Music herunter.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumTitle => 'Album';
|
String get albumTitle => 'Album';
|
||||||
@@ -509,246 +513,248 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
String _temp0 = intl.Intl.pluralLogic(
|
String _temp0 = intl.Intl.pluralLogic(
|
||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: '$count tracks',
|
other: '$count Songs',
|
||||||
one: '1 track',
|
one: '1 Song',
|
||||||
);
|
);
|
||||||
return '$_temp0';
|
return '$_temp0';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumDownloadAll => 'Download All';
|
String get albumDownloadAll => 'Alle Herunterladen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumDownloadRemaining => 'Download Remaining';
|
String get albumDownloadRemaining => 'Downloads verbleibend';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get playlistTitle => 'Playlist';
|
String get playlistTitle => 'Playlist';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get artistTitle => 'Artist';
|
String get artistTitle => 'Künstler';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get artistAlbums => 'Albums';
|
String get artistAlbums => 'Alben';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get artistSingles => 'Singles & EPs';
|
String get artistSingles => 'Singles & EPs';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get artistCompilations => 'Compilations';
|
String get artistCompilations => 'Zusammenstellungen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String artistReleases(int count) {
|
String artistReleases(int count) {
|
||||||
String _temp0 = intl.Intl.pluralLogic(
|
String _temp0 = intl.Intl.pluralLogic(
|
||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: '$count releases',
|
other: '$count Veröffentlichungen',
|
||||||
one: '1 release',
|
one: '1 Veröffentlichung',
|
||||||
);
|
);
|
||||||
return '$_temp0';
|
return '$_temp0';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get artistPopular => 'Popular';
|
String get artistPopular => 'Beliebt';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String artistMonthlyListeners(String count) {
|
String artistMonthlyListeners(String count) {
|
||||||
return '$count monthly listeners';
|
return '$count monatliche Hörer';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataTitle => 'Track Info';
|
String get trackMetadataTitle => 'Titel Info';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataArtist => 'Artist';
|
String get trackMetadataArtist => 'Künstler';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataAlbum => 'Album';
|
String get trackMetadataAlbum => 'Album';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataDuration => 'Duration';
|
String get trackMetadataDuration => 'Länge';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataQuality => 'Quality';
|
String get trackMetadataQuality => 'Qualität';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataPath => 'File Path';
|
String get trackMetadataPath => 'Dateipfad';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataDownloadedAt => 'Downloaded';
|
String get trackMetadataDownloadedAt => 'Heruntergeladen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataService => 'Service';
|
String get trackMetadataService => 'Anbieter';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataPlay => 'Play';
|
String get trackMetadataPlay => 'Abspielen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataShare => 'Share';
|
String get trackMetadataShare => 'Teilen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataDelete => 'Delete';
|
String get trackMetadataDelete => 'Löschen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataRedownload => 'Re-download';
|
String get trackMetadataRedownload => 'Erneut herunterladen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackMetadataOpenFolder => 'Open Folder';
|
String get trackMetadataOpenFolder => 'Ordner öffnen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupTitle => 'Welcome to SpotiFLAC';
|
String get setupTitle => 'Willkommen bei SpotiFLAC';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupSubtitle => 'Let\'s get you started';
|
String get setupSubtitle => 'Los geht\'s';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStoragePermission => 'Storage Permission';
|
String get setupStoragePermission => 'Speicherberechtigung';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStoragePermissionSubtitle =>
|
String get setupStoragePermissionSubtitle =>
|
||||||
'Required to save downloaded files';
|
'Benötigt um heruntergeladene Dateien zu Speichern';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStoragePermissionGranted => 'Permission granted';
|
String get setupStoragePermissionGranted => 'Berechtigung erteilt';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStoragePermissionDenied => 'Permission denied';
|
String get setupStoragePermissionDenied => 'Berechtigung verweigert';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupGrantPermission => 'Grant Permission';
|
String get setupGrantPermission => 'Berechtigung erlauben';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupDownloadLocation => 'Download Location';
|
String get setupDownloadLocation => 'Speicherort';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupChooseFolder => 'Choose Folder';
|
String get setupChooseFolder => 'Ordner wählen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupContinue => 'Continue';
|
String get setupContinue => 'Fortfahren';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupSkip => 'Skip for now';
|
String get setupSkip => 'Vorerst überspringen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStorageAccessRequired => 'Storage Access Required';
|
String get setupStorageAccessRequired => 'Speicherzugriff erforderlich';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStorageAccessMessage =>
|
String get setupStorageAccessMessage =>
|
||||||
'SpotiFLAC needs \"All files access\" permission to save music files to your chosen folder.';
|
'SpotiFLAC benötigt die Berechtigung \"Auf alle Dateien zugreifen\", um Musikdateien in deinen gewählten Ordner zu speichern.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStorageAccessMessageAndroid11 =>
|
String get setupStorageAccessMessageAndroid11 =>
|
||||||
'Android 11+ requires \"All files access\" permission to save files to your chosen download folder.';
|
'Android 11+ benötigt die Berechtigung „Auf alle Dateien“, um Dateien im ausgewählten Download-Ordner zu speichern.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupOpenSettings => 'Open Settings';
|
String get setupOpenSettings => 'Einstellungen öffnen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupPermissionDeniedMessage =>
|
String get setupPermissionDeniedMessage =>
|
||||||
'Permission denied. Please grant all permissions to continue.';
|
'Berechtigung verweigert. Bitte erteilen Sie alle Berechtigungen um fortzufahren.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String setupPermissionRequired(String permissionType) {
|
String setupPermissionRequired(String permissionType) {
|
||||||
return '$permissionType Permission Required';
|
return '$permissionType Zugriff verweigert';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String setupPermissionRequiredMessage(String permissionType) {
|
String setupPermissionRequiredMessage(String permissionType) {
|
||||||
return '$permissionType permission is required for the best experience. You can change this later in Settings.';
|
return '$permissionType Berechtigung ist erforderlich für\ndie beste Benutzererfahrung. Für kannst dies später in den Einstellungen ändern.';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupSelectDownloadFolder => 'Select Download Folder';
|
String get setupSelectDownloadFolder => 'Wähle Download-Ordner aus';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupUseDefaultFolder => 'Use Default Folder?';
|
String get setupUseDefaultFolder => 'Als Standardordner verwenden?';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupNoFolderSelected =>
|
String get setupNoFolderSelected =>
|
||||||
'No folder selected. Would you like to use the default Music folder?';
|
'Kein Ordner ausgewählt. Soll der Standard-Musikordner verwendet werden?';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupUseDefault => 'Use Default';
|
String get setupUseDefault => 'Standart benutzen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupDownloadLocationTitle => 'Download Location';
|
String get setupDownloadLocationTitle => 'Speicherort';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupDownloadLocationIosMessage =>
|
String get setupDownloadLocationIosMessage =>
|
||||||
'On iOS, downloads are saved to the app\'s Documents folder. You can access them via the Files app.';
|
'Auf iOS werden Downloads im Dokumentenverzeichnis der App gespeichert. Sie können sie über die Datei-App aufrufen.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupAppDocumentsFolder => 'App Documents Folder';
|
String get setupAppDocumentsFolder => 'App-Dokumentenordner';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupAppDocumentsFolderSubtitle =>
|
String get setupAppDocumentsFolderSubtitle =>
|
||||||
'Recommended - accessible via Files app';
|
'Empfohlen - zugänglich über die Datei-App';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupChooseFromFiles => 'Choose from Files';
|
String get setupChooseFromFiles => 'Aus Dateien auswählen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupChooseFromFilesSubtitle => 'Select iCloud or other location';
|
String get setupChooseFromFilesSubtitle =>
|
||||||
|
'Wählen Sie iCloud oder einen anderen Ort';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupIosEmptyFolderWarning =>
|
String get setupIosEmptyFolderWarning =>
|
||||||
'iOS limitation: Empty folders cannot be selected. Choose a folder with at least one file.';
|
'iOS-Einschränkung: Leere Ordner können nicht ausgewählt werden. Wählen Sie einen Ordner mit mindestens einer Datei.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupDownloadInFlac => 'Download Spotify tracks in FLAC';
|
String get setupDownloadInFlac => 'Spotify Titel in FLAC herunterladen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStepStorage => 'Storage';
|
String get setupStepStorage => 'Speicherort';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStepNotification => 'Notification';
|
String get setupStepNotification => 'Benachrichtigung';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStepFolder => 'Folder';
|
String get setupStepFolder => 'Ordner';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStepSpotify => 'Spotify';
|
String get setupStepSpotify => 'Spotify';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStepPermission => 'Permission';
|
String get setupStepPermission => 'Berechtigung';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStorageGranted => 'Storage Permission Granted!';
|
String get setupStorageGranted => 'Speicherberechtigung erlaubt!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStorageRequired => 'Storage Permission Required';
|
String get setupStorageRequired => 'Speicherzugriff erforderlich';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStorageDescription =>
|
String get setupStorageDescription =>
|
||||||
'SpotiFLAC needs storage permission to save your downloaded music files.';
|
'SpotiFLAC benötigt Speicherrechte, um die heruntergeladenen Musikdateien zu speichern.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupNotificationGranted => 'Notification Permission Granted!';
|
String get setupNotificationGranted =>
|
||||||
|
'Benachrichtigungs-Berechtigung erteilt';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupNotificationEnable => 'Enable Notifications';
|
String get setupNotificationEnable => 'Benachrichtigungen aktivieren';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupNotificationDescription =>
|
String get setupNotificationDescription =>
|
||||||
'Get notified when downloads complete or require attention.';
|
'Benachrichtigt werden, wenn Downloads abgeschlossen sind.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupFolderSelected => 'Download Folder Selected!';
|
String get setupFolderSelected => 'Download Ordner ausgewählt!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupFolderChoose => 'Choose Download Folder';
|
String get setupFolderChoose => 'Speicherort auwählen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupFolderDescription =>
|
String get setupFolderDescription =>
|
||||||
'Select a folder where your downloaded music will be saved.';
|
'Wählen Sie einen Ordner, in dem Ihre heruntergeladene Musik gespeichert wird.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupChangeFolder => 'Change Folder';
|
String get setupChangeFolder => 'Ordner ändern';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupSelectFolder => 'Select Folder';
|
String get setupSelectFolder => 'Ordner wählen';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupSpotifyApiOptional => 'Spotify API (Optional)';
|
String get setupSpotifyApiOptional => 'Spotify-API (optional)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupSpotifyApiDescription =>
|
String get setupSpotifyApiDescription =>
|
||||||
@@ -1869,20 +1875,36 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3 => 'MP3';
|
String get qualityLossy => 'Lossy';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3Subtitle => '320kbps (converted from FLAC)';
|
String get qualityLossyMp3Subtitle => 'MP3 320kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3Option => 'Enable MP3 Option';
|
String get qualityLossyOpusSubtitle => 'Opus 128kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOn => 'MP3 quality option is available';
|
String get enableLossyOption => 'Enable Lossy Option';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOff =>
|
String get enableLossyOptionSubtitleOn => 'Lossy quality option is available';
|
||||||
'Downloads FLAC then converts to 320kbps MP3';
|
|
||||||
|
@override
|
||||||
|
String get enableLossyOptionSubtitleOff =>
|
||||||
|
'Downloads FLAC then converts to lossy format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormat => 'Lossy Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatDescription => 'Choose the lossy format for conversion';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatMp3Subtitle => '320kbps, best compatibility';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatOpusSubtitle =>
|
||||||
|
'128kbps, better quality at smaller size';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityNote =>
|
String get qualityNote =>
|
||||||
@@ -2177,4 +2199,28 @@ class AppLocalizationsDe extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get sectionStorageAccess => 'Storage Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccess => 'All Files Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessEnabledSubtitle => 'Can write to any folder';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledSubtitle => 'Limited to media folders only';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDescription =>
|
||||||
|
'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDeniedMessage =>
|
||||||
|
'Permission was denied. Please enable \'All files access\' manually in system settings.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledMessage =>
|
||||||
|
'All Files Access disabled. The app will use limited storage access.';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -470,6 +470,10 @@ class AppLocalizationsEn extends AppLocalizations {
|
|||||||
String get aboutSachinsenalDesc =>
|
String get aboutSachinsenalDesc =>
|
||||||
'The original HiFi project creator. The foundation of Tidal integration!';
|
'The original HiFi project creator. The foundation of Tidal integration!';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get aboutSjdonadoDesc =>
|
||||||
|
'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutDoubleDouble => 'DoubleDouble';
|
String get aboutDoubleDouble => 'DoubleDouble';
|
||||||
|
|
||||||
@@ -1856,20 +1860,36 @@ class AppLocalizationsEn extends AppLocalizations {
|
|||||||
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3 => 'MP3';
|
String get qualityLossy => 'Lossy';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3Subtitle => '320kbps (converted from FLAC)';
|
String get qualityLossyMp3Subtitle => 'MP3 320kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3Option => 'Enable MP3 Option';
|
String get qualityLossyOpusSubtitle => 'Opus 128kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOn => 'MP3 quality option is available';
|
String get enableLossyOption => 'Enable Lossy Option';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOff =>
|
String get enableLossyOptionSubtitleOn => 'Lossy quality option is available';
|
||||||
'Downloads FLAC then converts to 320kbps MP3';
|
|
||||||
|
@override
|
||||||
|
String get enableLossyOptionSubtitleOff =>
|
||||||
|
'Downloads FLAC then converts to lossy format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormat => 'Lossy Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatDescription => 'Choose the lossy format for conversion';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatMp3Subtitle => '320kbps, best compatibility';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatOpusSubtitle =>
|
||||||
|
'128kbps, better quality at smaller size';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityNote =>
|
String get qualityNote =>
|
||||||
@@ -2164,4 +2184,28 @@ class AppLocalizationsEn extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get sectionStorageAccess => 'Storage Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccess => 'All Files Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessEnabledSubtitle => 'Can write to any folder';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledSubtitle => 'Limited to media folders only';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDescription =>
|
||||||
|
'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDeniedMessage =>
|
||||||
|
'Permission was denied. Please enable \'All files access\' manually in system settings.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledMessage =>
|
||||||
|
'All Files Access disabled. The app will use limited storage access.';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -470,6 +470,10 @@ class AppLocalizationsEs extends AppLocalizations {
|
|||||||
String get aboutSachinsenalDesc =>
|
String get aboutSachinsenalDesc =>
|
||||||
'The original HiFi project creator. The foundation of Tidal integration!';
|
'The original HiFi project creator. The foundation of Tidal integration!';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get aboutSjdonadoDesc =>
|
||||||
|
'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutDoubleDouble => 'DoubleDouble';
|
String get aboutDoubleDouble => 'DoubleDouble';
|
||||||
|
|
||||||
@@ -1856,20 +1860,36 @@ class AppLocalizationsEs extends AppLocalizations {
|
|||||||
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3 => 'MP3';
|
String get qualityLossy => 'Lossy';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3Subtitle => '320kbps (converted from FLAC)';
|
String get qualityLossyMp3Subtitle => 'MP3 320kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3Option => 'Enable MP3 Option';
|
String get qualityLossyOpusSubtitle => 'Opus 128kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOn => 'MP3 quality option is available';
|
String get enableLossyOption => 'Enable Lossy Option';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOff =>
|
String get enableLossyOptionSubtitleOn => 'Lossy quality option is available';
|
||||||
'Downloads FLAC then converts to 320kbps MP3';
|
|
||||||
|
@override
|
||||||
|
String get enableLossyOptionSubtitleOff =>
|
||||||
|
'Downloads FLAC then converts to lossy format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormat => 'Lossy Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatDescription => 'Choose the lossy format for conversion';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatMp3Subtitle => '320kbps, best compatibility';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatOpusSubtitle =>
|
||||||
|
'128kbps, better quality at smaller size';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityNote =>
|
String get qualityNote =>
|
||||||
@@ -2164,6 +2184,30 @@ class AppLocalizationsEs extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get sectionStorageAccess => 'Storage Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccess => 'All Files Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessEnabledSubtitle => 'Can write to any folder';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledSubtitle => 'Limited to media folders only';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDescription =>
|
||||||
|
'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDeniedMessage =>
|
||||||
|
'Permission was denied. Please enable \'All files access\' manually in system settings.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledMessage =>
|
||||||
|
'All Files Access disabled. The app will use limited storage access.';
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The translations for Spanish Castilian, as used in Spain (`es_ES`).
|
/// The translations for Spanish Castilian, as used in Spain (`es_ES`).
|
||||||
|
|||||||
@@ -470,6 +470,10 @@ class AppLocalizationsFr extends AppLocalizations {
|
|||||||
String get aboutSachinsenalDesc =>
|
String get aboutSachinsenalDesc =>
|
||||||
'The original HiFi project creator. The foundation of Tidal integration!';
|
'The original HiFi project creator. The foundation of Tidal integration!';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get aboutSjdonadoDesc =>
|
||||||
|
'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutDoubleDouble => 'DoubleDouble';
|
String get aboutDoubleDouble => 'DoubleDouble';
|
||||||
|
|
||||||
@@ -1856,20 +1860,36 @@ class AppLocalizationsFr extends AppLocalizations {
|
|||||||
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3 => 'MP3';
|
String get qualityLossy => 'Lossy';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3Subtitle => '320kbps (converted from FLAC)';
|
String get qualityLossyMp3Subtitle => 'MP3 320kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3Option => 'Enable MP3 Option';
|
String get qualityLossyOpusSubtitle => 'Opus 128kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOn => 'MP3 quality option is available';
|
String get enableLossyOption => 'Enable Lossy Option';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOff =>
|
String get enableLossyOptionSubtitleOn => 'Lossy quality option is available';
|
||||||
'Downloads FLAC then converts to 320kbps MP3';
|
|
||||||
|
@override
|
||||||
|
String get enableLossyOptionSubtitleOff =>
|
||||||
|
'Downloads FLAC then converts to lossy format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormat => 'Lossy Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatDescription => 'Choose the lossy format for conversion';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatMp3Subtitle => '320kbps, best compatibility';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatOpusSubtitle =>
|
||||||
|
'128kbps, better quality at smaller size';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityNote =>
|
String get qualityNote =>
|
||||||
@@ -2164,4 +2184,28 @@ class AppLocalizationsFr extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get sectionStorageAccess => 'Storage Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccess => 'All Files Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessEnabledSubtitle => 'Can write to any folder';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledSubtitle => 'Limited to media folders only';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDescription =>
|
||||||
|
'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDeniedMessage =>
|
||||||
|
'Permission was denied. Please enable \'All files access\' manually in system settings.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledMessage =>
|
||||||
|
'All Files Access disabled. The app will use limited storage access.';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,20 +9,20 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
AppLocalizationsHi([String locale = 'hi']) : super(locale);
|
AppLocalizationsHi([String locale = 'hi']) : super(locale);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appName => 'SpotiFLAC';
|
String get appName => 'SpotiFlac';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appDescription =>
|
String get appDescription =>
|
||||||
'Download Spotify tracks in lossless quality from Tidal, Qobuz, and Amazon Music.';
|
'स्पॉटीफाई ट्रैक डाउनलोड करें टाइडल, क्वाबज एवं एवं अमेजन म्यूजिक से उच्चतम क्वालिटी में।';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get navHome => 'Home';
|
String get navHome => 'होम';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get navHistory => 'History';
|
String get navHistory => 'इतिहास';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get navSettings => 'Settings';
|
String get navSettings => 'विकल्प';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get navStore => 'Store';
|
String get navStore => 'Store';
|
||||||
@@ -184,7 +184,7 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
String get quality128 => '128 kbps';
|
String get quality128 => '128 kbps';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appearanceTitle => 'Appearance';
|
String get appearanceTitle => 'दिखावट';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appearanceTheme => 'Theme';
|
String get appearanceTheme => 'Theme';
|
||||||
@@ -199,10 +199,10 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
String get appearanceThemeDark => 'Dark';
|
String get appearanceThemeDark => 'Dark';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appearanceDynamicColor => 'Dynamic Color';
|
String get appearanceDynamicColor => 'डायनेमिक रंग';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appearanceDynamicColorSubtitle => 'Use colors from your wallpaper';
|
String get appearanceDynamicColorSubtitle => 'वॉलपेपर से रंग इस्तेमाल करें';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appearanceAccentColor => 'Accent Color';
|
String get appearanceAccentColor => 'Accent Color';
|
||||||
@@ -470,6 +470,10 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
String get aboutSachinsenalDesc =>
|
String get aboutSachinsenalDesc =>
|
||||||
'The original HiFi project creator. The foundation of Tidal integration!';
|
'The original HiFi project creator. The foundation of Tidal integration!';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get aboutSjdonadoDesc =>
|
||||||
|
'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutDoubleDouble => 'DoubleDouble';
|
String get aboutDoubleDouble => 'DoubleDouble';
|
||||||
|
|
||||||
@@ -1856,20 +1860,36 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3 => 'MP3';
|
String get qualityLossy => 'Lossy';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3Subtitle => '320kbps (converted from FLAC)';
|
String get qualityLossyMp3Subtitle => 'MP3 320kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3Option => 'Enable MP3 Option';
|
String get qualityLossyOpusSubtitle => 'Opus 128kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOn => 'MP3 quality option is available';
|
String get enableLossyOption => 'Enable Lossy Option';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOff =>
|
String get enableLossyOptionSubtitleOn => 'Lossy quality option is available';
|
||||||
'Downloads FLAC then converts to 320kbps MP3';
|
|
||||||
|
@override
|
||||||
|
String get enableLossyOptionSubtitleOff =>
|
||||||
|
'Downloads FLAC then converts to lossy format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormat => 'Lossy Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatDescription => 'Choose the lossy format for conversion';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatMp3Subtitle => '320kbps, best compatibility';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatOpusSubtitle =>
|
||||||
|
'128kbps, better quality at smaller size';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityNote =>
|
String get qualityNote =>
|
||||||
@@ -2164,4 +2184,28 @@ class AppLocalizationsHi extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get sectionStorageAccess => 'Storage Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccess => 'All Files Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessEnabledSubtitle => 'Can write to any folder';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledSubtitle => 'Limited to media folders only';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDescription =>
|
||||||
|
'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDeniedMessage =>
|
||||||
|
'Permission was denied. Please enable \'All files access\' manually in system settings.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledMessage =>
|
||||||
|
'All Files Access disabled. The app will use limited storage access.';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -475,6 +475,10 @@ class AppLocalizationsId extends AppLocalizations {
|
|||||||
String get aboutSachinsenalDesc =>
|
String get aboutSachinsenalDesc =>
|
||||||
'Pembuat proyek HiFi asli. Fondasi dari integrasi Tidal!';
|
'Pembuat proyek HiFi asli. Fondasi dari integrasi Tidal!';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get aboutSjdonadoDesc =>
|
||||||
|
'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutDoubleDouble => 'DoubleDouble';
|
String get aboutDoubleDouble => 'DoubleDouble';
|
||||||
|
|
||||||
@@ -1868,20 +1872,36 @@ class AppLocalizationsId extends AppLocalizations {
|
|||||||
String get qualityHiResFlacMaxSubtitle => '24-bit / hingga 192kHz';
|
String get qualityHiResFlacMaxSubtitle => '24-bit / hingga 192kHz';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3 => 'MP3';
|
String get qualityLossy => 'Lossy';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3Subtitle => '320kbps (konversi dari FLAC)';
|
String get qualityLossyMp3Subtitle => 'MP3 320kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3Option => 'Aktifkan Opsi MP3';
|
String get qualityLossyOpusSubtitle => 'Opus 128kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOn => 'Opsi kualitas MP3 tersedia';
|
String get enableLossyOption => 'Enable Lossy Option';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOff =>
|
String get enableLossyOptionSubtitleOn => 'Lossy quality option is available';
|
||||||
'Unduh FLAC lalu konversi ke MP3 320kbps';
|
|
||||||
|
@override
|
||||||
|
String get enableLossyOptionSubtitleOff =>
|
||||||
|
'Downloads FLAC then converts to lossy format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormat => 'Lossy Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatDescription => 'Choose the lossy format for conversion';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatMp3Subtitle => '320kbps, best compatibility';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatOpusSubtitle =>
|
||||||
|
'128kbps, better quality at smaller size';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityNote =>
|
String get qualityNote =>
|
||||||
@@ -2113,68 +2133,92 @@ class AppLocalizationsId extends AppLocalizations {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyDownload => 'Unduh Diskografi';
|
String get discographyDownload => 'Download Discography';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyDownloadAll => 'Unduh Semua';
|
String get discographyDownloadAll => 'Unduh Semua';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographyDownloadAllSubtitle(int count, int albumCount) {
|
String discographyDownloadAllSubtitle(int count, int albumCount) {
|
||||||
return '$count lagu dari $albumCount rilis';
|
return '$count tracks from $albumCount releases';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyAlbumsOnly => 'Album Saja';
|
String get discographyAlbumsOnly => 'Albums Only';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographyAlbumsOnlySubtitle(int count, int albumCount) {
|
String discographyAlbumsOnlySubtitle(int count, int albumCount) {
|
||||||
return '$count lagu dari $albumCount album';
|
return '$count tracks from $albumCount albums';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographySinglesOnly => 'Single & EP Saja';
|
String get discographySinglesOnly => 'Singles & EPs Only';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographySinglesOnlySubtitle(int count, int albumCount) {
|
String discographySinglesOnlySubtitle(int count, int albumCount) {
|
||||||
return '$count lagu dari $albumCount single';
|
return '$count tracks from $albumCount singles';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographySelectAlbums => 'Pilih Album...';
|
String get discographySelectAlbums => 'Select Albums...';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographySelectAlbumsSubtitle =>
|
String get discographySelectAlbumsSubtitle =>
|
||||||
'Pilih album atau single tertentu';
|
'Choose specific albums or singles';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFetchingTracks => 'Mengambil lagu...';
|
String get discographyFetchingTracks => 'Fetching tracks...';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographyFetchingAlbum(int current, int total) {
|
String discographyFetchingAlbum(int current, int total) {
|
||||||
return 'Mengambil $current dari $total...';
|
return 'Fetching $current of $total...';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographySelectedCount(int count) {
|
String discographySelectedCount(int count) {
|
||||||
return '$count dipilih';
|
return '$count selected';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyDownloadSelected => 'Unduh yang Dipilih';
|
String get discographyDownloadSelected => 'Download Selected';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographyAddedToQueue(int count) {
|
String discographyAddedToQueue(int count) {
|
||||||
return 'Menambahkan $count lagu ke antrian';
|
return 'Added $count tracks to queue';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographySkippedDownloaded(int added, int skipped) {
|
String discographySkippedDownloaded(int added, int skipped) {
|
||||||
return '$added ditambahkan, $skipped sudah diunduh';
|
return '$added added, $skipped already downloaded';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyNoAlbums => 'Tidak ada album tersedia';
|
String get discographyNoAlbums => 'No albums available';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFailedToFetch => 'Gagal mengambil beberapa album';
|
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get sectionStorageAccess => 'Storage Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccess => 'All Files Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessEnabledSubtitle => 'Can write to any folder';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledSubtitle => 'Limited to media folders only';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDescription =>
|
||||||
|
'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDeniedMessage =>
|
||||||
|
'Permission was denied. Please enable \'All files access\' manually in system settings.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledMessage =>
|
||||||
|
'All Files Access disabled. The app will use limited storage access.';
|
||||||
}
|
}
|
||||||
|
|||||||
+469
-439
File diff suppressed because it is too large
Load Diff
@@ -470,6 +470,10 @@ class AppLocalizationsKo extends AppLocalizations {
|
|||||||
String get aboutSachinsenalDesc =>
|
String get aboutSachinsenalDesc =>
|
||||||
'The original HiFi project creator. The foundation of Tidal integration!';
|
'The original HiFi project creator. The foundation of Tidal integration!';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get aboutSjdonadoDesc =>
|
||||||
|
'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutDoubleDouble => 'DoubleDouble';
|
String get aboutDoubleDouble => 'DoubleDouble';
|
||||||
|
|
||||||
@@ -1856,20 +1860,36 @@ class AppLocalizationsKo extends AppLocalizations {
|
|||||||
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3 => 'MP3';
|
String get qualityLossy => 'Lossy';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3Subtitle => '320kbps (converted from FLAC)';
|
String get qualityLossyMp3Subtitle => 'MP3 320kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3Option => 'Enable MP3 Option';
|
String get qualityLossyOpusSubtitle => 'Opus 128kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOn => 'MP3 quality option is available';
|
String get enableLossyOption => 'Enable Lossy Option';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOff =>
|
String get enableLossyOptionSubtitleOn => 'Lossy quality option is available';
|
||||||
'Downloads FLAC then converts to 320kbps MP3';
|
|
||||||
|
@override
|
||||||
|
String get enableLossyOptionSubtitleOff =>
|
||||||
|
'Downloads FLAC then converts to lossy format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormat => 'Lossy Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatDescription => 'Choose the lossy format for conversion';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatMp3Subtitle => '320kbps, best compatibility';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatOpusSubtitle =>
|
||||||
|
'128kbps, better quality at smaller size';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityNote =>
|
String get qualityNote =>
|
||||||
@@ -2164,4 +2184,28 @@ class AppLocalizationsKo extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get sectionStorageAccess => 'Storage Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccess => 'All Files Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessEnabledSubtitle => 'Can write to any folder';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledSubtitle => 'Limited to media folders only';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDescription =>
|
||||||
|
'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDeniedMessage =>
|
||||||
|
'Permission was denied. Please enable \'All files access\' manually in system settings.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledMessage =>
|
||||||
|
'All Files Access disabled. The app will use limited storage access.';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -470,6 +470,10 @@ class AppLocalizationsNl extends AppLocalizations {
|
|||||||
String get aboutSachinsenalDesc =>
|
String get aboutSachinsenalDesc =>
|
||||||
'The original HiFi project creator. The foundation of Tidal integration!';
|
'The original HiFi project creator. The foundation of Tidal integration!';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get aboutSjdonadoDesc =>
|
||||||
|
'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutDoubleDouble => 'DoubleDouble';
|
String get aboutDoubleDouble => 'DoubleDouble';
|
||||||
|
|
||||||
@@ -1856,20 +1860,36 @@ class AppLocalizationsNl extends AppLocalizations {
|
|||||||
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3 => 'MP3';
|
String get qualityLossy => 'Lossy';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3Subtitle => '320kbps (converted from FLAC)';
|
String get qualityLossyMp3Subtitle => 'MP3 320kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3Option => 'Enable MP3 Option';
|
String get qualityLossyOpusSubtitle => 'Opus 128kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOn => 'MP3 quality option is available';
|
String get enableLossyOption => 'Enable Lossy Option';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOff =>
|
String get enableLossyOptionSubtitleOn => 'Lossy quality option is available';
|
||||||
'Downloads FLAC then converts to 320kbps MP3';
|
|
||||||
|
@override
|
||||||
|
String get enableLossyOptionSubtitleOff =>
|
||||||
|
'Downloads FLAC then converts to lossy format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormat => 'Lossy Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatDescription => 'Choose the lossy format for conversion';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatMp3Subtitle => '320kbps, best compatibility';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatOpusSubtitle =>
|
||||||
|
'128kbps, better quality at smaller size';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityNote =>
|
String get qualityNote =>
|
||||||
@@ -2164,4 +2184,28 @@ class AppLocalizationsNl extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get sectionStorageAccess => 'Storage Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccess => 'All Files Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessEnabledSubtitle => 'Can write to any folder';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledSubtitle => 'Limited to media folders only';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDescription =>
|
||||||
|
'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDeniedMessage =>
|
||||||
|
'Permission was denied. Please enable \'All files access\' manually in system settings.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledMessage =>
|
||||||
|
'All Files Access disabled. The app will use limited storage access.';
|
||||||
}
|
}
|
||||||
|
|||||||
+206
-155
@@ -470,6 +470,10 @@ class AppLocalizationsPt extends AppLocalizations {
|
|||||||
String get aboutSachinsenalDesc =>
|
String get aboutSachinsenalDesc =>
|
||||||
'The original HiFi project creator. The foundation of Tidal integration!';
|
'The original HiFi project creator. The foundation of Tidal integration!';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get aboutSjdonadoDesc =>
|
||||||
|
'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutDoubleDouble => 'DoubleDouble';
|
String get aboutDoubleDouble => 'DoubleDouble';
|
||||||
|
|
||||||
@@ -1856,20 +1860,36 @@ class AppLocalizationsPt extends AppLocalizations {
|
|||||||
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
String get qualityHiResFlacMaxSubtitle => '24-bit / up to 192kHz';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3 => 'MP3';
|
String get qualityLossy => 'Lossy';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3Subtitle => '320kbps (converted from FLAC)';
|
String get qualityLossyMp3Subtitle => 'MP3 320kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3Option => 'Enable MP3 Option';
|
String get qualityLossyOpusSubtitle => 'Opus 128kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOn => 'MP3 quality option is available';
|
String get enableLossyOption => 'Enable Lossy Option';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOff =>
|
String get enableLossyOptionSubtitleOn => 'Lossy quality option is available';
|
||||||
'Downloads FLAC then converts to 320kbps MP3';
|
|
||||||
|
@override
|
||||||
|
String get enableLossyOptionSubtitleOff =>
|
||||||
|
'Downloads FLAC then converts to lossy format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormat => 'Lossy Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatDescription => 'Choose the lossy format for conversion';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatMp3Subtitle => '320kbps, best compatibility';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatOpusSubtitle =>
|
||||||
|
'128kbps, better quality at smaller size';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityNote =>
|
String get qualityNote =>
|
||||||
@@ -2164,6 +2184,30 @@ class AppLocalizationsPt extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get sectionStorageAccess => 'Storage Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccess => 'All Files Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessEnabledSubtitle => 'Can write to any folder';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledSubtitle => 'Limited to media folders only';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDescription =>
|
||||||
|
'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDeniedMessage =>
|
||||||
|
'Permission was denied. Please enable \'All files access\' manually in system settings.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledMessage =>
|
||||||
|
'All Files Access disabled. The app will use limited storage access.';
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The translations for Portuguese, as used in Portugal (`pt_PT`).
|
/// The translations for Portuguese, as used in Portugal (`pt_PT`).
|
||||||
@@ -2833,32 +2877,32 @@ class AppLocalizationsPtPt extends AppLocalizationsPt {
|
|||||||
'Limitação do iOS: Pastas vazias não podem ser selecionadas. Escolha uma pasta com pelo menos um arquivo.';
|
'Limitação do iOS: Pastas vazias não podem ser selecionadas. Escolha uma pasta com pelo menos um arquivo.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupDownloadInFlac => 'Download Spotify tracks in FLAC';
|
String get setupDownloadInFlac => 'Baixe faixas do Spotify em FLAC';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStepStorage => 'Storage';
|
String get setupStepStorage => 'Armazenamento';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStepNotification => 'Notification';
|
String get setupStepNotification => 'Notificação';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStepFolder => 'Folder';
|
String get setupStepFolder => 'Pasta';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStepSpotify => 'Spotify';
|
String get setupStepSpotify => 'Spotify';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStepPermission => 'Permission';
|
String get setupStepPermission => 'Permissão';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStorageGranted => 'Storage Permission Granted!';
|
String get setupStorageGranted => 'Permissão de Armazenamento Concedida!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStorageRequired => 'Storage Permission Required';
|
String get setupStorageRequired => 'Permissão de Armazenamento Necessária';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupStorageDescription =>
|
String get setupStorageDescription =>
|
||||||
'SpotiFLAC needs storage permission to save your downloaded music files.';
|
'O SpotiFLAC precisa de permissão de armazenamento para salvar os seus arquivos de música baixados.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get setupNotificationGranted => 'Permissão de Notificações Concedida!';
|
String get setupNotificationGranted => 'Permissão de Notificações Concedida!';
|
||||||
@@ -3022,171 +3066,172 @@ class AppLocalizationsPtPt extends AppLocalizationsPt {
|
|||||||
'Você tem certeza que deseja limpar todos os downloads?';
|
'Você tem certeza que deseja limpar todos os downloads?';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get dialogRemoveFromDevice => 'Remove from device?';
|
String get dialogRemoveFromDevice => 'Remover do dispositivo?';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get dialogRemoveExtension => 'Remove Extension';
|
String get dialogRemoveExtension => 'Remover Extensão';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get dialogRemoveExtensionMessage =>
|
String get dialogRemoveExtensionMessage =>
|
||||||
'Are you sure you want to remove this extension? This cannot be undone.';
|
'Tem certeza de que deseja remover esta extensão? Isso não pode ser desfeito.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get dialogUninstallExtension => 'Uninstall Extension?';
|
String get dialogUninstallExtension => 'Desinstalar Extensão?';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String dialogUninstallExtensionMessage(String extensionName) {
|
String dialogUninstallExtensionMessage(String extensionName) {
|
||||||
return 'Are you sure you want to remove $extensionName?';
|
return 'Tem certeza de que deseja remover $extensionName?';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get dialogClearHistoryTitle => 'Clear History';
|
String get dialogClearHistoryTitle => 'Limpar Histórico';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get dialogClearHistoryMessage =>
|
String get dialogClearHistoryMessage =>
|
||||||
'Are you sure you want to clear all download history? This cannot be undone.';
|
'Tem certeza de que deseja limpar todo o histórico de downloads? Isso não pode ser desfeito.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get dialogDeleteSelectedTitle => 'Delete Selected';
|
String get dialogDeleteSelectedTitle => 'Apagar Selecionados';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String dialogDeleteSelectedMessage(int count) {
|
String dialogDeleteSelectedMessage(int count) {
|
||||||
String _temp0 = intl.Intl.pluralLogic(
|
String _temp0 = intl.Intl.pluralLogic(
|
||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: 'tracks',
|
other: 'faixas',
|
||||||
one: 'track',
|
one: 'faixa',
|
||||||
);
|
);
|
||||||
return 'Delete $count $_temp0 from history?\n\nThis will also delete the files from storage.';
|
return 'Apagar $count $_temp0 do histórico?\n\nIsso também apagará os arquivos do armazenamento.';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get dialogImportPlaylistTitle => 'Import Playlist';
|
String get dialogImportPlaylistTitle => 'Importar Playlist';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String dialogImportPlaylistMessage(int count) {
|
String dialogImportPlaylistMessage(int count) {
|
||||||
return 'Found $count tracks in CSV. Add them to download queue?';
|
return 'Encontradas $count faixas no CSV. Adicionar à fila de download?';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String snackbarAddedToQueue(String trackName) {
|
String snackbarAddedToQueue(String trackName) {
|
||||||
return 'Added \"$trackName\" to queue';
|
return '\"$trackName\" adicionada à fila';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String snackbarAddedTracksToQueue(int count) {
|
String snackbarAddedTracksToQueue(int count) {
|
||||||
return 'Added $count tracks to queue';
|
return '$count faixas adicionadas à fila';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String snackbarAlreadyDownloaded(String trackName) {
|
String snackbarAlreadyDownloaded(String trackName) {
|
||||||
return '\"$trackName\" already downloaded';
|
return '\"$trackName\" já foi baixada';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarHistoryCleared => 'History cleared';
|
String get snackbarHistoryCleared => 'Histórico limpo';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarCredentialsSaved => 'Credentials saved';
|
String get snackbarCredentialsSaved => 'Credenciais salvas';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarCredentialsCleared => 'Credentials cleared';
|
String get snackbarCredentialsCleared => 'Credenciais removidas';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String snackbarDeletedTracks(int count) {
|
String snackbarDeletedTracks(int count) {
|
||||||
String _temp0 = intl.Intl.pluralLogic(
|
String _temp0 = intl.Intl.pluralLogic(
|
||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: 'tracks',
|
other: 'faixas apagadas',
|
||||||
one: 'track',
|
one: 'faixa apagada',
|
||||||
);
|
);
|
||||||
return 'Deleted $count $_temp0';
|
return '$count $_temp0';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String snackbarCannotOpenFile(String error) {
|
String snackbarCannotOpenFile(String error) {
|
||||||
return 'Cannot open file: $error';
|
return 'Não foi possível abrir o arquivo: $error';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarFillAllFields => 'Please fill all fields';
|
String get snackbarFillAllFields => 'Por favor, preencha todos os campos';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarViewQueue => 'View Queue';
|
String get snackbarViewQueue => 'Ver Fila';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String snackbarFailedToLoad(String error) {
|
String snackbarFailedToLoad(String error) {
|
||||||
return 'Failed to load: $error';
|
return 'Falha ao carregar: $error';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String snackbarUrlCopied(String platform) {
|
String snackbarUrlCopied(String platform) {
|
||||||
return '$platform URL copied to clipboard';
|
return 'URL do $platform copiada para a área de transferência';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarFileNotFound => 'File not found';
|
String get snackbarFileNotFound => 'Arquivo não encontrado';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarSelectExtFile => 'Please select a .spotiflac-ext file';
|
String get snackbarSelectExtFile =>
|
||||||
|
'Por favor, selecione um arquivo .spotiflac-ext';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarProviderPrioritySaved => 'Provider priority saved';
|
String get snackbarProviderPrioritySaved => 'Prioridade de provedor salva';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarMetadataProviderSaved =>
|
String get snackbarMetadataProviderSaved =>
|
||||||
'Metadata provider priority saved';
|
'Prioridade de provedor de metadados salva';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String snackbarExtensionInstalled(String extensionName) {
|
String snackbarExtensionInstalled(String extensionName) {
|
||||||
return '$extensionName installed.';
|
return '$extensionName instalada.';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String snackbarExtensionUpdated(String extensionName) {
|
String snackbarExtensionUpdated(String extensionName) {
|
||||||
return '$extensionName updated.';
|
return '$extensionName atualizada.';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarFailedToInstall => 'Failed to install extension';
|
String get snackbarFailedToInstall => 'Falha ao instalar extensão';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get snackbarFailedToUpdate => 'Failed to update extension';
|
String get snackbarFailedToUpdate => 'Falha ao atualizar extensão';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get errorRateLimited => 'Rate Limited';
|
String get errorRateLimited => 'Taxa Limitada';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get errorRateLimitedMessage =>
|
String get errorRateLimitedMessage =>
|
||||||
'Too many requests. Please wait a moment before searching again.';
|
'Muitas solicitações. Por favor, aguarde um momento antes de pesquisar novamente.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String errorFailedToLoad(String item) {
|
String errorFailedToLoad(String item) {
|
||||||
return 'Failed to load $item';
|
return 'Falha ao carregar $item';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get errorNoTracksFound => 'No tracks found';
|
String get errorNoTracksFound => 'Nenhuma faixa encontrada';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String errorMissingExtensionSource(String item) {
|
String errorMissingExtensionSource(String item) {
|
||||||
return 'Cannot load $item: missing extension source';
|
return 'Não foi possível carregar $item: fonte de extensão ausente';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get statusQueued => 'Queued';
|
String get statusQueued => 'Na Fila';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get statusDownloading => 'Downloading';
|
String get statusDownloading => 'Baixando';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get statusFinalizing => 'Finalizing';
|
String get statusFinalizing => 'Finalizando';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get statusCompleted => 'Completed';
|
String get statusCompleted => 'Concluído';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get statusFailed => 'Failed';
|
String get statusFailed => 'Falhou';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get statusSkipped => 'Ignorado';
|
String get statusSkipped => 'Ignorado';
|
||||||
@@ -3523,42 +3568,43 @@ class AppLocalizationsPtPt extends AppLocalizationsPt {
|
|||||||
String get logNetworkErrorDescription => 'Problemas de conexão detectados';
|
String get logNetworkErrorDescription => 'Problemas de conexão detectados';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get logNetworkErrorSuggestion => 'Check your internet connection';
|
String get logNetworkErrorSuggestion =>
|
||||||
|
'Verifique a sua conexão com a internet';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get logTrackNotFoundDescription =>
|
String get logTrackNotFoundDescription =>
|
||||||
'Some tracks could not be found on download services';
|
'Algumas faixas não foram encontradas nos serviços de download';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get logTrackNotFoundSuggestion =>
|
String get logTrackNotFoundSuggestion =>
|
||||||
'The track may not be available in lossless quality';
|
'A faixa pode não estar disponível em qualidade lossless';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String logTotalErrors(int count) {
|
String logTotalErrors(int count) {
|
||||||
return 'Total errors: $count';
|
return 'Total de erros: $count';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String logAffected(String domains) {
|
String logAffected(String domains) {
|
||||||
return 'Affected: $domains';
|
return 'Afetados: $domains';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String logEntriesFiltered(int count) {
|
String logEntriesFiltered(int count) {
|
||||||
return 'Entries ($count filtered)';
|
return 'Entradas ($count filtradas)';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String logEntries(int count) {
|
String logEntries(int count) {
|
||||||
return 'Entries ($count)';
|
return 'Entradas ($count)';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get credentialsTitle => 'Spotify Credentials';
|
String get credentialsTitle => 'Credenciais do Spotify';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get credentialsDescription =>
|
String get credentialsDescription =>
|
||||||
'Enter your Client ID and Secret to use your own Spotify application quota.';
|
'Insira o seu Client ID e Secret para usar a sua própria cota de aplicativo do Spotify.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get credentialsClientId => 'Client ID';
|
String get credentialsClientId => 'Client ID';
|
||||||
@@ -3723,136 +3769,138 @@ class AppLocalizationsPtPt extends AppLocalizationsPt {
|
|||||||
String get trackDownloaded => 'Baixado';
|
String get trackDownloaded => 'Baixado';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackCopyLyrics => 'Copy lyrics';
|
String get trackCopyLyrics => 'Copiar letras';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsNotAvailable => 'Lyrics not available for this track';
|
String get trackLyricsNotAvailable =>
|
||||||
|
'Letras não disponíveis para esta faixa';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsTimeout => 'Request timed out. Try again later.';
|
String get trackLyricsTimeout =>
|
||||||
|
'A solicitação expirou. Tente novamente mais tarde.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsLoadFailed => 'Failed to load lyrics';
|
String get trackLyricsLoadFailed => 'Falha ao carregar letras';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackCopiedToClipboard => 'Copied to clipboard';
|
String get trackCopiedToClipboard => 'Copiado para a área de transferência';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackDeleteConfirmTitle => 'Remove from device?';
|
String get trackDeleteConfirmTitle => 'Remover do dispositivo?';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackDeleteConfirmMessage =>
|
String get trackDeleteConfirmMessage =>
|
||||||
'This will permanently delete the downloaded file and remove it from your history.';
|
'Isso apagará permanentemente o arquivo baixado e o removerá do seu histórico.';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String trackCannotOpen(String message) {
|
String trackCannotOpen(String message) {
|
||||||
return 'Cannot open: $message';
|
return 'Não foi possível abrir: $message';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get dateToday => 'Today';
|
String get dateToday => 'Hoje';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get dateYesterday => 'Yesterday';
|
String get dateYesterday => 'Ontem';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String dateDaysAgo(int count) {
|
String dateDaysAgo(int count) {
|
||||||
return '$count days ago';
|
return 'Há $count dias';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String dateWeeksAgo(int count) {
|
String dateWeeksAgo(int count) {
|
||||||
return '$count weeks ago';
|
return 'Há $count semanas';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String dateMonthsAgo(int count) {
|
String dateMonthsAgo(int count) {
|
||||||
return '$count months ago';
|
return 'Há $count meses';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get concurrentSequential => 'Sequential';
|
String get concurrentSequential => 'Sequencial';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get concurrentParallel2 => '2 Parallel';
|
String get concurrentParallel2 => '2 Paralelos';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get concurrentParallel3 => '3 Parallel';
|
String get concurrentParallel3 => '3 Paralelos';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get tapToSeeError => 'Tap to see error details';
|
String get tapToSeeError => 'Toque para ver detalhes do erro';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get storeFilterAll => 'All';
|
String get storeFilterAll => 'Todos';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get storeFilterMetadata => 'Metadata';
|
String get storeFilterMetadata => 'Metadados';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get storeFilterDownload => 'Download';
|
String get storeFilterDownload => 'Download';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get storeFilterUtility => 'Utility';
|
String get storeFilterUtility => 'Utilitário';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get storeFilterLyrics => 'Lyrics';
|
String get storeFilterLyrics => 'Letras';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get storeFilterIntegration => 'Integration';
|
String get storeFilterIntegration => 'Integração';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get storeClearFilters => 'Clear filters';
|
String get storeClearFilters => 'Limpar filtros';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get storeNoResults => 'No extensions found';
|
String get storeNoResults => 'Nenhuma extensão encontrada';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionProviderPriority => 'Provider Priority';
|
String get extensionProviderPriority => 'Prioridade de Provedor';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionInstallButton => 'Install Extension';
|
String get extensionInstallButton => 'Instalar Extensão';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionDefaultProvider => 'Default (Deezer/Spotify)';
|
String get extensionDefaultProvider => 'Padrão (Deezer/Spotify)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionDefaultProviderSubtitle => 'Use built-in search';
|
String get extensionDefaultProviderSubtitle => 'Usar pesquisa integrada';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionAuthor => 'Author';
|
String get extensionAuthor => 'Autor';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionId => 'ID';
|
String get extensionId => 'ID';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionError => 'Error';
|
String get extensionError => 'Erro';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionCapabilities => 'Capabilities';
|
String get extensionCapabilities => 'Capacidades';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionMetadataProvider => 'Metadata Provider';
|
String get extensionMetadataProvider => 'Provedor de Metadados';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionDownloadProvider => 'Download Provider';
|
String get extensionDownloadProvider => 'Provedor de Download';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionLyricsProvider => 'Lyrics Provider';
|
String get extensionLyricsProvider => 'Provedor de Letras';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionUrlHandler => 'URL Handler';
|
String get extensionUrlHandler => 'Manipulador de URL';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionQualityOptions => 'Quality Options';
|
String get extensionQualityOptions => 'Opções de Qualidade';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionPostProcessingHooks => 'Post-Processing Hooks';
|
String get extensionPostProcessingHooks => 'Ganchos de Pós-Processamento';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionPermissions => 'Permissions';
|
String get extensionPermissions => 'Permissões';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionSettings => 'Settings';
|
String get extensionSettings => 'Configurações';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get extensionRemoveButton => 'Remover Extensão';
|
String get extensionRemoveButton => 'Remover Extensão';
|
||||||
@@ -4003,25 +4051,27 @@ class AppLocalizationsPtPt extends AppLocalizationsPt {
|
|||||||
String get folderNone => 'Nenhum';
|
String get folderNone => 'Nenhum';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get folderNoneSubtitle => 'Save all files directly to download folder';
|
String get folderNoneSubtitle =>
|
||||||
|
'Salvar todos os arquivos diretamente na pasta de download';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get folderArtist => 'Artist';
|
String get folderArtist => 'Artista';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get folderArtistSubtitle => 'Artist Name/filename';
|
String get folderArtistSubtitle => 'Nome do Artista/nome do arquivo';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get folderAlbum => 'Album';
|
String get folderAlbum => 'Álbum';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get folderAlbumSubtitle => 'Album Name/filename';
|
String get folderAlbumSubtitle => 'Nome do Álbum/nome do arquivo';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get folderArtistAlbum => 'Artist/Album';
|
String get folderArtistAlbum => 'Artista/Álbum';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get folderArtistAlbumSubtitle => 'Artist Name/Album Name/filename';
|
String get folderArtistAlbumSubtitle =>
|
||||||
|
'Nome do Artista/Nome do Álbum/nome do arquivo';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get serviceTidal => 'Tidal';
|
String get serviceTidal => 'Tidal';
|
||||||
@@ -4039,134 +4089,135 @@ class AppLocalizationsPtPt extends AppLocalizationsPt {
|
|||||||
String get serviceSpotify => 'Spotify';
|
String get serviceSpotify => 'Spotify';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appearanceAmoledDark => 'AMOLED Dark';
|
String get appearanceAmoledDark => 'AMOLED Escuro';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appearanceAmoledDarkSubtitle => 'Pure black background';
|
String get appearanceAmoledDarkSubtitle => 'Fundo preto puro';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appearanceChooseAccentColor => 'Choose Accent Color';
|
String get appearanceChooseAccentColor => 'Escolher Cor de Destaque';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get appearanceChooseTheme => 'Theme Mode';
|
String get appearanceChooseTheme => 'Modo de Tema';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueTitle => 'Download Queue';
|
String get queueTitle => 'Fila de Download';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueClearAll => 'Clear All';
|
String get queueClearAll => 'Limpar Tudo';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueClearAllMessage =>
|
String get queueClearAllMessage =>
|
||||||
'Are you sure you want to clear all downloads?';
|
'Tem certeza de que deseja limpar todos os downloads?';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueEmpty => 'No downloads in queue';
|
String get queueEmpty => 'Nenhum download na fila';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueEmptySubtitle => 'Add tracks from the home screen';
|
String get queueEmptySubtitle => 'Adicione faixas a partir da tela inicial';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueClearCompleted => 'Clear completed';
|
String get queueClearCompleted => 'Limpar concluídos';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueDownloadFailed => 'Download Failed';
|
String get queueDownloadFailed => 'Download Falhou';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueTrackLabel => 'Track:';
|
String get queueTrackLabel => 'Faixa:';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueArtistLabel => 'Artist:';
|
String get queueArtistLabel => 'Artista:';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueErrorLabel => 'Error:';
|
String get queueErrorLabel => 'Erro:';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get queueUnknownError => 'Unknown error';
|
String get queueUnknownError => 'Erro desconhecido';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumFolderArtistAlbum => 'Artist / Album';
|
String get albumFolderArtistAlbum => 'Artista / Álbum';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumFolderArtistAlbumSubtitle => 'Albums/Artist Name/Album Name/';
|
String get albumFolderArtistAlbumSubtitle =>
|
||||||
|
'Álbuns/Nome do Artista/Nome do Álbum/';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumFolderArtistYearAlbum => 'Artist / [Year] Album';
|
String get albumFolderArtistYearAlbum => 'Artista / [Ano] Álbum';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumFolderArtistYearAlbumSubtitle =>
|
String get albumFolderArtistYearAlbumSubtitle =>
|
||||||
'Albums/Artist Name/[2005] Album Name/';
|
'Álbuns/Nome do Artista/[2005] Nome do Álbum/';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumFolderAlbumOnly => 'Album Only';
|
String get albumFolderAlbumOnly => 'Apenas Álbum';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumFolderAlbumOnlySubtitle => 'Albums/Album Name/';
|
String get albumFolderAlbumOnlySubtitle => 'Álbuns/Nome do Álbum/';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumFolderYearAlbum => '[Year] Album';
|
String get albumFolderYearAlbum => '[Ano] Álbum';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumFolderYearAlbumSubtitle => 'Albums/[2005] Album Name/';
|
String get albumFolderYearAlbumSubtitle => 'Álbuns/[2005] Nome do Álbum/';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadedAlbumDeleteSelected => 'Delete Selected';
|
String get downloadedAlbumDeleteSelected => 'Apagar Selecionados';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String downloadedAlbumDeleteMessage(int count) {
|
String downloadedAlbumDeleteMessage(int count) {
|
||||||
String _temp0 = intl.Intl.pluralLogic(
|
String _temp0 = intl.Intl.pluralLogic(
|
||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: 'tracks',
|
other: 'faixas',
|
||||||
one: 'track',
|
one: 'faixa',
|
||||||
);
|
);
|
||||||
return 'Delete $count $_temp0 from this album?\n\nThis will also delete the files from storage.';
|
return 'Apagar $count $_temp0 deste álbum?\n\nIsso também apagará os arquivos do armazenamento.';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadedAlbumTracksHeader => 'Tracks';
|
String get downloadedAlbumTracksHeader => 'Faixas';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String downloadedAlbumDownloadedCount(int count) {
|
String downloadedAlbumDownloadedCount(int count) {
|
||||||
return '$count downloaded';
|
return '$count baixadas';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String downloadedAlbumSelectedCount(int count) {
|
String downloadedAlbumSelectedCount(int count) {
|
||||||
return '$count selected';
|
return '$count selecionadas';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadedAlbumAllSelected => 'All tracks selected';
|
String get downloadedAlbumAllSelected => 'Todas as faixas selecionadas';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadedAlbumTapToSelect => 'Tap tracks to select';
|
String get downloadedAlbumTapToSelect => 'Toque nas faixas para selecionar';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String downloadedAlbumDeleteCount(int count) {
|
String downloadedAlbumDeleteCount(int count) {
|
||||||
String _temp0 = intl.Intl.pluralLogic(
|
String _temp0 = intl.Intl.pluralLogic(
|
||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: 'tracks',
|
other: 'faixas',
|
||||||
one: 'track',
|
one: 'faixa',
|
||||||
);
|
);
|
||||||
return 'Delete $count $_temp0';
|
return 'Apagar $count $_temp0';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadedAlbumSelectToDelete => 'Select tracks to delete';
|
String get downloadedAlbumSelectToDelete => 'Selecione faixas para apagar';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get utilityFunctions => 'Utility Functions';
|
String get utilityFunctions => 'Funções Utilitárias';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get recentTypeArtist => 'Artist';
|
String get recentTypeArtist => 'Artista';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get recentTypeAlbum => 'Album';
|
String get recentTypeAlbum => 'Álbum';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get recentTypeSong => 'Song';
|
String get recentTypeSong => 'Música';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get recentTypePlaylist => 'Playlist';
|
String get recentTypePlaylist => 'Playlist';
|
||||||
@@ -4178,6 +4229,6 @@ class AppLocalizationsPtPt extends AppLocalizationsPt {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String errorGeneric(String message) {
|
String errorGeneric(String message) {
|
||||||
return 'Error: $message';
|
return 'Erro: $message';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -74,9 +74,9 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: '$count треков',
|
other: '$count треков',
|
||||||
|
one: '1 трек',
|
||||||
many: '$count треков',
|
many: '$count треков',
|
||||||
few: '$count трека',
|
few: '$count трека',
|
||||||
one: '$count трек',
|
|
||||||
);
|
);
|
||||||
return '$_temp0';
|
return '$_temp0';
|
||||||
}
|
}
|
||||||
@@ -87,9 +87,9 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: '$count альбомов',
|
other: '$count альбомов',
|
||||||
|
one: '1 альбом',
|
||||||
many: '$count альбомов',
|
many: '$count альбомов',
|
||||||
few: '$count альбома',
|
few: '$count альбома',
|
||||||
one: '$count альбом',
|
|
||||||
);
|
);
|
||||||
return '$_temp0';
|
return '$_temp0';
|
||||||
}
|
}
|
||||||
@@ -115,7 +115,7 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
'Здесь будут отображаться загрузки синглов';
|
'Здесь будут отображаться загрузки синглов';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get historySearchHint => 'Search history...';
|
String get historySearchHint => 'Поиск в истории...';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get settingsTitle => 'Настройки';
|
String get settingsTitle => 'Настройки';
|
||||||
@@ -418,7 +418,7 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
'Талантливый художник, который создал наш красивый логотип приложения!';
|
'Талантливый художник, который создал наш красивый логотип приложения!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutTranslators => 'Translators';
|
String get aboutTranslators => 'Переводчики';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutSpecialThanks => 'Особая благодарность';
|
String get aboutSpecialThanks => 'Особая благодарность';
|
||||||
@@ -446,19 +446,19 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
'Предложить новые функции для приложения';
|
'Предложить новые функции для приложения';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutTelegramChannel => 'Telegram Channel';
|
String get aboutTelegramChannel => 'Telegram канал';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutTelegramChannelSubtitle => 'Announcements and updates';
|
String get aboutTelegramChannelSubtitle => 'Объявления и обновления';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutTelegramChat => 'Telegram Community';
|
String get aboutTelegramChat => 'Сообщество в Telegram';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutTelegramChatSubtitle => 'Chat with other users';
|
String get aboutTelegramChatSubtitle => 'Чат с другими пользователями';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutSocial => 'Social';
|
String get aboutSocial => 'Соцсети';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutSupport => 'Поддержка';
|
String get aboutSupport => 'Поддержка';
|
||||||
@@ -483,6 +483,10 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
String get aboutSachinsenalDesc =>
|
String get aboutSachinsenalDesc =>
|
||||||
'Оригинальный создатель проекта HiFi. Основатель Tidal интеграции!';
|
'Оригинальный создатель проекта HiFi. Основатель Tidal интеграции!';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get aboutSjdonadoDesc =>
|
||||||
|
'Creator of I Don\'t Have Spotify (IDHS). The fallback link resolver that saves the day!';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get aboutDoubleDouble => 'DoubleDouble';
|
String get aboutDoubleDouble => 'DoubleDouble';
|
||||||
|
|
||||||
@@ -510,9 +514,9 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: '$count треков',
|
other: '$count треков',
|
||||||
|
one: '1 трек',
|
||||||
many: '$count треков',
|
many: '$count треков',
|
||||||
few: '$count трека',
|
few: '$count трека',
|
||||||
one: '$count трек',
|
|
||||||
);
|
);
|
||||||
return '$_temp0';
|
return '$_temp0';
|
||||||
}
|
}
|
||||||
@@ -544,9 +548,9 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: '$count релизов',
|
other: '$count релизов',
|
||||||
|
one: '1 релиз',
|
||||||
many: '$count релизов',
|
many: '$count релизов',
|
||||||
few: '$count релиза',
|
few: '$count релиза',
|
||||||
one: '$count релиз',
|
|
||||||
);
|
);
|
||||||
return '$_temp0';
|
return '$_temp0';
|
||||||
}
|
}
|
||||||
@@ -922,9 +926,9 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: 'треков',
|
other: 'треков',
|
||||||
|
one: 'трек',
|
||||||
many: 'треков',
|
many: 'треков',
|
||||||
few: 'трека',
|
few: 'трека',
|
||||||
one: 'трек',
|
|
||||||
);
|
);
|
||||||
return 'Удалить $count $_temp0 из истории?\n\nЭто также удалит файлы из хранилища.';
|
return 'Удалить $count $_temp0 из истории?\n\nЭто также удалит файлы из хранилища.';
|
||||||
}
|
}
|
||||||
@@ -939,7 +943,7 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String csvImportTracks(int count) {
|
String csvImportTracks(int count) {
|
||||||
return '$count tracks from CSV';
|
return '$count треков из CSV';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -972,9 +976,9 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: 'треков',
|
other: 'треков',
|
||||||
|
one: 'трек',
|
||||||
many: 'треков',
|
many: 'треков',
|
||||||
few: 'трека',
|
few: 'трека',
|
||||||
one: 'трек',
|
|
||||||
);
|
);
|
||||||
return 'Удалено $count $_temp0';
|
return 'Удалено $count $_temp0';
|
||||||
}
|
}
|
||||||
@@ -1121,9 +1125,9 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: 'треков',
|
other: 'треков',
|
||||||
|
one: 'трек',
|
||||||
many: 'треков',
|
many: 'треков',
|
||||||
few: 'трека',
|
few: 'трека',
|
||||||
one: 'трек',
|
|
||||||
);
|
);
|
||||||
return 'Удалить $count $_temp0';
|
return 'Удалить $count $_temp0';
|
||||||
}
|
}
|
||||||
@@ -1482,33 +1486,33 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
String get sectionFileSettings => 'Настройки файла';
|
String get sectionFileSettings => 'Настройки файла';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get sectionLyrics => 'Lyrics';
|
String get sectionLyrics => 'Тексты песен';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsMode => 'Lyrics Mode';
|
String get lyricsMode => 'Режим текстов песен';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsModeDescription =>
|
String get lyricsModeDescription =>
|
||||||
'Choose how lyrics are saved with your downloads';
|
'Выберите как сохранить тексты песен при скачивании';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsModeEmbed => 'Embed in file';
|
String get lyricsModeEmbed => 'Вставить в файл';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsModeEmbedSubtitle => 'Lyrics stored inside FLAC metadata';
|
String get lyricsModeEmbedSubtitle => 'Встроить текст в метаданные FLAC';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsModeExternal => 'External .lrc file';
|
String get lyricsModeExternal => 'Внешний файл .lrc';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsModeExternalSubtitle =>
|
String get lyricsModeExternalSubtitle =>
|
||||||
'Separate .lrc file for players like Samsung Music';
|
'Отдельный файл .lrc для плееров, таких, как Samsung Music';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsModeBoth => 'Both';
|
String get lyricsModeBoth => 'Оба варианта';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get lyricsModeBothSubtitle => 'Embed and save .lrc file';
|
String get lyricsModeBothSubtitle => 'Вставить и сохранить файл .lrc';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get sectionColor => 'Цвет';
|
String get sectionColor => 'Цвет';
|
||||||
@@ -1565,9 +1569,9 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: '$count треков',
|
other: '$count треков',
|
||||||
|
one: '1 трек',
|
||||||
many: '$count треков',
|
many: '$count треков',
|
||||||
few: '$count трека',
|
few: '$count трека',
|
||||||
one: '$count трек',
|
|
||||||
);
|
);
|
||||||
return '$_temp0';
|
return '$_temp0';
|
||||||
}
|
}
|
||||||
@@ -1627,13 +1631,13 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
String get trackReleaseDate => 'Дата выхода';
|
String get trackReleaseDate => 'Дата выхода';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackGenre => 'Genre';
|
String get trackGenre => 'Жанр';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLabel => 'Label';
|
String get trackLabel => 'Заголовок';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackCopyright => 'Copyright';
|
String get trackCopyright => 'Авторские права';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackDownloaded => 'Скачано';
|
String get trackDownloaded => 'Скачано';
|
||||||
@@ -1653,13 +1657,13 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
String get trackLyricsLoadFailed => 'Не удалось загрузить текст песни';
|
String get trackLyricsLoadFailed => 'Не удалось загрузить текст песни';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackEmbedLyrics => 'Embed Lyrics';
|
String get trackEmbedLyrics => 'Вставить текст песни';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackLyricsEmbedded => 'Lyrics embedded successfully';
|
String get trackLyricsEmbedded => 'Текст успешно добавлен';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackInstrumental => 'Instrumental track';
|
String get trackInstrumental => 'Инструментальный трек';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get trackCopiedToClipboard => 'Скопировано в буфер обмена';
|
String get trackCopiedToClipboard => 'Скопировано в буфер обмена';
|
||||||
@@ -1894,20 +1898,36 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
String get qualityHiResFlacMaxSubtitle => '24-бит / до 192кГц';
|
String get qualityHiResFlacMaxSubtitle => '24-бит / до 192кГц';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3 => 'MP3';
|
String get qualityLossy => 'Lossy';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityMp3Subtitle => '320kbps (converted from FLAC)';
|
String get qualityLossyMp3Subtitle => 'MP3 320kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3Option => 'Enable MP3 Option';
|
String get qualityLossyOpusSubtitle => 'Opus 128kbps (converted from FLAC)';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOn => 'MP3 quality option is available';
|
String get enableLossyOption => 'Enable Lossy Option';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get enableMp3OptionSubtitleOff =>
|
String get enableLossyOptionSubtitleOn => 'Lossy quality option is available';
|
||||||
'Downloads FLAC then converts to 320kbps MP3';
|
|
||||||
|
@override
|
||||||
|
String get enableLossyOptionSubtitleOff =>
|
||||||
|
'Downloads FLAC then converts to lossy format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormat => 'Lossy Format';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatDescription => 'Choose the lossy format for conversion';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatMp3Subtitle => '320kbps, best compatibility';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get lossyFormatOpusSubtitle =>
|
||||||
|
'128kbps, better quality at smaller size';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get qualityNote =>
|
String get qualityNote =>
|
||||||
@@ -2057,11 +2077,11 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
'Альбомы/[2005] Название Альбома /';
|
'Альбомы/[2005] Название Альбома /';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumFolderArtistAlbumSingles => 'Artist / Album + Singles';
|
String get albumFolderArtistAlbumSingles => 'Исполнитель / Альбом + Синглы';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get albumFolderArtistAlbumSinglesSubtitle =>
|
String get albumFolderArtistAlbumSinglesSubtitle =>
|
||||||
'Artist/Album/ and Artist/Singles/';
|
'Исполнитель/Альбом и Исполнитель/Сингл/';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get downloadedAlbumDeleteSelected => 'Удалить выбранные';
|
String get downloadedAlbumDeleteSelected => 'Удалить выбранные';
|
||||||
@@ -2072,9 +2092,9 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: 'треков',
|
other: 'треков',
|
||||||
|
one: 'трек',
|
||||||
many: 'треков',
|
many: 'треков',
|
||||||
few: 'трека',
|
few: 'трека',
|
||||||
one: 'трек',
|
|
||||||
);
|
);
|
||||||
return 'Удалить $count $_temp0 из этого альбома?\n\nЭто также удалит файлы из хранилища.';
|
return 'Удалить $count $_temp0 из этого альбома?\n\nЭто также удалит файлы из хранилища.';
|
||||||
}
|
}
|
||||||
@@ -2104,9 +2124,9 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
count,
|
count,
|
||||||
locale: localeName,
|
locale: localeName,
|
||||||
other: 'треков',
|
other: 'треков',
|
||||||
|
one: 'трек',
|
||||||
many: 'треков',
|
many: 'треков',
|
||||||
few: 'трека',
|
few: 'трека',
|
||||||
one: 'трек',
|
|
||||||
);
|
);
|
||||||
return 'Удалить $count $_temp0';
|
return 'Удалить $count $_temp0';
|
||||||
}
|
}
|
||||||
@@ -2116,7 +2136,7 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
String downloadedAlbumDiscHeader(int discNumber) {
|
String downloadedAlbumDiscHeader(int discNumber) {
|
||||||
return 'Disc $discNumber';
|
return 'Диск $discNumber';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -2145,68 +2165,93 @@ class AppLocalizationsRu extends AppLocalizations {
|
|||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyDownload => 'Download Discography';
|
String get discographyDownload => 'Скачать дискографию';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyDownloadAll => 'Download All';
|
String get discographyDownloadAll => 'Скачать всё';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographyDownloadAllSubtitle(int count, int albumCount) {
|
String discographyDownloadAllSubtitle(int count, int albumCount) {
|
||||||
return '$count tracks from $albumCount releases';
|
return '$count треков из $albumCount релизов';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyAlbumsOnly => 'Albums Only';
|
String get discographyAlbumsOnly => 'Только альбомы';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographyAlbumsOnlySubtitle(int count, int albumCount) {
|
String discographyAlbumsOnlySubtitle(int count, int albumCount) {
|
||||||
return '$count tracks from $albumCount albums';
|
return '$count треков из $albumCount альбомов';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographySinglesOnly => 'Singles & EPs Only';
|
String get discographySinglesOnly => 'Только синглы и EP';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographySinglesOnlySubtitle(int count, int albumCount) {
|
String discographySinglesOnlySubtitle(int count, int albumCount) {
|
||||||
return '$count tracks from $albumCount singles';
|
return '$count треков из $albumCount синглов';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographySelectAlbums => 'Select Albums...';
|
String get discographySelectAlbums => 'Выбрать альбомы...';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographySelectAlbumsSubtitle =>
|
String get discographySelectAlbumsSubtitle =>
|
||||||
'Choose specific albums or singles';
|
'Выберите конкретные альбомы или синглы';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFetchingTracks => 'Fetching tracks...';
|
String get discographyFetchingTracks => 'Получение треков...';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographyFetchingAlbum(int current, int total) {
|
String discographyFetchingAlbum(int current, int total) {
|
||||||
return 'Fetching $current of $total...';
|
return 'Получение $current из $total...';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographySelectedCount(int count) {
|
String discographySelectedCount(int count) {
|
||||||
return '$count selected';
|
return '$count выбрано';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyDownloadSelected => 'Download Selected';
|
String get discographyDownloadSelected => 'Скачать выбранное';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographyAddedToQueue(int count) {
|
String discographyAddedToQueue(int count) {
|
||||||
return 'Added $count tracks to queue';
|
return 'Добавлено $count треков в очередь';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String discographySkippedDownloaded(int added, int skipped) {
|
String discographySkippedDownloaded(int added, int skipped) {
|
||||||
return '$added added, $skipped already downloaded';
|
return '$added добавлено, $skipped уже скачано';
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyNoAlbums => 'No albums available';
|
String get discographyNoAlbums => 'Нет доступных альбомов';
|
||||||
|
|
||||||
@override
|
@override
|
||||||
String get discographyFailedToFetch => 'Failed to fetch some albums';
|
String get discographyFailedToFetch =>
|
||||||
|
'Не удалось получить некоторые альбомы';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get sectionStorageAccess => 'Storage Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccess => 'All Files Access';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessEnabledSubtitle => 'Can write to any folder';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledSubtitle => 'Limited to media folders only';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDescription =>
|
||||||
|
'Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDeniedMessage =>
|
||||||
|
'Permission was denied. Please enable \'All files access\' manually in system settings.';
|
||||||
|
|
||||||
|
@override
|
||||||
|
String get allFilesAccessDisabledMessage =>
|
||||||
|
'All Files Access disabled. The app will use limited storage access.';
|
||||||
}
|
}
|
||||||
|
|||||||
+598
-539
File diff suppressed because it is too large
Load Diff
+4342
-3996
File diff suppressed because it is too large
Load Diff
+321
-68
@@ -127,6 +127,10 @@
|
|||||||
"@historyNoSinglesSubtitle": {
|
"@historyNoSinglesSubtitle": {
|
||||||
"description": "Empty state subtitle for singles filter"
|
"description": "Empty state subtitle for singles filter"
|
||||||
},
|
},
|
||||||
|
"historySearchHint": "Suchverlauf...",
|
||||||
|
"@historySearchHint": {
|
||||||
|
"description": "Search bar placeholder in history"
|
||||||
|
},
|
||||||
"settingsTitle": "Einstellungen",
|
"settingsTitle": "Einstellungen",
|
||||||
"@settingsTitle": {
|
"@settingsTitle": {
|
||||||
"description": "Settings screen title"
|
"description": "Settings screen title"
|
||||||
@@ -512,6 +516,10 @@
|
|||||||
"@aboutLogoArtist": {
|
"@aboutLogoArtist": {
|
||||||
"description": "Role description for logo artist"
|
"description": "Role description for logo artist"
|
||||||
},
|
},
|
||||||
|
"aboutTranslators": "Übersetzer",
|
||||||
|
"@aboutTranslators": {
|
||||||
|
"description": "Section for translators"
|
||||||
|
},
|
||||||
"aboutSpecialThanks": "Besonderer Dank",
|
"aboutSpecialThanks": "Besonderer Dank",
|
||||||
"@aboutSpecialThanks": {
|
"@aboutSpecialThanks": {
|
||||||
"description": "Section for special thanks"
|
"description": "Section for special thanks"
|
||||||
@@ -544,6 +552,26 @@
|
|||||||
"@aboutFeatureRequestSubtitle": {
|
"@aboutFeatureRequestSubtitle": {
|
||||||
"description": "Subtitle for feature request"
|
"description": "Subtitle for feature request"
|
||||||
},
|
},
|
||||||
|
"aboutTelegramChannel": "Telegram Kanal",
|
||||||
|
"@aboutTelegramChannel": {
|
||||||
|
"description": "Link to Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChannelSubtitle": "Ankündigungen und Updates",
|
||||||
|
"@aboutTelegramChannelSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChat": "Telegram Community",
|
||||||
|
"@aboutTelegramChat": {
|
||||||
|
"description": "Link to Telegram chat group"
|
||||||
|
},
|
||||||
|
"aboutTelegramChatSubtitle": "Mit anderen Nutzern chatten",
|
||||||
|
"@aboutTelegramChatSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram chat"
|
||||||
|
},
|
||||||
|
"aboutSocial": "Sozial",
|
||||||
|
"@aboutSocial": {
|
||||||
|
"description": "Section for social links"
|
||||||
|
},
|
||||||
"aboutSupport": "Support",
|
"aboutSupport": "Support",
|
||||||
"@aboutSupport": {
|
"@aboutSupport": {
|
||||||
"description": "Section for support/donation links"
|
"description": "Section for support/donation links"
|
||||||
@@ -588,7 +616,7 @@
|
|||||||
"@aboutDabMusicDesc": {
|
"@aboutDabMusicDesc": {
|
||||||
"description": "Credit for DAB Music API"
|
"description": "Credit for DAB Music API"
|
||||||
},
|
},
|
||||||
"aboutAppDescription": "Download Spotify tracks in lossless quality from Tidal, Qobuz, and Amazon Music.",
|
"aboutAppDescription": "Lade Spotify-Titel in verlustfreier Qualität von Tidal, Qobuz und Amazon Music herunter.",
|
||||||
"@aboutAppDescription": {
|
"@aboutAppDescription": {
|
||||||
"description": "App description in header card"
|
"description": "App description in header card"
|
||||||
},
|
},
|
||||||
@@ -596,7 +624,7 @@
|
|||||||
"@albumTitle": {
|
"@albumTitle": {
|
||||||
"description": "Album screen title"
|
"description": "Album screen title"
|
||||||
},
|
},
|
||||||
"albumTracks": "{count, plural, =1{1 track} other{{count} tracks}}",
|
"albumTracks": "{count, plural,=1{1 Song} other{{count} Songs}}",
|
||||||
"@albumTracks": {
|
"@albumTracks": {
|
||||||
"description": "Album track count",
|
"description": "Album track count",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -605,11 +633,11 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"albumDownloadAll": "Download All",
|
"albumDownloadAll": "Alle Herunterladen",
|
||||||
"@albumDownloadAll": {
|
"@albumDownloadAll": {
|
||||||
"description": "Button to download all tracks"
|
"description": "Button to download all tracks"
|
||||||
},
|
},
|
||||||
"albumDownloadRemaining": "Download Remaining",
|
"albumDownloadRemaining": "Downloads verbleibend",
|
||||||
"@albumDownloadRemaining": {
|
"@albumDownloadRemaining": {
|
||||||
"description": "Button to download remaining tracks"
|
"description": "Button to download remaining tracks"
|
||||||
},
|
},
|
||||||
@@ -617,11 +645,11 @@
|
|||||||
"@playlistTitle": {
|
"@playlistTitle": {
|
||||||
"description": "Playlist screen title"
|
"description": "Playlist screen title"
|
||||||
},
|
},
|
||||||
"artistTitle": "Artist",
|
"artistTitle": "Künstler",
|
||||||
"@artistTitle": {
|
"@artistTitle": {
|
||||||
"description": "Artist screen title"
|
"description": "Artist screen title"
|
||||||
},
|
},
|
||||||
"artistAlbums": "Albums",
|
"artistAlbums": "Alben",
|
||||||
"@artistAlbums": {
|
"@artistAlbums": {
|
||||||
"description": "Section header for artist albums"
|
"description": "Section header for artist albums"
|
||||||
},
|
},
|
||||||
@@ -629,11 +657,11 @@
|
|||||||
"@artistSingles": {
|
"@artistSingles": {
|
||||||
"description": "Section header for singles/EPs"
|
"description": "Section header for singles/EPs"
|
||||||
},
|
},
|
||||||
"artistCompilations": "Compilations",
|
"artistCompilations": "Zusammenstellungen",
|
||||||
"@artistCompilations": {
|
"@artistCompilations": {
|
||||||
"description": "Section header for compilations"
|
"description": "Section header for compilations"
|
||||||
},
|
},
|
||||||
"artistReleases": "{count, plural, =1{1 release} other{{count} releases}}",
|
"artistReleases": "{count, plural,=1{1 Veröffentlichung} other{{count} Veröffentlichungen}}",
|
||||||
"@artistReleases": {
|
"@artistReleases": {
|
||||||
"description": "Artist release count",
|
"description": "Artist release count",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -642,11 +670,11 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"artistPopular": "Popular",
|
"artistPopular": "Beliebt",
|
||||||
"@artistPopular": {
|
"@artistPopular": {
|
||||||
"description": "Section header for popular/top tracks"
|
"description": "Section header for popular/top tracks"
|
||||||
},
|
},
|
||||||
"artistMonthlyListeners": "{count} monthly listeners",
|
"artistMonthlyListeners": "{count} monatliche Hörer",
|
||||||
"@artistMonthlyListeners": {
|
"@artistMonthlyListeners": {
|
||||||
"description": "Monthly listener count display",
|
"description": "Monthly listener count display",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -656,11 +684,11 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"trackMetadataTitle": "Track Info",
|
"trackMetadataTitle": "Titel Info",
|
||||||
"@trackMetadataTitle": {
|
"@trackMetadataTitle": {
|
||||||
"description": "Track metadata screen title"
|
"description": "Track metadata screen title"
|
||||||
},
|
},
|
||||||
"trackMetadataArtist": "Artist",
|
"trackMetadataArtist": "Künstler",
|
||||||
"@trackMetadataArtist": {
|
"@trackMetadataArtist": {
|
||||||
"description": "Metadata field - artist name"
|
"description": "Metadata field - artist name"
|
||||||
},
|
},
|
||||||
@@ -668,111 +696,111 @@
|
|||||||
"@trackMetadataAlbum": {
|
"@trackMetadataAlbum": {
|
||||||
"description": "Metadata field - album name"
|
"description": "Metadata field - album name"
|
||||||
},
|
},
|
||||||
"trackMetadataDuration": "Duration",
|
"trackMetadataDuration": "Länge",
|
||||||
"@trackMetadataDuration": {
|
"@trackMetadataDuration": {
|
||||||
"description": "Metadata field - track length"
|
"description": "Metadata field - track length"
|
||||||
},
|
},
|
||||||
"trackMetadataQuality": "Quality",
|
"trackMetadataQuality": "Qualität",
|
||||||
"@trackMetadataQuality": {
|
"@trackMetadataQuality": {
|
||||||
"description": "Metadata field - audio quality"
|
"description": "Metadata field - audio quality"
|
||||||
},
|
},
|
||||||
"trackMetadataPath": "File Path",
|
"trackMetadataPath": "Dateipfad",
|
||||||
"@trackMetadataPath": {
|
"@trackMetadataPath": {
|
||||||
"description": "Metadata field - file location"
|
"description": "Metadata field - file location"
|
||||||
},
|
},
|
||||||
"trackMetadataDownloadedAt": "Downloaded",
|
"trackMetadataDownloadedAt": "Heruntergeladen",
|
||||||
"@trackMetadataDownloadedAt": {
|
"@trackMetadataDownloadedAt": {
|
||||||
"description": "Metadata field - download date"
|
"description": "Metadata field - download date"
|
||||||
},
|
},
|
||||||
"trackMetadataService": "Service",
|
"trackMetadataService": "Anbieter",
|
||||||
"@trackMetadataService": {
|
"@trackMetadataService": {
|
||||||
"description": "Metadata field - download service used"
|
"description": "Metadata field - download service used"
|
||||||
},
|
},
|
||||||
"trackMetadataPlay": "Play",
|
"trackMetadataPlay": "Abspielen",
|
||||||
"@trackMetadataPlay": {
|
"@trackMetadataPlay": {
|
||||||
"description": "Action button - play track"
|
"description": "Action button - play track"
|
||||||
},
|
},
|
||||||
"trackMetadataShare": "Share",
|
"trackMetadataShare": "Teilen",
|
||||||
"@trackMetadataShare": {
|
"@trackMetadataShare": {
|
||||||
"description": "Action button - share track"
|
"description": "Action button - share track"
|
||||||
},
|
},
|
||||||
"trackMetadataDelete": "Delete",
|
"trackMetadataDelete": "Löschen",
|
||||||
"@trackMetadataDelete": {
|
"@trackMetadataDelete": {
|
||||||
"description": "Action button - delete track"
|
"description": "Action button - delete track"
|
||||||
},
|
},
|
||||||
"trackMetadataRedownload": "Re-download",
|
"trackMetadataRedownload": "Erneut herunterladen",
|
||||||
"@trackMetadataRedownload": {
|
"@trackMetadataRedownload": {
|
||||||
"description": "Action button - download again"
|
"description": "Action button - download again"
|
||||||
},
|
},
|
||||||
"trackMetadataOpenFolder": "Open Folder",
|
"trackMetadataOpenFolder": "Ordner öffnen",
|
||||||
"@trackMetadataOpenFolder": {
|
"@trackMetadataOpenFolder": {
|
||||||
"description": "Action button - open containing folder"
|
"description": "Action button - open containing folder"
|
||||||
},
|
},
|
||||||
"setupTitle": "Welcome to SpotiFLAC",
|
"setupTitle": "Willkommen bei SpotiFLAC",
|
||||||
"@setupTitle": {
|
"@setupTitle": {
|
||||||
"description": "Setup wizard title"
|
"description": "Setup wizard title"
|
||||||
},
|
},
|
||||||
"setupSubtitle": "Let's get you started",
|
"setupSubtitle": "Los geht's",
|
||||||
"@setupSubtitle": {
|
"@setupSubtitle": {
|
||||||
"description": "Setup wizard subtitle"
|
"description": "Setup wizard subtitle"
|
||||||
},
|
},
|
||||||
"setupStoragePermission": "Storage Permission",
|
"setupStoragePermission": "Speicherberechtigung",
|
||||||
"@setupStoragePermission": {
|
"@setupStoragePermission": {
|
||||||
"description": "Storage permission step title"
|
"description": "Storage permission step title"
|
||||||
},
|
},
|
||||||
"setupStoragePermissionSubtitle": "Required to save downloaded files",
|
"setupStoragePermissionSubtitle": "Benötigt um heruntergeladene Dateien zu Speichern",
|
||||||
"@setupStoragePermissionSubtitle": {
|
"@setupStoragePermissionSubtitle": {
|
||||||
"description": "Explanation for storage permission"
|
"description": "Explanation for storage permission"
|
||||||
},
|
},
|
||||||
"setupStoragePermissionGranted": "Permission granted",
|
"setupStoragePermissionGranted": "Berechtigung erteilt",
|
||||||
"@setupStoragePermissionGranted": {
|
"@setupStoragePermissionGranted": {
|
||||||
"description": "Status when permission granted"
|
"description": "Status when permission granted"
|
||||||
},
|
},
|
||||||
"setupStoragePermissionDenied": "Permission denied",
|
"setupStoragePermissionDenied": "Berechtigung verweigert",
|
||||||
"@setupStoragePermissionDenied": {
|
"@setupStoragePermissionDenied": {
|
||||||
"description": "Status when permission denied"
|
"description": "Status when permission denied"
|
||||||
},
|
},
|
||||||
"setupGrantPermission": "Grant Permission",
|
"setupGrantPermission": "Berechtigung erlauben",
|
||||||
"@setupGrantPermission": {
|
"@setupGrantPermission": {
|
||||||
"description": "Button to request permission"
|
"description": "Button to request permission"
|
||||||
},
|
},
|
||||||
"setupDownloadLocation": "Download Location",
|
"setupDownloadLocation": "Speicherort",
|
||||||
"@setupDownloadLocation": {
|
"@setupDownloadLocation": {
|
||||||
"description": "Download folder step title"
|
"description": "Download folder step title"
|
||||||
},
|
},
|
||||||
"setupChooseFolder": "Choose Folder",
|
"setupChooseFolder": "Ordner wählen",
|
||||||
"@setupChooseFolder": {
|
"@setupChooseFolder": {
|
||||||
"description": "Button to pick folder"
|
"description": "Button to pick folder"
|
||||||
},
|
},
|
||||||
"setupContinue": "Continue",
|
"setupContinue": "Fortfahren",
|
||||||
"@setupContinue": {
|
"@setupContinue": {
|
||||||
"description": "Continue to next step button"
|
"description": "Continue to next step button"
|
||||||
},
|
},
|
||||||
"setupSkip": "Skip for now",
|
"setupSkip": "Vorerst überspringen",
|
||||||
"@setupSkip": {
|
"@setupSkip": {
|
||||||
"description": "Skip current step button"
|
"description": "Skip current step button"
|
||||||
},
|
},
|
||||||
"setupStorageAccessRequired": "Storage Access Required",
|
"setupStorageAccessRequired": "Speicherzugriff erforderlich",
|
||||||
"@setupStorageAccessRequired": {
|
"@setupStorageAccessRequired": {
|
||||||
"description": "Title when storage access needed"
|
"description": "Title when storage access needed"
|
||||||
},
|
},
|
||||||
"setupStorageAccessMessage": "SpotiFLAC needs \"All files access\" permission to save music files to your chosen folder.",
|
"setupStorageAccessMessage": "SpotiFLAC benötigt die Berechtigung \"Auf alle Dateien zugreifen\", um Musikdateien in deinen gewählten Ordner zu speichern.",
|
||||||
"@setupStorageAccessMessage": {
|
"@setupStorageAccessMessage": {
|
||||||
"description": "Explanation for storage access"
|
"description": "Explanation for storage access"
|
||||||
},
|
},
|
||||||
"setupStorageAccessMessageAndroid11": "Android 11+ requires \"All files access\" permission to save files to your chosen download folder.",
|
"setupStorageAccessMessageAndroid11": "Android 11+ benötigt die Berechtigung „Auf alle Dateien“, um Dateien im ausgewählten Download-Ordner zu speichern.",
|
||||||
"@setupStorageAccessMessageAndroid11": {
|
"@setupStorageAccessMessageAndroid11": {
|
||||||
"description": "Android 11+ specific explanation"
|
"description": "Android 11+ specific explanation"
|
||||||
},
|
},
|
||||||
"setupOpenSettings": "Open Settings",
|
"setupOpenSettings": "Einstellungen öffnen",
|
||||||
"@setupOpenSettings": {
|
"@setupOpenSettings": {
|
||||||
"description": "Button to open system settings"
|
"description": "Button to open system settings"
|
||||||
},
|
},
|
||||||
"setupPermissionDeniedMessage": "Permission denied. Please grant all permissions to continue.",
|
"setupPermissionDeniedMessage": "Berechtigung verweigert. Bitte erteilen Sie alle Berechtigungen um fortzufahren.",
|
||||||
"@setupPermissionDeniedMessage": {
|
"@setupPermissionDeniedMessage": {
|
||||||
"description": "Error when permission denied"
|
"description": "Error when permission denied"
|
||||||
},
|
},
|
||||||
"setupPermissionRequired": "{permissionType} Permission Required",
|
"setupPermissionRequired": "{permissionType} Zugriff verweigert",
|
||||||
"@setupPermissionRequired": {
|
"@setupPermissionRequired": {
|
||||||
"description": "Generic permission required title",
|
"description": "Generic permission required title",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -782,7 +810,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"setupPermissionRequiredMessage": "{permissionType} permission is required for the best experience. You can change this later in Settings.",
|
"setupPermissionRequiredMessage": "{permissionType} Berechtigung ist erforderlich für\ndie beste Benutzererfahrung. Für kannst dies später in den Einstellungen ändern.",
|
||||||
"@setupPermissionRequiredMessage": {
|
"@setupPermissionRequiredMessage": {
|
||||||
"description": "Generic permission required message",
|
"description": "Generic permission required message",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -791,63 +819,63 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"setupSelectDownloadFolder": "Select Download Folder",
|
"setupSelectDownloadFolder": "Wähle Download-Ordner aus",
|
||||||
"@setupSelectDownloadFolder": {
|
"@setupSelectDownloadFolder": {
|
||||||
"description": "Folder selection step title"
|
"description": "Folder selection step title"
|
||||||
},
|
},
|
||||||
"setupUseDefaultFolder": "Use Default Folder?",
|
"setupUseDefaultFolder": "Als Standardordner verwenden?",
|
||||||
"@setupUseDefaultFolder": {
|
"@setupUseDefaultFolder": {
|
||||||
"description": "Dialog title for default folder"
|
"description": "Dialog title for default folder"
|
||||||
},
|
},
|
||||||
"setupNoFolderSelected": "No folder selected. Would you like to use the default Music folder?",
|
"setupNoFolderSelected": "Kein Ordner ausgewählt. Soll der Standard-Musikordner verwendet werden?",
|
||||||
"@setupNoFolderSelected": {
|
"@setupNoFolderSelected": {
|
||||||
"description": "Prompt when no folder selected"
|
"description": "Prompt when no folder selected"
|
||||||
},
|
},
|
||||||
"setupUseDefault": "Use Default",
|
"setupUseDefault": "Standart benutzen",
|
||||||
"@setupUseDefault": {
|
"@setupUseDefault": {
|
||||||
"description": "Button to use default folder"
|
"description": "Button to use default folder"
|
||||||
},
|
},
|
||||||
"setupDownloadLocationTitle": "Download Location",
|
"setupDownloadLocationTitle": "Speicherort",
|
||||||
"@setupDownloadLocationTitle": {
|
"@setupDownloadLocationTitle": {
|
||||||
"description": "Download location dialog title"
|
"description": "Download location dialog title"
|
||||||
},
|
},
|
||||||
"setupDownloadLocationIosMessage": "On iOS, downloads are saved to the app's Documents folder. You can access them via the Files app.",
|
"setupDownloadLocationIosMessage": "Auf iOS werden Downloads im Dokumentenverzeichnis der App gespeichert. Sie können sie über die Datei-App aufrufen.",
|
||||||
"@setupDownloadLocationIosMessage": {
|
"@setupDownloadLocationIosMessage": {
|
||||||
"description": "iOS-specific folder info"
|
"description": "iOS-specific folder info"
|
||||||
},
|
},
|
||||||
"setupAppDocumentsFolder": "App Documents Folder",
|
"setupAppDocumentsFolder": "App-Dokumentenordner",
|
||||||
"@setupAppDocumentsFolder": {
|
"@setupAppDocumentsFolder": {
|
||||||
"description": "iOS documents folder option"
|
"description": "iOS documents folder option"
|
||||||
},
|
},
|
||||||
"setupAppDocumentsFolderSubtitle": "Recommended - accessible via Files app",
|
"setupAppDocumentsFolderSubtitle": "Empfohlen - zugänglich über die Datei-App",
|
||||||
"@setupAppDocumentsFolderSubtitle": {
|
"@setupAppDocumentsFolderSubtitle": {
|
||||||
"description": "Subtitle for documents folder"
|
"description": "Subtitle for documents folder"
|
||||||
},
|
},
|
||||||
"setupChooseFromFiles": "Choose from Files",
|
"setupChooseFromFiles": "Aus Dateien auswählen",
|
||||||
"@setupChooseFromFiles": {
|
"@setupChooseFromFiles": {
|
||||||
"description": "iOS file picker option"
|
"description": "iOS file picker option"
|
||||||
},
|
},
|
||||||
"setupChooseFromFilesSubtitle": "Select iCloud or other location",
|
"setupChooseFromFilesSubtitle": "Wählen Sie iCloud oder einen anderen Ort",
|
||||||
"@setupChooseFromFilesSubtitle": {
|
"@setupChooseFromFilesSubtitle": {
|
||||||
"description": "Subtitle for file picker"
|
"description": "Subtitle for file picker"
|
||||||
},
|
},
|
||||||
"setupIosEmptyFolderWarning": "iOS limitation: Empty folders cannot be selected. Choose a folder with at least one file.",
|
"setupIosEmptyFolderWarning": "iOS-Einschränkung: Leere Ordner können nicht ausgewählt werden. Wählen Sie einen Ordner mit mindestens einer Datei.",
|
||||||
"@setupIosEmptyFolderWarning": {
|
"@setupIosEmptyFolderWarning": {
|
||||||
"description": "iOS folder selection warning"
|
"description": "iOS folder selection warning"
|
||||||
},
|
},
|
||||||
"setupDownloadInFlac": "Download Spotify tracks in FLAC",
|
"setupDownloadInFlac": "Spotify Titel in FLAC herunterladen",
|
||||||
"@setupDownloadInFlac": {
|
"@setupDownloadInFlac": {
|
||||||
"description": "App tagline in setup"
|
"description": "App tagline in setup"
|
||||||
},
|
},
|
||||||
"setupStepStorage": "Storage",
|
"setupStepStorage": "Speicherort",
|
||||||
"@setupStepStorage": {
|
"@setupStepStorage": {
|
||||||
"description": "Setup step indicator - storage"
|
"description": "Setup step indicator - storage"
|
||||||
},
|
},
|
||||||
"setupStepNotification": "Notification",
|
"setupStepNotification": "Benachrichtigung",
|
||||||
"@setupStepNotification": {
|
"@setupStepNotification": {
|
||||||
"description": "Setup step indicator - notification"
|
"description": "Setup step indicator - notification"
|
||||||
},
|
},
|
||||||
"setupStepFolder": "Folder",
|
"setupStepFolder": "Ordner",
|
||||||
"@setupStepFolder": {
|
"@setupStepFolder": {
|
||||||
"description": "Setup step indicator - folder"
|
"description": "Setup step indicator - folder"
|
||||||
},
|
},
|
||||||
@@ -855,55 +883,55 @@
|
|||||||
"@setupStepSpotify": {
|
"@setupStepSpotify": {
|
||||||
"description": "Setup step indicator - Spotify API"
|
"description": "Setup step indicator - Spotify API"
|
||||||
},
|
},
|
||||||
"setupStepPermission": "Permission",
|
"setupStepPermission": "Berechtigung",
|
||||||
"@setupStepPermission": {
|
"@setupStepPermission": {
|
||||||
"description": "Setup step indicator - permission"
|
"description": "Setup step indicator - permission"
|
||||||
},
|
},
|
||||||
"setupStorageGranted": "Storage Permission Granted!",
|
"setupStorageGranted": "Speicherberechtigung erlaubt!",
|
||||||
"@setupStorageGranted": {
|
"@setupStorageGranted": {
|
||||||
"description": "Success message for storage permission"
|
"description": "Success message for storage permission"
|
||||||
},
|
},
|
||||||
"setupStorageRequired": "Storage Permission Required",
|
"setupStorageRequired": "Speicherzugriff erforderlich",
|
||||||
"@setupStorageRequired": {
|
"@setupStorageRequired": {
|
||||||
"description": "Title when storage permission needed"
|
"description": "Title when storage permission needed"
|
||||||
},
|
},
|
||||||
"setupStorageDescription": "SpotiFLAC needs storage permission to save your downloaded music files.",
|
"setupStorageDescription": "SpotiFLAC benötigt Speicherrechte, um die heruntergeladenen Musikdateien zu speichern.",
|
||||||
"@setupStorageDescription": {
|
"@setupStorageDescription": {
|
||||||
"description": "Explanation for storage permission"
|
"description": "Explanation for storage permission"
|
||||||
},
|
},
|
||||||
"setupNotificationGranted": "Notification Permission Granted!",
|
"setupNotificationGranted": "Benachrichtigungs-Berechtigung erteilt",
|
||||||
"@setupNotificationGranted": {
|
"@setupNotificationGranted": {
|
||||||
"description": "Success message for notification permission"
|
"description": "Success message for notification permission"
|
||||||
},
|
},
|
||||||
"setupNotificationEnable": "Enable Notifications",
|
"setupNotificationEnable": "Benachrichtigungen aktivieren",
|
||||||
"@setupNotificationEnable": {
|
"@setupNotificationEnable": {
|
||||||
"description": "Button to enable notifications"
|
"description": "Button to enable notifications"
|
||||||
},
|
},
|
||||||
"setupNotificationDescription": "Get notified when downloads complete or require attention.",
|
"setupNotificationDescription": "Benachrichtigt werden, wenn Downloads abgeschlossen sind.",
|
||||||
"@setupNotificationDescription": {
|
"@setupNotificationDescription": {
|
||||||
"description": "Explanation for notifications"
|
"description": "Explanation for notifications"
|
||||||
},
|
},
|
||||||
"setupFolderSelected": "Download Folder Selected!",
|
"setupFolderSelected": "Download Ordner ausgewählt!",
|
||||||
"@setupFolderSelected": {
|
"@setupFolderSelected": {
|
||||||
"description": "Success message for folder selection"
|
"description": "Success message for folder selection"
|
||||||
},
|
},
|
||||||
"setupFolderChoose": "Choose Download Folder",
|
"setupFolderChoose": "Speicherort auwählen",
|
||||||
"@setupFolderChoose": {
|
"@setupFolderChoose": {
|
||||||
"description": "Button to choose folder"
|
"description": "Button to choose folder"
|
||||||
},
|
},
|
||||||
"setupFolderDescription": "Select a folder where your downloaded music will be saved.",
|
"setupFolderDescription": "Wählen Sie einen Ordner, in dem Ihre heruntergeladene Musik gespeichert wird.",
|
||||||
"@setupFolderDescription": {
|
"@setupFolderDescription": {
|
||||||
"description": "Explanation for folder selection"
|
"description": "Explanation for folder selection"
|
||||||
},
|
},
|
||||||
"setupChangeFolder": "Change Folder",
|
"setupChangeFolder": "Ordner ändern",
|
||||||
"@setupChangeFolder": {
|
"@setupChangeFolder": {
|
||||||
"description": "Button to change selected folder"
|
"description": "Button to change selected folder"
|
||||||
},
|
},
|
||||||
"setupSelectFolder": "Select Folder",
|
"setupSelectFolder": "Ordner wählen",
|
||||||
"@setupSelectFolder": {
|
"@setupSelectFolder": {
|
||||||
"description": "Button to select folder"
|
"description": "Button to select folder"
|
||||||
},
|
},
|
||||||
"setupSpotifyApiOptional": "Spotify API (Optional)",
|
"setupSpotifyApiOptional": "Spotify-API (optional)",
|
||||||
"@setupSpotifyApiOptional": {
|
"@setupSpotifyApiOptional": {
|
||||||
"description": "Spotify API step title"
|
"description": "Spotify API step title"
|
||||||
},
|
},
|
||||||
@@ -1122,6 +1150,15 @@
|
|||||||
"description": "Dialog title - import CSV playlist"
|
"description": "Dialog title - import CSV playlist"
|
||||||
},
|
},
|
||||||
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
||||||
|
"csvImportTracks": "{count} tracks from CSV",
|
||||||
|
"@csvImportTracks": {
|
||||||
|
"description": "Label shown in quality picker for CSV import",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"@dialogImportPlaylistMessage": {
|
"@dialogImportPlaylistMessage": {
|
||||||
"description": "Dialog message - import playlist confirmation",
|
"description": "Dialog message - import playlist confirmation",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1851,6 +1888,42 @@
|
|||||||
"@sectionFileSettings": {
|
"@sectionFileSettings": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
},
|
},
|
||||||
|
"sectionLyrics": "Lyrics",
|
||||||
|
"@sectionLyrics": {
|
||||||
|
"description": "Settings section header"
|
||||||
|
},
|
||||||
|
"lyricsMode": "Lyrics Mode",
|
||||||
|
"@lyricsMode": {
|
||||||
|
"description": "Setting - how to save lyrics"
|
||||||
|
},
|
||||||
|
"lyricsModeDescription": "Choose how lyrics are saved with your downloads",
|
||||||
|
"@lyricsModeDescription": {
|
||||||
|
"description": "Lyrics mode picker description"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbed": "Embed in file",
|
||||||
|
"@lyricsModeEmbed": {
|
||||||
|
"description": "Lyrics mode option - embed in audio file"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbedSubtitle": "Lyrics stored inside FLAC metadata",
|
||||||
|
"@lyricsModeEmbedSubtitle": {
|
||||||
|
"description": "Subtitle for embed option"
|
||||||
|
},
|
||||||
|
"lyricsModeExternal": "External .lrc file",
|
||||||
|
"@lyricsModeExternal": {
|
||||||
|
"description": "Lyrics mode option - separate LRC file"
|
||||||
|
},
|
||||||
|
"lyricsModeExternalSubtitle": "Separate .lrc file for players like Samsung Music",
|
||||||
|
"@lyricsModeExternalSubtitle": {
|
||||||
|
"description": "Subtitle for external option"
|
||||||
|
},
|
||||||
|
"lyricsModeBoth": "Both",
|
||||||
|
"@lyricsModeBoth": {
|
||||||
|
"description": "Lyrics mode option - embed and external"
|
||||||
|
},
|
||||||
|
"lyricsModeBothSubtitle": "Embed and save .lrc file",
|
||||||
|
"@lyricsModeBothSubtitle": {
|
||||||
|
"description": "Subtitle for both option"
|
||||||
|
},
|
||||||
"sectionColor": "Color",
|
"sectionColor": "Color",
|
||||||
"@sectionColor": {
|
"@sectionColor": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
@@ -1997,6 +2070,18 @@
|
|||||||
"@trackReleaseDate": {
|
"@trackReleaseDate": {
|
||||||
"description": "Metadata label - release date"
|
"description": "Metadata label - release date"
|
||||||
},
|
},
|
||||||
|
"trackGenre": "Genre",
|
||||||
|
"@trackGenre": {
|
||||||
|
"description": "Metadata label - music genre"
|
||||||
|
},
|
||||||
|
"trackLabel": "Label",
|
||||||
|
"@trackLabel": {
|
||||||
|
"description": "Metadata label - record label"
|
||||||
|
},
|
||||||
|
"trackCopyright": "Copyright",
|
||||||
|
"@trackCopyright": {
|
||||||
|
"description": "Metadata label - copyright information"
|
||||||
|
},
|
||||||
"trackDownloaded": "Downloaded",
|
"trackDownloaded": "Downloaded",
|
||||||
"@trackDownloaded": {
|
"@trackDownloaded": {
|
||||||
"description": "Metadata label - download date"
|
"description": "Metadata label - download date"
|
||||||
@@ -2017,6 +2102,18 @@
|
|||||||
"@trackLyricsLoadFailed": {
|
"@trackLyricsLoadFailed": {
|
||||||
"description": "Message when lyrics loading fails"
|
"description": "Message when lyrics loading fails"
|
||||||
},
|
},
|
||||||
|
"trackEmbedLyrics": "Embed Lyrics",
|
||||||
|
"@trackEmbedLyrics": {
|
||||||
|
"description": "Action - embed lyrics into audio file"
|
||||||
|
},
|
||||||
|
"trackLyricsEmbedded": "Lyrics embedded successfully",
|
||||||
|
"@trackLyricsEmbedded": {
|
||||||
|
"description": "Snackbar - lyrics saved to file"
|
||||||
|
},
|
||||||
|
"trackInstrumental": "Instrumental track",
|
||||||
|
"@trackInstrumental": {
|
||||||
|
"description": "Message when track is instrumental (no lyrics)"
|
||||||
|
},
|
||||||
"trackCopiedToClipboard": "Copied to clipboard",
|
"trackCopiedToClipboard": "Copied to clipboard",
|
||||||
"@trackCopiedToClipboard": {
|
"@trackCopiedToClipboard": {
|
||||||
"description": "Snackbar - content copied"
|
"description": "Snackbar - content copied"
|
||||||
@@ -2328,6 +2425,26 @@
|
|||||||
"@qualityHiResFlacMaxSubtitle": {
|
"@qualityHiResFlacMaxSubtitle": {
|
||||||
"description": "Technical spec for hi-res max"
|
"description": "Technical spec for hi-res max"
|
||||||
},
|
},
|
||||||
|
"qualityMp3": "MP3",
|
||||||
|
"@qualityMp3": {
|
||||||
|
"description": "Quality option - MP3 lossy format"
|
||||||
|
},
|
||||||
|
"qualityMp3Subtitle": "320kbps (converted from FLAC)",
|
||||||
|
"@qualityMp3Subtitle": {
|
||||||
|
"description": "Technical spec for MP3"
|
||||||
|
},
|
||||||
|
"enableMp3Option": "Enable MP3 Option",
|
||||||
|
"@enableMp3Option": {
|
||||||
|
"description": "Setting - enable MP3 quality option"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOn": "MP3 quality option is available",
|
||||||
|
"@enableMp3OptionSubtitleOn": {
|
||||||
|
"description": "Subtitle when MP3 is enabled"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOff": "Downloads FLAC then converts to 320kbps MP3",
|
||||||
|
"@enableMp3OptionSubtitleOff": {
|
||||||
|
"description": "Subtitle when MP3 is disabled"
|
||||||
|
},
|
||||||
"qualityNote": "Actual quality depends on track availability from the service",
|
"qualityNote": "Actual quality depends on track availability from the service",
|
||||||
"@qualityNote": {
|
"@qualityNote": {
|
||||||
"description": "Note about quality availability"
|
"description": "Note about quality availability"
|
||||||
@@ -2516,6 +2633,14 @@
|
|||||||
"@albumFolderYearAlbumSubtitle": {
|
"@albumFolderYearAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
|
"albumFolderArtistAlbumSingles": "Artist / Album + Singles",
|
||||||
|
"@albumFolderArtistAlbumSingles": {
|
||||||
|
"description": "Album folder option with singles inside artist"
|
||||||
|
},
|
||||||
|
"albumFolderArtistAlbumSinglesSubtitle": "Artist/Album/ and Artist/Singles/",
|
||||||
|
"@albumFolderArtistAlbumSinglesSubtitle": {
|
||||||
|
"description": "Folder structure example"
|
||||||
|
},
|
||||||
"downloadedAlbumDeleteSelected": "Delete Selected",
|
"downloadedAlbumDeleteSelected": "Delete Selected",
|
||||||
"@downloadedAlbumDeleteSelected": {
|
"@downloadedAlbumDeleteSelected": {
|
||||||
"description": "Button - delete selected tracks"
|
"description": "Button - delete selected tracks"
|
||||||
@@ -2572,6 +2697,16 @@
|
|||||||
"@downloadedAlbumSelectToDelete": {
|
"@downloadedAlbumSelectToDelete": {
|
||||||
"description": "Placeholder when nothing selected"
|
"description": "Placeholder when nothing selected"
|
||||||
},
|
},
|
||||||
|
"downloadedAlbumDiscHeader": "Disc {discNumber}",
|
||||||
|
"@downloadedAlbumDiscHeader": {
|
||||||
|
"description": "Header for disc separator in multi-disc albums",
|
||||||
|
"placeholders": {
|
||||||
|
"discNumber": {
|
||||||
|
"type": "int",
|
||||||
|
"example": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"utilityFunctions": "Utility Functions",
|
"utilityFunctions": "Utility Functions",
|
||||||
"@utilityFunctions": {
|
"@utilityFunctions": {
|
||||||
"description": "Extension capability - utility functions"
|
"description": "Extension capability - utility functions"
|
||||||
@@ -2611,5 +2746,123 @@
|
|||||||
"description": "Error message"
|
"description": "Error message"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"discographyDownload": "Download Discography",
|
||||||
|
"@discographyDownload": {
|
||||||
|
"description": "Button - download artist discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAll": "Download All",
|
||||||
|
"@discographyDownloadAll": {
|
||||||
|
"description": "Option - download entire discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAllSubtitle": "{count} tracks from {albumCount} releases",
|
||||||
|
"@discographyDownloadAllSubtitle": {
|
||||||
|
"description": "Subtitle showing total tracks and albums",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnly": "Albums Only",
|
||||||
|
"@discographyAlbumsOnly": {
|
||||||
|
"description": "Option - download only albums"
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnlySubtitle": "{count} tracks from {albumCount} albums",
|
||||||
|
"@discographyAlbumsOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing album tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySinglesOnly": "Singles & EPs Only",
|
||||||
|
"@discographySinglesOnly": {
|
||||||
|
"description": "Option - download only singles"
|
||||||
|
},
|
||||||
|
"discographySinglesOnlySubtitle": "{count} tracks from {albumCount} singles",
|
||||||
|
"@discographySinglesOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing singles tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectAlbums": "Select Albums...",
|
||||||
|
"@discographySelectAlbums": {
|
||||||
|
"description": "Option - manually select albums to download"
|
||||||
|
},
|
||||||
|
"discographySelectAlbumsSubtitle": "Choose specific albums or singles",
|
||||||
|
"@discographySelectAlbumsSubtitle": {
|
||||||
|
"description": "Subtitle for select albums option"
|
||||||
|
},
|
||||||
|
"discographyFetchingTracks": "Fetching tracks...",
|
||||||
|
"@discographyFetchingTracks": {
|
||||||
|
"description": "Progress - fetching album tracks"
|
||||||
|
},
|
||||||
|
"discographyFetchingAlbum": "Fetching {current} of {total}...",
|
||||||
|
"@discographyFetchingAlbum": {
|
||||||
|
"description": "Progress - fetching specific album",
|
||||||
|
"placeholders": {
|
||||||
|
"current": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"total": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectedCount": "{count} selected",
|
||||||
|
"@discographySelectedCount": {
|
||||||
|
"description": "Selection count badge",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyDownloadSelected": "Download Selected",
|
||||||
|
"@discographyDownloadSelected": {
|
||||||
|
"description": "Button - download selected albums"
|
||||||
|
},
|
||||||
|
"discographyAddedToQueue": "Added {count} tracks to queue",
|
||||||
|
"@discographyAddedToQueue": {
|
||||||
|
"description": "Snackbar - tracks added from discography",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySkippedDownloaded": "{added} added, {skipped} already downloaded",
|
||||||
|
"@discographySkippedDownloaded": {
|
||||||
|
"description": "Snackbar - with skipped tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"added": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"skipped": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyNoAlbums": "No albums available",
|
||||||
|
"@discographyNoAlbums": {
|
||||||
|
"description": "Error - no albums found for artist"
|
||||||
|
},
|
||||||
|
"discographyFailedToFetch": "Failed to fetch some albums",
|
||||||
|
"@discographyFailedToFetch": {
|
||||||
|
"description": "Error - some albums failed to load"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
+38
-11
@@ -334,6 +334,8 @@
|
|||||||
"@aboutBinimumDesc": {"description": "Credit description for binimum"},
|
"@aboutBinimumDesc": {"description": "Credit description for binimum"},
|
||||||
"aboutSachinsenalDesc": "The original HiFi project creator. The foundation of Tidal integration!",
|
"aboutSachinsenalDesc": "The original HiFi project creator. The foundation of Tidal integration!",
|
||||||
"@aboutSachinsenalDesc": {"description": "Credit description for sachinsenal0x64"},
|
"@aboutSachinsenalDesc": {"description": "Credit description for sachinsenal0x64"},
|
||||||
|
"aboutSjdonadoDesc": "Creator of I Don't Have Spotify (IDHS). The fallback link resolver that saves the day!",
|
||||||
|
"@aboutSjdonadoDesc": {"description": "Credit description for sjdonado"},
|
||||||
"aboutDoubleDouble": "DoubleDouble",
|
"aboutDoubleDouble": "DoubleDouble",
|
||||||
"@aboutDoubleDouble": {"description": "Name of Amazon API service - DO NOT TRANSLATE"},
|
"@aboutDoubleDouble": {"description": "Name of Amazon API service - DO NOT TRANSLATE"},
|
||||||
"aboutDoubleDoubleDesc": "Amazing API for Amazon Music downloads. Thank you for making it free!",
|
"aboutDoubleDoubleDesc": "Amazing API for Amazon Music downloads. Thank you for making it free!",
|
||||||
@@ -1373,16 +1375,26 @@
|
|||||||
"@qualityHiResFlacMax": {"description": "Quality option - maximum resolution FLAC"},
|
"@qualityHiResFlacMax": {"description": "Quality option - maximum resolution FLAC"},
|
||||||
"qualityHiResFlacMaxSubtitle": "24-bit / up to 192kHz",
|
"qualityHiResFlacMaxSubtitle": "24-bit / up to 192kHz",
|
||||||
"@qualityHiResFlacMaxSubtitle": {"description": "Technical spec for hi-res max"},
|
"@qualityHiResFlacMaxSubtitle": {"description": "Technical spec for hi-res max"},
|
||||||
"qualityMp3": "MP3",
|
"qualityLossy": "Lossy",
|
||||||
"@qualityMp3": {"description": "Quality option - MP3 lossy format"},
|
"@qualityLossy": {"description": "Quality option - lossy format (MP3/Opus)"},
|
||||||
"qualityMp3Subtitle": "320kbps (converted from FLAC)",
|
"qualityLossyMp3Subtitle": "MP3 320kbps (converted from FLAC)",
|
||||||
"@qualityMp3Subtitle": {"description": "Technical spec for MP3"},
|
"@qualityLossyMp3Subtitle": {"description": "Technical spec for lossy MP3"},
|
||||||
"enableMp3Option": "Enable MP3 Option",
|
"qualityLossyOpusSubtitle": "Opus 128kbps (converted from FLAC)",
|
||||||
"@enableMp3Option": {"description": "Setting - enable MP3 quality option"},
|
"@qualityLossyOpusSubtitle": {"description": "Technical spec for lossy Opus"},
|
||||||
"enableMp3OptionSubtitleOn": "MP3 quality option is available",
|
"enableLossyOption": "Enable Lossy Option",
|
||||||
"@enableMp3OptionSubtitleOn": {"description": "Subtitle when MP3 is enabled"},
|
"@enableLossyOption": {"description": "Setting - enable lossy quality option"},
|
||||||
"enableMp3OptionSubtitleOff": "Downloads FLAC then converts to 320kbps MP3",
|
"enableLossyOptionSubtitleOn": "Lossy quality option is available",
|
||||||
"@enableMp3OptionSubtitleOff": {"description": "Subtitle when MP3 is disabled"},
|
"@enableLossyOptionSubtitleOn": {"description": "Subtitle when lossy is enabled"},
|
||||||
|
"enableLossyOptionSubtitleOff": "Downloads FLAC then converts to lossy format",
|
||||||
|
"@enableLossyOptionSubtitleOff": {"description": "Subtitle when lossy is disabled"},
|
||||||
|
"lossyFormat": "Lossy Format",
|
||||||
|
"@lossyFormat": {"description": "Setting - choose lossy format"},
|
||||||
|
"lossyFormatDescription": "Choose the lossy format for conversion",
|
||||||
|
"@lossyFormatDescription": {"description": "Description for lossy format picker"},
|
||||||
|
"lossyFormatMp3Subtitle": "320kbps, best compatibility",
|
||||||
|
"@lossyFormatMp3Subtitle": {"description": "MP3 format description"},
|
||||||
|
"lossyFormatOpusSubtitle": "128kbps, better quality at smaller size",
|
||||||
|
"@lossyFormatOpusSubtitle": {"description": "Opus format description"},
|
||||||
"qualityNote": "Actual quality depends on track availability from the service",
|
"qualityNote": "Actual quality depends on track availability from the service",
|
||||||
"@qualityNote": {"description": "Note about quality availability"},
|
"@qualityNote": {"description": "Note about quality availability"},
|
||||||
|
|
||||||
@@ -1634,5 +1646,20 @@
|
|||||||
"discographyNoAlbums": "No albums available",
|
"discographyNoAlbums": "No albums available",
|
||||||
"@discographyNoAlbums": {"description": "Error - no albums found for artist"},
|
"@discographyNoAlbums": {"description": "Error - no albums found for artist"},
|
||||||
"discographyFailedToFetch": "Failed to fetch some albums",
|
"discographyFailedToFetch": "Failed to fetch some albums",
|
||||||
"@discographyFailedToFetch": {"description": "Error - some albums failed to load"}
|
"@discographyFailedToFetch": {"description": "Error - some albums failed to load"},
|
||||||
|
|
||||||
|
"sectionStorageAccess": "Storage Access",
|
||||||
|
"@sectionStorageAccess": {"description": "Section header for storage access settings"},
|
||||||
|
"allFilesAccess": "All Files Access",
|
||||||
|
"@allFilesAccess": {"description": "Toggle for MANAGE_EXTERNAL_STORAGE permission"},
|
||||||
|
"allFilesAccessEnabledSubtitle": "Can write to any folder",
|
||||||
|
"@allFilesAccessEnabledSubtitle": {"description": "Subtitle when all files access is enabled"},
|
||||||
|
"allFilesAccessDisabledSubtitle": "Limited to media folders only",
|
||||||
|
"@allFilesAccessDisabledSubtitle": {"description": "Subtitle when all files access is disabled"},
|
||||||
|
"allFilesAccessDescription": "Enable this if you encounter write errors when saving to custom folders. Android 13+ restricts access to certain directories by default.",
|
||||||
|
"@allFilesAccessDescription": {"description": "Description explaining when to enable all files access"},
|
||||||
|
"allFilesAccessDeniedMessage": "Permission was denied. Please enable 'All files access' manually in system settings.",
|
||||||
|
"@allFilesAccessDeniedMessage": {"description": "Message when permission is permanently denied"},
|
||||||
|
"allFilesAccessDisabledMessage": "All Files Access disabled. The app will use limited storage access.",
|
||||||
|
"@allFilesAccessDisabledMessage": {"description": "Snackbar message when user disables all files access"}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -127,6 +127,10 @@
|
|||||||
"@historyNoSinglesSubtitle": {
|
"@historyNoSinglesSubtitle": {
|
||||||
"description": "Empty state subtitle for singles filter"
|
"description": "Empty state subtitle for singles filter"
|
||||||
},
|
},
|
||||||
|
"historySearchHint": "Search history...",
|
||||||
|
"@historySearchHint": {
|
||||||
|
"description": "Search bar placeholder in history"
|
||||||
|
},
|
||||||
"settingsTitle": "Settings",
|
"settingsTitle": "Settings",
|
||||||
"@settingsTitle": {
|
"@settingsTitle": {
|
||||||
"description": "Settings screen title"
|
"description": "Settings screen title"
|
||||||
@@ -512,6 +516,10 @@
|
|||||||
"@aboutLogoArtist": {
|
"@aboutLogoArtist": {
|
||||||
"description": "Role description for logo artist"
|
"description": "Role description for logo artist"
|
||||||
},
|
},
|
||||||
|
"aboutTranslators": "Translators",
|
||||||
|
"@aboutTranslators": {
|
||||||
|
"description": "Section for translators"
|
||||||
|
},
|
||||||
"aboutSpecialThanks": "Special Thanks",
|
"aboutSpecialThanks": "Special Thanks",
|
||||||
"@aboutSpecialThanks": {
|
"@aboutSpecialThanks": {
|
||||||
"description": "Section for special thanks"
|
"description": "Section for special thanks"
|
||||||
@@ -544,6 +552,26 @@
|
|||||||
"@aboutFeatureRequestSubtitle": {
|
"@aboutFeatureRequestSubtitle": {
|
||||||
"description": "Subtitle for feature request"
|
"description": "Subtitle for feature request"
|
||||||
},
|
},
|
||||||
|
"aboutTelegramChannel": "Telegram Channel",
|
||||||
|
"@aboutTelegramChannel": {
|
||||||
|
"description": "Link to Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChannelSubtitle": "Announcements and updates",
|
||||||
|
"@aboutTelegramChannelSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChat": "Telegram Community",
|
||||||
|
"@aboutTelegramChat": {
|
||||||
|
"description": "Link to Telegram chat group"
|
||||||
|
},
|
||||||
|
"aboutTelegramChatSubtitle": "Chat with other users",
|
||||||
|
"@aboutTelegramChatSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram chat"
|
||||||
|
},
|
||||||
|
"aboutSocial": "Social",
|
||||||
|
"@aboutSocial": {
|
||||||
|
"description": "Section for social links"
|
||||||
|
},
|
||||||
"aboutSupport": "Support",
|
"aboutSupport": "Support",
|
||||||
"@aboutSupport": {
|
"@aboutSupport": {
|
||||||
"description": "Section for support/donation links"
|
"description": "Section for support/donation links"
|
||||||
@@ -1122,6 +1150,15 @@
|
|||||||
"description": "Dialog title - import CSV playlist"
|
"description": "Dialog title - import CSV playlist"
|
||||||
},
|
},
|
||||||
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
||||||
|
"csvImportTracks": "{count} tracks from CSV",
|
||||||
|
"@csvImportTracks": {
|
||||||
|
"description": "Label shown in quality picker for CSV import",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"@dialogImportPlaylistMessage": {
|
"@dialogImportPlaylistMessage": {
|
||||||
"description": "Dialog message - import playlist confirmation",
|
"description": "Dialog message - import playlist confirmation",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1851,6 +1888,42 @@
|
|||||||
"@sectionFileSettings": {
|
"@sectionFileSettings": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
},
|
},
|
||||||
|
"sectionLyrics": "Lyrics",
|
||||||
|
"@sectionLyrics": {
|
||||||
|
"description": "Settings section header"
|
||||||
|
},
|
||||||
|
"lyricsMode": "Lyrics Mode",
|
||||||
|
"@lyricsMode": {
|
||||||
|
"description": "Setting - how to save lyrics"
|
||||||
|
},
|
||||||
|
"lyricsModeDescription": "Choose how lyrics are saved with your downloads",
|
||||||
|
"@lyricsModeDescription": {
|
||||||
|
"description": "Lyrics mode picker description"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbed": "Embed in file",
|
||||||
|
"@lyricsModeEmbed": {
|
||||||
|
"description": "Lyrics mode option - embed in audio file"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbedSubtitle": "Lyrics stored inside FLAC metadata",
|
||||||
|
"@lyricsModeEmbedSubtitle": {
|
||||||
|
"description": "Subtitle for embed option"
|
||||||
|
},
|
||||||
|
"lyricsModeExternal": "External .lrc file",
|
||||||
|
"@lyricsModeExternal": {
|
||||||
|
"description": "Lyrics mode option - separate LRC file"
|
||||||
|
},
|
||||||
|
"lyricsModeExternalSubtitle": "Separate .lrc file for players like Samsung Music",
|
||||||
|
"@lyricsModeExternalSubtitle": {
|
||||||
|
"description": "Subtitle for external option"
|
||||||
|
},
|
||||||
|
"lyricsModeBoth": "Both",
|
||||||
|
"@lyricsModeBoth": {
|
||||||
|
"description": "Lyrics mode option - embed and external"
|
||||||
|
},
|
||||||
|
"lyricsModeBothSubtitle": "Embed and save .lrc file",
|
||||||
|
"@lyricsModeBothSubtitle": {
|
||||||
|
"description": "Subtitle for both option"
|
||||||
|
},
|
||||||
"sectionColor": "Color",
|
"sectionColor": "Color",
|
||||||
"@sectionColor": {
|
"@sectionColor": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
@@ -1997,6 +2070,18 @@
|
|||||||
"@trackReleaseDate": {
|
"@trackReleaseDate": {
|
||||||
"description": "Metadata label - release date"
|
"description": "Metadata label - release date"
|
||||||
},
|
},
|
||||||
|
"trackGenre": "Genre",
|
||||||
|
"@trackGenre": {
|
||||||
|
"description": "Metadata label - music genre"
|
||||||
|
},
|
||||||
|
"trackLabel": "Label",
|
||||||
|
"@trackLabel": {
|
||||||
|
"description": "Metadata label - record label"
|
||||||
|
},
|
||||||
|
"trackCopyright": "Copyright",
|
||||||
|
"@trackCopyright": {
|
||||||
|
"description": "Metadata label - copyright information"
|
||||||
|
},
|
||||||
"trackDownloaded": "Downloaded",
|
"trackDownloaded": "Downloaded",
|
||||||
"@trackDownloaded": {
|
"@trackDownloaded": {
|
||||||
"description": "Metadata label - download date"
|
"description": "Metadata label - download date"
|
||||||
@@ -2017,6 +2102,18 @@
|
|||||||
"@trackLyricsLoadFailed": {
|
"@trackLyricsLoadFailed": {
|
||||||
"description": "Message when lyrics loading fails"
|
"description": "Message when lyrics loading fails"
|
||||||
},
|
},
|
||||||
|
"trackEmbedLyrics": "Embed Lyrics",
|
||||||
|
"@trackEmbedLyrics": {
|
||||||
|
"description": "Action - embed lyrics into audio file"
|
||||||
|
},
|
||||||
|
"trackLyricsEmbedded": "Lyrics embedded successfully",
|
||||||
|
"@trackLyricsEmbedded": {
|
||||||
|
"description": "Snackbar - lyrics saved to file"
|
||||||
|
},
|
||||||
|
"trackInstrumental": "Instrumental track",
|
||||||
|
"@trackInstrumental": {
|
||||||
|
"description": "Message when track is instrumental (no lyrics)"
|
||||||
|
},
|
||||||
"trackCopiedToClipboard": "Copied to clipboard",
|
"trackCopiedToClipboard": "Copied to clipboard",
|
||||||
"@trackCopiedToClipboard": {
|
"@trackCopiedToClipboard": {
|
||||||
"description": "Snackbar - content copied"
|
"description": "Snackbar - content copied"
|
||||||
@@ -2328,6 +2425,26 @@
|
|||||||
"@qualityHiResFlacMaxSubtitle": {
|
"@qualityHiResFlacMaxSubtitle": {
|
||||||
"description": "Technical spec for hi-res max"
|
"description": "Technical spec for hi-res max"
|
||||||
},
|
},
|
||||||
|
"qualityMp3": "MP3",
|
||||||
|
"@qualityMp3": {
|
||||||
|
"description": "Quality option - MP3 lossy format"
|
||||||
|
},
|
||||||
|
"qualityMp3Subtitle": "320kbps (converted from FLAC)",
|
||||||
|
"@qualityMp3Subtitle": {
|
||||||
|
"description": "Technical spec for MP3"
|
||||||
|
},
|
||||||
|
"enableMp3Option": "Enable MP3 Option",
|
||||||
|
"@enableMp3Option": {
|
||||||
|
"description": "Setting - enable MP3 quality option"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOn": "MP3 quality option is available",
|
||||||
|
"@enableMp3OptionSubtitleOn": {
|
||||||
|
"description": "Subtitle when MP3 is enabled"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOff": "Downloads FLAC then converts to 320kbps MP3",
|
||||||
|
"@enableMp3OptionSubtitleOff": {
|
||||||
|
"description": "Subtitle when MP3 is disabled"
|
||||||
|
},
|
||||||
"qualityNote": "Actual quality depends on track availability from the service",
|
"qualityNote": "Actual quality depends on track availability from the service",
|
||||||
"@qualityNote": {
|
"@qualityNote": {
|
||||||
"description": "Note about quality availability"
|
"description": "Note about quality availability"
|
||||||
@@ -2516,6 +2633,14 @@
|
|||||||
"@albumFolderYearAlbumSubtitle": {
|
"@albumFolderYearAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
|
"albumFolderArtistAlbumSingles": "Artist / Album + Singles",
|
||||||
|
"@albumFolderArtistAlbumSingles": {
|
||||||
|
"description": "Album folder option with singles inside artist"
|
||||||
|
},
|
||||||
|
"albumFolderArtistAlbumSinglesSubtitle": "Artist/Album/ and Artist/Singles/",
|
||||||
|
"@albumFolderArtistAlbumSinglesSubtitle": {
|
||||||
|
"description": "Folder structure example"
|
||||||
|
},
|
||||||
"downloadedAlbumDeleteSelected": "Delete Selected",
|
"downloadedAlbumDeleteSelected": "Delete Selected",
|
||||||
"@downloadedAlbumDeleteSelected": {
|
"@downloadedAlbumDeleteSelected": {
|
||||||
"description": "Button - delete selected tracks"
|
"description": "Button - delete selected tracks"
|
||||||
@@ -2572,6 +2697,16 @@
|
|||||||
"@downloadedAlbumSelectToDelete": {
|
"@downloadedAlbumSelectToDelete": {
|
||||||
"description": "Placeholder when nothing selected"
|
"description": "Placeholder when nothing selected"
|
||||||
},
|
},
|
||||||
|
"downloadedAlbumDiscHeader": "Disc {discNumber}",
|
||||||
|
"@downloadedAlbumDiscHeader": {
|
||||||
|
"description": "Header for disc separator in multi-disc albums",
|
||||||
|
"placeholders": {
|
||||||
|
"discNumber": {
|
||||||
|
"type": "int",
|
||||||
|
"example": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"utilityFunctions": "Utility Functions",
|
"utilityFunctions": "Utility Functions",
|
||||||
"@utilityFunctions": {
|
"@utilityFunctions": {
|
||||||
"description": "Extension capability - utility functions"
|
"description": "Extension capability - utility functions"
|
||||||
@@ -2611,5 +2746,123 @@
|
|||||||
"description": "Error message"
|
"description": "Error message"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"discographyDownload": "Download Discography",
|
||||||
|
"@discographyDownload": {
|
||||||
|
"description": "Button - download artist discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAll": "Download All",
|
||||||
|
"@discographyDownloadAll": {
|
||||||
|
"description": "Option - download entire discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAllSubtitle": "{count} tracks from {albumCount} releases",
|
||||||
|
"@discographyDownloadAllSubtitle": {
|
||||||
|
"description": "Subtitle showing total tracks and albums",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnly": "Albums Only",
|
||||||
|
"@discographyAlbumsOnly": {
|
||||||
|
"description": "Option - download only albums"
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnlySubtitle": "{count} tracks from {albumCount} albums",
|
||||||
|
"@discographyAlbumsOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing album tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySinglesOnly": "Singles & EPs Only",
|
||||||
|
"@discographySinglesOnly": {
|
||||||
|
"description": "Option - download only singles"
|
||||||
|
},
|
||||||
|
"discographySinglesOnlySubtitle": "{count} tracks from {albumCount} singles",
|
||||||
|
"@discographySinglesOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing singles tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectAlbums": "Select Albums...",
|
||||||
|
"@discographySelectAlbums": {
|
||||||
|
"description": "Option - manually select albums to download"
|
||||||
|
},
|
||||||
|
"discographySelectAlbumsSubtitle": "Choose specific albums or singles",
|
||||||
|
"@discographySelectAlbumsSubtitle": {
|
||||||
|
"description": "Subtitle for select albums option"
|
||||||
|
},
|
||||||
|
"discographyFetchingTracks": "Fetching tracks...",
|
||||||
|
"@discographyFetchingTracks": {
|
||||||
|
"description": "Progress - fetching album tracks"
|
||||||
|
},
|
||||||
|
"discographyFetchingAlbum": "Fetching {current} of {total}...",
|
||||||
|
"@discographyFetchingAlbum": {
|
||||||
|
"description": "Progress - fetching specific album",
|
||||||
|
"placeholders": {
|
||||||
|
"current": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"total": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectedCount": "{count} selected",
|
||||||
|
"@discographySelectedCount": {
|
||||||
|
"description": "Selection count badge",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyDownloadSelected": "Download Selected",
|
||||||
|
"@discographyDownloadSelected": {
|
||||||
|
"description": "Button - download selected albums"
|
||||||
|
},
|
||||||
|
"discographyAddedToQueue": "Added {count} tracks to queue",
|
||||||
|
"@discographyAddedToQueue": {
|
||||||
|
"description": "Snackbar - tracks added from discography",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySkippedDownloaded": "{added} added, {skipped} already downloaded",
|
||||||
|
"@discographySkippedDownloaded": {
|
||||||
|
"description": "Snackbar - with skipped tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"added": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"skipped": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyNoAlbums": "No albums available",
|
||||||
|
"@discographyNoAlbums": {
|
||||||
|
"description": "Error - no albums found for artist"
|
||||||
|
},
|
||||||
|
"discographyFailedToFetch": "Failed to fetch some albums",
|
||||||
|
"@discographyFailedToFetch": {
|
||||||
|
"description": "Error - some albums failed to load"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
+261
-8
@@ -1,23 +1,23 @@
|
|||||||
{
|
{
|
||||||
"@@locale": "hi",
|
"@@locale": "hi",
|
||||||
"@@last_modified": "2026-01-16",
|
"@@last_modified": "2026-01-16",
|
||||||
"appName": "SpotiFLAC",
|
"appName": "SpotiFlac",
|
||||||
"@appName": {
|
"@appName": {
|
||||||
"description": "App name - DO NOT TRANSLATE"
|
"description": "App name - DO NOT TRANSLATE"
|
||||||
},
|
},
|
||||||
"appDescription": "Download Spotify tracks in lossless quality from Tidal, Qobuz, and Amazon Music.",
|
"appDescription": "स्पॉटीफाई ट्रैक डाउनलोड करें टाइडल, क्वाबज एवं एवं अमेजन म्यूजिक से उच्चतम क्वालिटी में।",
|
||||||
"@appDescription": {
|
"@appDescription": {
|
||||||
"description": "App description shown in about page"
|
"description": "App description shown in about page"
|
||||||
},
|
},
|
||||||
"navHome": "Home",
|
"navHome": "होम",
|
||||||
"@navHome": {
|
"@navHome": {
|
||||||
"description": "Bottom navigation - Home tab"
|
"description": "Bottom navigation - Home tab"
|
||||||
},
|
},
|
||||||
"navHistory": "History",
|
"navHistory": "इतिहास",
|
||||||
"@navHistory": {
|
"@navHistory": {
|
||||||
"description": "Bottom navigation - History tab"
|
"description": "Bottom navigation - History tab"
|
||||||
},
|
},
|
||||||
"navSettings": "Settings",
|
"navSettings": "विकल्प",
|
||||||
"@navSettings": {
|
"@navSettings": {
|
||||||
"description": "Bottom navigation - Settings tab"
|
"description": "Bottom navigation - Settings tab"
|
||||||
},
|
},
|
||||||
@@ -127,6 +127,10 @@
|
|||||||
"@historyNoSinglesSubtitle": {
|
"@historyNoSinglesSubtitle": {
|
||||||
"description": "Empty state subtitle for singles filter"
|
"description": "Empty state subtitle for singles filter"
|
||||||
},
|
},
|
||||||
|
"historySearchHint": "Search history...",
|
||||||
|
"@historySearchHint": {
|
||||||
|
"description": "Search bar placeholder in history"
|
||||||
|
},
|
||||||
"settingsTitle": "Settings",
|
"settingsTitle": "Settings",
|
||||||
"@settingsTitle": {
|
"@settingsTitle": {
|
||||||
"description": "Settings screen title"
|
"description": "Settings screen title"
|
||||||
@@ -219,7 +223,7 @@
|
|||||||
"@quality128": {
|
"@quality128": {
|
||||||
"description": "Audio quality option - 128kbps MP3"
|
"description": "Audio quality option - 128kbps MP3"
|
||||||
},
|
},
|
||||||
"appearanceTitle": "Appearance",
|
"appearanceTitle": "दिखावट",
|
||||||
"@appearanceTitle": {
|
"@appearanceTitle": {
|
||||||
"description": "Appearance settings page title"
|
"description": "Appearance settings page title"
|
||||||
},
|
},
|
||||||
@@ -239,11 +243,11 @@
|
|||||||
"@appearanceThemeDark": {
|
"@appearanceThemeDark": {
|
||||||
"description": "Dark theme"
|
"description": "Dark theme"
|
||||||
},
|
},
|
||||||
"appearanceDynamicColor": "Dynamic Color",
|
"appearanceDynamicColor": "डायनेमिक रंग",
|
||||||
"@appearanceDynamicColor": {
|
"@appearanceDynamicColor": {
|
||||||
"description": "Material You dynamic colors"
|
"description": "Material You dynamic colors"
|
||||||
},
|
},
|
||||||
"appearanceDynamicColorSubtitle": "Use colors from your wallpaper",
|
"appearanceDynamicColorSubtitle": "वॉलपेपर से रंग इस्तेमाल करें",
|
||||||
"@appearanceDynamicColorSubtitle": {
|
"@appearanceDynamicColorSubtitle": {
|
||||||
"description": "Subtitle for dynamic color"
|
"description": "Subtitle for dynamic color"
|
||||||
},
|
},
|
||||||
@@ -512,6 +516,10 @@
|
|||||||
"@aboutLogoArtist": {
|
"@aboutLogoArtist": {
|
||||||
"description": "Role description for logo artist"
|
"description": "Role description for logo artist"
|
||||||
},
|
},
|
||||||
|
"aboutTranslators": "Translators",
|
||||||
|
"@aboutTranslators": {
|
||||||
|
"description": "Section for translators"
|
||||||
|
},
|
||||||
"aboutSpecialThanks": "Special Thanks",
|
"aboutSpecialThanks": "Special Thanks",
|
||||||
"@aboutSpecialThanks": {
|
"@aboutSpecialThanks": {
|
||||||
"description": "Section for special thanks"
|
"description": "Section for special thanks"
|
||||||
@@ -544,6 +552,26 @@
|
|||||||
"@aboutFeatureRequestSubtitle": {
|
"@aboutFeatureRequestSubtitle": {
|
||||||
"description": "Subtitle for feature request"
|
"description": "Subtitle for feature request"
|
||||||
},
|
},
|
||||||
|
"aboutTelegramChannel": "Telegram Channel",
|
||||||
|
"@aboutTelegramChannel": {
|
||||||
|
"description": "Link to Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChannelSubtitle": "Announcements and updates",
|
||||||
|
"@aboutTelegramChannelSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChat": "Telegram Community",
|
||||||
|
"@aboutTelegramChat": {
|
||||||
|
"description": "Link to Telegram chat group"
|
||||||
|
},
|
||||||
|
"aboutTelegramChatSubtitle": "Chat with other users",
|
||||||
|
"@aboutTelegramChatSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram chat"
|
||||||
|
},
|
||||||
|
"aboutSocial": "Social",
|
||||||
|
"@aboutSocial": {
|
||||||
|
"description": "Section for social links"
|
||||||
|
},
|
||||||
"aboutSupport": "Support",
|
"aboutSupport": "Support",
|
||||||
"@aboutSupport": {
|
"@aboutSupport": {
|
||||||
"description": "Section for support/donation links"
|
"description": "Section for support/donation links"
|
||||||
@@ -1122,6 +1150,15 @@
|
|||||||
"description": "Dialog title - import CSV playlist"
|
"description": "Dialog title - import CSV playlist"
|
||||||
},
|
},
|
||||||
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
||||||
|
"csvImportTracks": "{count} tracks from CSV",
|
||||||
|
"@csvImportTracks": {
|
||||||
|
"description": "Label shown in quality picker for CSV import",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"@dialogImportPlaylistMessage": {
|
"@dialogImportPlaylistMessage": {
|
||||||
"description": "Dialog message - import playlist confirmation",
|
"description": "Dialog message - import playlist confirmation",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1851,6 +1888,42 @@
|
|||||||
"@sectionFileSettings": {
|
"@sectionFileSettings": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
},
|
},
|
||||||
|
"sectionLyrics": "Lyrics",
|
||||||
|
"@sectionLyrics": {
|
||||||
|
"description": "Settings section header"
|
||||||
|
},
|
||||||
|
"lyricsMode": "Lyrics Mode",
|
||||||
|
"@lyricsMode": {
|
||||||
|
"description": "Setting - how to save lyrics"
|
||||||
|
},
|
||||||
|
"lyricsModeDescription": "Choose how lyrics are saved with your downloads",
|
||||||
|
"@lyricsModeDescription": {
|
||||||
|
"description": "Lyrics mode picker description"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbed": "Embed in file",
|
||||||
|
"@lyricsModeEmbed": {
|
||||||
|
"description": "Lyrics mode option - embed in audio file"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbedSubtitle": "Lyrics stored inside FLAC metadata",
|
||||||
|
"@lyricsModeEmbedSubtitle": {
|
||||||
|
"description": "Subtitle for embed option"
|
||||||
|
},
|
||||||
|
"lyricsModeExternal": "External .lrc file",
|
||||||
|
"@lyricsModeExternal": {
|
||||||
|
"description": "Lyrics mode option - separate LRC file"
|
||||||
|
},
|
||||||
|
"lyricsModeExternalSubtitle": "Separate .lrc file for players like Samsung Music",
|
||||||
|
"@lyricsModeExternalSubtitle": {
|
||||||
|
"description": "Subtitle for external option"
|
||||||
|
},
|
||||||
|
"lyricsModeBoth": "Both",
|
||||||
|
"@lyricsModeBoth": {
|
||||||
|
"description": "Lyrics mode option - embed and external"
|
||||||
|
},
|
||||||
|
"lyricsModeBothSubtitle": "Embed and save .lrc file",
|
||||||
|
"@lyricsModeBothSubtitle": {
|
||||||
|
"description": "Subtitle for both option"
|
||||||
|
},
|
||||||
"sectionColor": "Color",
|
"sectionColor": "Color",
|
||||||
"@sectionColor": {
|
"@sectionColor": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
@@ -1997,6 +2070,18 @@
|
|||||||
"@trackReleaseDate": {
|
"@trackReleaseDate": {
|
||||||
"description": "Metadata label - release date"
|
"description": "Metadata label - release date"
|
||||||
},
|
},
|
||||||
|
"trackGenre": "Genre",
|
||||||
|
"@trackGenre": {
|
||||||
|
"description": "Metadata label - music genre"
|
||||||
|
},
|
||||||
|
"trackLabel": "Label",
|
||||||
|
"@trackLabel": {
|
||||||
|
"description": "Metadata label - record label"
|
||||||
|
},
|
||||||
|
"trackCopyright": "Copyright",
|
||||||
|
"@trackCopyright": {
|
||||||
|
"description": "Metadata label - copyright information"
|
||||||
|
},
|
||||||
"trackDownloaded": "Downloaded",
|
"trackDownloaded": "Downloaded",
|
||||||
"@trackDownloaded": {
|
"@trackDownloaded": {
|
||||||
"description": "Metadata label - download date"
|
"description": "Metadata label - download date"
|
||||||
@@ -2017,6 +2102,18 @@
|
|||||||
"@trackLyricsLoadFailed": {
|
"@trackLyricsLoadFailed": {
|
||||||
"description": "Message when lyrics loading fails"
|
"description": "Message when lyrics loading fails"
|
||||||
},
|
},
|
||||||
|
"trackEmbedLyrics": "Embed Lyrics",
|
||||||
|
"@trackEmbedLyrics": {
|
||||||
|
"description": "Action - embed lyrics into audio file"
|
||||||
|
},
|
||||||
|
"trackLyricsEmbedded": "Lyrics embedded successfully",
|
||||||
|
"@trackLyricsEmbedded": {
|
||||||
|
"description": "Snackbar - lyrics saved to file"
|
||||||
|
},
|
||||||
|
"trackInstrumental": "Instrumental track",
|
||||||
|
"@trackInstrumental": {
|
||||||
|
"description": "Message when track is instrumental (no lyrics)"
|
||||||
|
},
|
||||||
"trackCopiedToClipboard": "Copied to clipboard",
|
"trackCopiedToClipboard": "Copied to clipboard",
|
||||||
"@trackCopiedToClipboard": {
|
"@trackCopiedToClipboard": {
|
||||||
"description": "Snackbar - content copied"
|
"description": "Snackbar - content copied"
|
||||||
@@ -2328,6 +2425,26 @@
|
|||||||
"@qualityHiResFlacMaxSubtitle": {
|
"@qualityHiResFlacMaxSubtitle": {
|
||||||
"description": "Technical spec for hi-res max"
|
"description": "Technical spec for hi-res max"
|
||||||
},
|
},
|
||||||
|
"qualityMp3": "MP3",
|
||||||
|
"@qualityMp3": {
|
||||||
|
"description": "Quality option - MP3 lossy format"
|
||||||
|
},
|
||||||
|
"qualityMp3Subtitle": "320kbps (converted from FLAC)",
|
||||||
|
"@qualityMp3Subtitle": {
|
||||||
|
"description": "Technical spec for MP3"
|
||||||
|
},
|
||||||
|
"enableMp3Option": "Enable MP3 Option",
|
||||||
|
"@enableMp3Option": {
|
||||||
|
"description": "Setting - enable MP3 quality option"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOn": "MP3 quality option is available",
|
||||||
|
"@enableMp3OptionSubtitleOn": {
|
||||||
|
"description": "Subtitle when MP3 is enabled"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOff": "Downloads FLAC then converts to 320kbps MP3",
|
||||||
|
"@enableMp3OptionSubtitleOff": {
|
||||||
|
"description": "Subtitle when MP3 is disabled"
|
||||||
|
},
|
||||||
"qualityNote": "Actual quality depends on track availability from the service",
|
"qualityNote": "Actual quality depends on track availability from the service",
|
||||||
"@qualityNote": {
|
"@qualityNote": {
|
||||||
"description": "Note about quality availability"
|
"description": "Note about quality availability"
|
||||||
@@ -2516,6 +2633,14 @@
|
|||||||
"@albumFolderYearAlbumSubtitle": {
|
"@albumFolderYearAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
|
"albumFolderArtistAlbumSingles": "Artist / Album + Singles",
|
||||||
|
"@albumFolderArtistAlbumSingles": {
|
||||||
|
"description": "Album folder option with singles inside artist"
|
||||||
|
},
|
||||||
|
"albumFolderArtistAlbumSinglesSubtitle": "Artist/Album/ and Artist/Singles/",
|
||||||
|
"@albumFolderArtistAlbumSinglesSubtitle": {
|
||||||
|
"description": "Folder structure example"
|
||||||
|
},
|
||||||
"downloadedAlbumDeleteSelected": "Delete Selected",
|
"downloadedAlbumDeleteSelected": "Delete Selected",
|
||||||
"@downloadedAlbumDeleteSelected": {
|
"@downloadedAlbumDeleteSelected": {
|
||||||
"description": "Button - delete selected tracks"
|
"description": "Button - delete selected tracks"
|
||||||
@@ -2572,6 +2697,16 @@
|
|||||||
"@downloadedAlbumSelectToDelete": {
|
"@downloadedAlbumSelectToDelete": {
|
||||||
"description": "Placeholder when nothing selected"
|
"description": "Placeholder when nothing selected"
|
||||||
},
|
},
|
||||||
|
"downloadedAlbumDiscHeader": "Disc {discNumber}",
|
||||||
|
"@downloadedAlbumDiscHeader": {
|
||||||
|
"description": "Header for disc separator in multi-disc albums",
|
||||||
|
"placeholders": {
|
||||||
|
"discNumber": {
|
||||||
|
"type": "int",
|
||||||
|
"example": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"utilityFunctions": "Utility Functions",
|
"utilityFunctions": "Utility Functions",
|
||||||
"@utilityFunctions": {
|
"@utilityFunctions": {
|
||||||
"description": "Extension capability - utility functions"
|
"description": "Extension capability - utility functions"
|
||||||
@@ -2611,5 +2746,123 @@
|
|||||||
"description": "Error message"
|
"description": "Error message"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"discographyDownload": "Download Discography",
|
||||||
|
"@discographyDownload": {
|
||||||
|
"description": "Button - download artist discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAll": "Download All",
|
||||||
|
"@discographyDownloadAll": {
|
||||||
|
"description": "Option - download entire discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAllSubtitle": "{count} tracks from {albumCount} releases",
|
||||||
|
"@discographyDownloadAllSubtitle": {
|
||||||
|
"description": "Subtitle showing total tracks and albums",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnly": "Albums Only",
|
||||||
|
"@discographyAlbumsOnly": {
|
||||||
|
"description": "Option - download only albums"
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnlySubtitle": "{count} tracks from {albumCount} albums",
|
||||||
|
"@discographyAlbumsOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing album tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySinglesOnly": "Singles & EPs Only",
|
||||||
|
"@discographySinglesOnly": {
|
||||||
|
"description": "Option - download only singles"
|
||||||
|
},
|
||||||
|
"discographySinglesOnlySubtitle": "{count} tracks from {albumCount} singles",
|
||||||
|
"@discographySinglesOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing singles tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectAlbums": "Select Albums...",
|
||||||
|
"@discographySelectAlbums": {
|
||||||
|
"description": "Option - manually select albums to download"
|
||||||
|
},
|
||||||
|
"discographySelectAlbumsSubtitle": "Choose specific albums or singles",
|
||||||
|
"@discographySelectAlbumsSubtitle": {
|
||||||
|
"description": "Subtitle for select albums option"
|
||||||
|
},
|
||||||
|
"discographyFetchingTracks": "Fetching tracks...",
|
||||||
|
"@discographyFetchingTracks": {
|
||||||
|
"description": "Progress - fetching album tracks"
|
||||||
|
},
|
||||||
|
"discographyFetchingAlbum": "Fetching {current} of {total}...",
|
||||||
|
"@discographyFetchingAlbum": {
|
||||||
|
"description": "Progress - fetching specific album",
|
||||||
|
"placeholders": {
|
||||||
|
"current": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"total": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectedCount": "{count} selected",
|
||||||
|
"@discographySelectedCount": {
|
||||||
|
"description": "Selection count badge",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyDownloadSelected": "Download Selected",
|
||||||
|
"@discographyDownloadSelected": {
|
||||||
|
"description": "Button - download selected albums"
|
||||||
|
},
|
||||||
|
"discographyAddedToQueue": "Added {count} tracks to queue",
|
||||||
|
"@discographyAddedToQueue": {
|
||||||
|
"description": "Snackbar - tracks added from discography",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySkippedDownloaded": "{added} added, {skipped} already downloaded",
|
||||||
|
"@discographySkippedDownloaded": {
|
||||||
|
"description": "Snackbar - with skipped tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"added": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"skipped": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyNoAlbums": "No albums available",
|
||||||
|
"@discographyNoAlbums": {
|
||||||
|
"description": "Error - no albums found for artist"
|
||||||
|
},
|
||||||
|
"discographyFailedToFetch": "Failed to fetch some albums",
|
||||||
|
"@discographyFailedToFetch": {
|
||||||
|
"description": "Error - some albums failed to load"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
+2727
-564
File diff suppressed because it is too large
Load Diff
+634
-381
File diff suppressed because it is too large
Load Diff
@@ -127,6 +127,10 @@
|
|||||||
"@historyNoSinglesSubtitle": {
|
"@historyNoSinglesSubtitle": {
|
||||||
"description": "Empty state subtitle for singles filter"
|
"description": "Empty state subtitle for singles filter"
|
||||||
},
|
},
|
||||||
|
"historySearchHint": "Search history...",
|
||||||
|
"@historySearchHint": {
|
||||||
|
"description": "Search bar placeholder in history"
|
||||||
|
},
|
||||||
"settingsTitle": "Settings",
|
"settingsTitle": "Settings",
|
||||||
"@settingsTitle": {
|
"@settingsTitle": {
|
||||||
"description": "Settings screen title"
|
"description": "Settings screen title"
|
||||||
@@ -512,6 +516,10 @@
|
|||||||
"@aboutLogoArtist": {
|
"@aboutLogoArtist": {
|
||||||
"description": "Role description for logo artist"
|
"description": "Role description for logo artist"
|
||||||
},
|
},
|
||||||
|
"aboutTranslators": "Translators",
|
||||||
|
"@aboutTranslators": {
|
||||||
|
"description": "Section for translators"
|
||||||
|
},
|
||||||
"aboutSpecialThanks": "Special Thanks",
|
"aboutSpecialThanks": "Special Thanks",
|
||||||
"@aboutSpecialThanks": {
|
"@aboutSpecialThanks": {
|
||||||
"description": "Section for special thanks"
|
"description": "Section for special thanks"
|
||||||
@@ -544,6 +552,26 @@
|
|||||||
"@aboutFeatureRequestSubtitle": {
|
"@aboutFeatureRequestSubtitle": {
|
||||||
"description": "Subtitle for feature request"
|
"description": "Subtitle for feature request"
|
||||||
},
|
},
|
||||||
|
"aboutTelegramChannel": "Telegram Channel",
|
||||||
|
"@aboutTelegramChannel": {
|
||||||
|
"description": "Link to Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChannelSubtitle": "Announcements and updates",
|
||||||
|
"@aboutTelegramChannelSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChat": "Telegram Community",
|
||||||
|
"@aboutTelegramChat": {
|
||||||
|
"description": "Link to Telegram chat group"
|
||||||
|
},
|
||||||
|
"aboutTelegramChatSubtitle": "Chat with other users",
|
||||||
|
"@aboutTelegramChatSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram chat"
|
||||||
|
},
|
||||||
|
"aboutSocial": "Social",
|
||||||
|
"@aboutSocial": {
|
||||||
|
"description": "Section for social links"
|
||||||
|
},
|
||||||
"aboutSupport": "Support",
|
"aboutSupport": "Support",
|
||||||
"@aboutSupport": {
|
"@aboutSupport": {
|
||||||
"description": "Section for support/donation links"
|
"description": "Section for support/donation links"
|
||||||
@@ -1122,6 +1150,15 @@
|
|||||||
"description": "Dialog title - import CSV playlist"
|
"description": "Dialog title - import CSV playlist"
|
||||||
},
|
},
|
||||||
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
||||||
|
"csvImportTracks": "{count} tracks from CSV",
|
||||||
|
"@csvImportTracks": {
|
||||||
|
"description": "Label shown in quality picker for CSV import",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"@dialogImportPlaylistMessage": {
|
"@dialogImportPlaylistMessage": {
|
||||||
"description": "Dialog message - import playlist confirmation",
|
"description": "Dialog message - import playlist confirmation",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1851,6 +1888,42 @@
|
|||||||
"@sectionFileSettings": {
|
"@sectionFileSettings": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
},
|
},
|
||||||
|
"sectionLyrics": "Lyrics",
|
||||||
|
"@sectionLyrics": {
|
||||||
|
"description": "Settings section header"
|
||||||
|
},
|
||||||
|
"lyricsMode": "Lyrics Mode",
|
||||||
|
"@lyricsMode": {
|
||||||
|
"description": "Setting - how to save lyrics"
|
||||||
|
},
|
||||||
|
"lyricsModeDescription": "Choose how lyrics are saved with your downloads",
|
||||||
|
"@lyricsModeDescription": {
|
||||||
|
"description": "Lyrics mode picker description"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbed": "Embed in file",
|
||||||
|
"@lyricsModeEmbed": {
|
||||||
|
"description": "Lyrics mode option - embed in audio file"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbedSubtitle": "Lyrics stored inside FLAC metadata",
|
||||||
|
"@lyricsModeEmbedSubtitle": {
|
||||||
|
"description": "Subtitle for embed option"
|
||||||
|
},
|
||||||
|
"lyricsModeExternal": "External .lrc file",
|
||||||
|
"@lyricsModeExternal": {
|
||||||
|
"description": "Lyrics mode option - separate LRC file"
|
||||||
|
},
|
||||||
|
"lyricsModeExternalSubtitle": "Separate .lrc file for players like Samsung Music",
|
||||||
|
"@lyricsModeExternalSubtitle": {
|
||||||
|
"description": "Subtitle for external option"
|
||||||
|
},
|
||||||
|
"lyricsModeBoth": "Both",
|
||||||
|
"@lyricsModeBoth": {
|
||||||
|
"description": "Lyrics mode option - embed and external"
|
||||||
|
},
|
||||||
|
"lyricsModeBothSubtitle": "Embed and save .lrc file",
|
||||||
|
"@lyricsModeBothSubtitle": {
|
||||||
|
"description": "Subtitle for both option"
|
||||||
|
},
|
||||||
"sectionColor": "Color",
|
"sectionColor": "Color",
|
||||||
"@sectionColor": {
|
"@sectionColor": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
@@ -1997,6 +2070,18 @@
|
|||||||
"@trackReleaseDate": {
|
"@trackReleaseDate": {
|
||||||
"description": "Metadata label - release date"
|
"description": "Metadata label - release date"
|
||||||
},
|
},
|
||||||
|
"trackGenre": "Genre",
|
||||||
|
"@trackGenre": {
|
||||||
|
"description": "Metadata label - music genre"
|
||||||
|
},
|
||||||
|
"trackLabel": "Label",
|
||||||
|
"@trackLabel": {
|
||||||
|
"description": "Metadata label - record label"
|
||||||
|
},
|
||||||
|
"trackCopyright": "Copyright",
|
||||||
|
"@trackCopyright": {
|
||||||
|
"description": "Metadata label - copyright information"
|
||||||
|
},
|
||||||
"trackDownloaded": "Downloaded",
|
"trackDownloaded": "Downloaded",
|
||||||
"@trackDownloaded": {
|
"@trackDownloaded": {
|
||||||
"description": "Metadata label - download date"
|
"description": "Metadata label - download date"
|
||||||
@@ -2017,6 +2102,18 @@
|
|||||||
"@trackLyricsLoadFailed": {
|
"@trackLyricsLoadFailed": {
|
||||||
"description": "Message when lyrics loading fails"
|
"description": "Message when lyrics loading fails"
|
||||||
},
|
},
|
||||||
|
"trackEmbedLyrics": "Embed Lyrics",
|
||||||
|
"@trackEmbedLyrics": {
|
||||||
|
"description": "Action - embed lyrics into audio file"
|
||||||
|
},
|
||||||
|
"trackLyricsEmbedded": "Lyrics embedded successfully",
|
||||||
|
"@trackLyricsEmbedded": {
|
||||||
|
"description": "Snackbar - lyrics saved to file"
|
||||||
|
},
|
||||||
|
"trackInstrumental": "Instrumental track",
|
||||||
|
"@trackInstrumental": {
|
||||||
|
"description": "Message when track is instrumental (no lyrics)"
|
||||||
|
},
|
||||||
"trackCopiedToClipboard": "Copied to clipboard",
|
"trackCopiedToClipboard": "Copied to clipboard",
|
||||||
"@trackCopiedToClipboard": {
|
"@trackCopiedToClipboard": {
|
||||||
"description": "Snackbar - content copied"
|
"description": "Snackbar - content copied"
|
||||||
@@ -2328,6 +2425,26 @@
|
|||||||
"@qualityHiResFlacMaxSubtitle": {
|
"@qualityHiResFlacMaxSubtitle": {
|
||||||
"description": "Technical spec for hi-res max"
|
"description": "Technical spec for hi-res max"
|
||||||
},
|
},
|
||||||
|
"qualityMp3": "MP3",
|
||||||
|
"@qualityMp3": {
|
||||||
|
"description": "Quality option - MP3 lossy format"
|
||||||
|
},
|
||||||
|
"qualityMp3Subtitle": "320kbps (converted from FLAC)",
|
||||||
|
"@qualityMp3Subtitle": {
|
||||||
|
"description": "Technical spec for MP3"
|
||||||
|
},
|
||||||
|
"enableMp3Option": "Enable MP3 Option",
|
||||||
|
"@enableMp3Option": {
|
||||||
|
"description": "Setting - enable MP3 quality option"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOn": "MP3 quality option is available",
|
||||||
|
"@enableMp3OptionSubtitleOn": {
|
||||||
|
"description": "Subtitle when MP3 is enabled"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOff": "Downloads FLAC then converts to 320kbps MP3",
|
||||||
|
"@enableMp3OptionSubtitleOff": {
|
||||||
|
"description": "Subtitle when MP3 is disabled"
|
||||||
|
},
|
||||||
"qualityNote": "Actual quality depends on track availability from the service",
|
"qualityNote": "Actual quality depends on track availability from the service",
|
||||||
"@qualityNote": {
|
"@qualityNote": {
|
||||||
"description": "Note about quality availability"
|
"description": "Note about quality availability"
|
||||||
@@ -2516,6 +2633,14 @@
|
|||||||
"@albumFolderYearAlbumSubtitle": {
|
"@albumFolderYearAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
|
"albumFolderArtistAlbumSingles": "Artist / Album + Singles",
|
||||||
|
"@albumFolderArtistAlbumSingles": {
|
||||||
|
"description": "Album folder option with singles inside artist"
|
||||||
|
},
|
||||||
|
"albumFolderArtistAlbumSinglesSubtitle": "Artist/Album/ and Artist/Singles/",
|
||||||
|
"@albumFolderArtistAlbumSinglesSubtitle": {
|
||||||
|
"description": "Folder structure example"
|
||||||
|
},
|
||||||
"downloadedAlbumDeleteSelected": "Delete Selected",
|
"downloadedAlbumDeleteSelected": "Delete Selected",
|
||||||
"@downloadedAlbumDeleteSelected": {
|
"@downloadedAlbumDeleteSelected": {
|
||||||
"description": "Button - delete selected tracks"
|
"description": "Button - delete selected tracks"
|
||||||
@@ -2572,6 +2697,16 @@
|
|||||||
"@downloadedAlbumSelectToDelete": {
|
"@downloadedAlbumSelectToDelete": {
|
||||||
"description": "Placeholder when nothing selected"
|
"description": "Placeholder when nothing selected"
|
||||||
},
|
},
|
||||||
|
"downloadedAlbumDiscHeader": "Disc {discNumber}",
|
||||||
|
"@downloadedAlbumDiscHeader": {
|
||||||
|
"description": "Header for disc separator in multi-disc albums",
|
||||||
|
"placeholders": {
|
||||||
|
"discNumber": {
|
||||||
|
"type": "int",
|
||||||
|
"example": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"utilityFunctions": "Utility Functions",
|
"utilityFunctions": "Utility Functions",
|
||||||
"@utilityFunctions": {
|
"@utilityFunctions": {
|
||||||
"description": "Extension capability - utility functions"
|
"description": "Extension capability - utility functions"
|
||||||
@@ -2611,5 +2746,123 @@
|
|||||||
"description": "Error message"
|
"description": "Error message"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"discographyDownload": "Download Discography",
|
||||||
|
"@discographyDownload": {
|
||||||
|
"description": "Button - download artist discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAll": "Download All",
|
||||||
|
"@discographyDownloadAll": {
|
||||||
|
"description": "Option - download entire discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAllSubtitle": "{count} tracks from {albumCount} releases",
|
||||||
|
"@discographyDownloadAllSubtitle": {
|
||||||
|
"description": "Subtitle showing total tracks and albums",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnly": "Albums Only",
|
||||||
|
"@discographyAlbumsOnly": {
|
||||||
|
"description": "Option - download only albums"
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnlySubtitle": "{count} tracks from {albumCount} albums",
|
||||||
|
"@discographyAlbumsOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing album tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySinglesOnly": "Singles & EPs Only",
|
||||||
|
"@discographySinglesOnly": {
|
||||||
|
"description": "Option - download only singles"
|
||||||
|
},
|
||||||
|
"discographySinglesOnlySubtitle": "{count} tracks from {albumCount} singles",
|
||||||
|
"@discographySinglesOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing singles tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectAlbums": "Select Albums...",
|
||||||
|
"@discographySelectAlbums": {
|
||||||
|
"description": "Option - manually select albums to download"
|
||||||
|
},
|
||||||
|
"discographySelectAlbumsSubtitle": "Choose specific albums or singles",
|
||||||
|
"@discographySelectAlbumsSubtitle": {
|
||||||
|
"description": "Subtitle for select albums option"
|
||||||
|
},
|
||||||
|
"discographyFetchingTracks": "Fetching tracks...",
|
||||||
|
"@discographyFetchingTracks": {
|
||||||
|
"description": "Progress - fetching album tracks"
|
||||||
|
},
|
||||||
|
"discographyFetchingAlbum": "Fetching {current} of {total}...",
|
||||||
|
"@discographyFetchingAlbum": {
|
||||||
|
"description": "Progress - fetching specific album",
|
||||||
|
"placeholders": {
|
||||||
|
"current": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"total": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectedCount": "{count} selected",
|
||||||
|
"@discographySelectedCount": {
|
||||||
|
"description": "Selection count badge",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyDownloadSelected": "Download Selected",
|
||||||
|
"@discographyDownloadSelected": {
|
||||||
|
"description": "Button - download selected albums"
|
||||||
|
},
|
||||||
|
"discographyAddedToQueue": "Added {count} tracks to queue",
|
||||||
|
"@discographyAddedToQueue": {
|
||||||
|
"description": "Snackbar - tracks added from discography",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySkippedDownloaded": "{added} added, {skipped} already downloaded",
|
||||||
|
"@discographySkippedDownloaded": {
|
||||||
|
"description": "Snackbar - with skipped tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"added": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"skipped": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyNoAlbums": "No albums available",
|
||||||
|
"@discographyNoAlbums": {
|
||||||
|
"description": "Error - no albums found for artist"
|
||||||
|
},
|
||||||
|
"discographyFailedToFetch": "Failed to fetch some albums",
|
||||||
|
"@discographyFailedToFetch": {
|
||||||
|
"description": "Error - some albums failed to load"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -127,6 +127,10 @@
|
|||||||
"@historyNoSinglesSubtitle": {
|
"@historyNoSinglesSubtitle": {
|
||||||
"description": "Empty state subtitle for singles filter"
|
"description": "Empty state subtitle for singles filter"
|
||||||
},
|
},
|
||||||
|
"historySearchHint": "Search history...",
|
||||||
|
"@historySearchHint": {
|
||||||
|
"description": "Search bar placeholder in history"
|
||||||
|
},
|
||||||
"settingsTitle": "Settings",
|
"settingsTitle": "Settings",
|
||||||
"@settingsTitle": {
|
"@settingsTitle": {
|
||||||
"description": "Settings screen title"
|
"description": "Settings screen title"
|
||||||
@@ -512,6 +516,10 @@
|
|||||||
"@aboutLogoArtist": {
|
"@aboutLogoArtist": {
|
||||||
"description": "Role description for logo artist"
|
"description": "Role description for logo artist"
|
||||||
},
|
},
|
||||||
|
"aboutTranslators": "Translators",
|
||||||
|
"@aboutTranslators": {
|
||||||
|
"description": "Section for translators"
|
||||||
|
},
|
||||||
"aboutSpecialThanks": "Special Thanks",
|
"aboutSpecialThanks": "Special Thanks",
|
||||||
"@aboutSpecialThanks": {
|
"@aboutSpecialThanks": {
|
||||||
"description": "Section for special thanks"
|
"description": "Section for special thanks"
|
||||||
@@ -544,6 +552,26 @@
|
|||||||
"@aboutFeatureRequestSubtitle": {
|
"@aboutFeatureRequestSubtitle": {
|
||||||
"description": "Subtitle for feature request"
|
"description": "Subtitle for feature request"
|
||||||
},
|
},
|
||||||
|
"aboutTelegramChannel": "Telegram Channel",
|
||||||
|
"@aboutTelegramChannel": {
|
||||||
|
"description": "Link to Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChannelSubtitle": "Announcements and updates",
|
||||||
|
"@aboutTelegramChannelSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChat": "Telegram Community",
|
||||||
|
"@aboutTelegramChat": {
|
||||||
|
"description": "Link to Telegram chat group"
|
||||||
|
},
|
||||||
|
"aboutTelegramChatSubtitle": "Chat with other users",
|
||||||
|
"@aboutTelegramChatSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram chat"
|
||||||
|
},
|
||||||
|
"aboutSocial": "Social",
|
||||||
|
"@aboutSocial": {
|
||||||
|
"description": "Section for social links"
|
||||||
|
},
|
||||||
"aboutSupport": "Support",
|
"aboutSupport": "Support",
|
||||||
"@aboutSupport": {
|
"@aboutSupport": {
|
||||||
"description": "Section for support/donation links"
|
"description": "Section for support/donation links"
|
||||||
@@ -1122,6 +1150,15 @@
|
|||||||
"description": "Dialog title - import CSV playlist"
|
"description": "Dialog title - import CSV playlist"
|
||||||
},
|
},
|
||||||
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
||||||
|
"csvImportTracks": "{count} tracks from CSV",
|
||||||
|
"@csvImportTracks": {
|
||||||
|
"description": "Label shown in quality picker for CSV import",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"@dialogImportPlaylistMessage": {
|
"@dialogImportPlaylistMessage": {
|
||||||
"description": "Dialog message - import playlist confirmation",
|
"description": "Dialog message - import playlist confirmation",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1851,6 +1888,42 @@
|
|||||||
"@sectionFileSettings": {
|
"@sectionFileSettings": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
},
|
},
|
||||||
|
"sectionLyrics": "Lyrics",
|
||||||
|
"@sectionLyrics": {
|
||||||
|
"description": "Settings section header"
|
||||||
|
},
|
||||||
|
"lyricsMode": "Lyrics Mode",
|
||||||
|
"@lyricsMode": {
|
||||||
|
"description": "Setting - how to save lyrics"
|
||||||
|
},
|
||||||
|
"lyricsModeDescription": "Choose how lyrics are saved with your downloads",
|
||||||
|
"@lyricsModeDescription": {
|
||||||
|
"description": "Lyrics mode picker description"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbed": "Embed in file",
|
||||||
|
"@lyricsModeEmbed": {
|
||||||
|
"description": "Lyrics mode option - embed in audio file"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbedSubtitle": "Lyrics stored inside FLAC metadata",
|
||||||
|
"@lyricsModeEmbedSubtitle": {
|
||||||
|
"description": "Subtitle for embed option"
|
||||||
|
},
|
||||||
|
"lyricsModeExternal": "External .lrc file",
|
||||||
|
"@lyricsModeExternal": {
|
||||||
|
"description": "Lyrics mode option - separate LRC file"
|
||||||
|
},
|
||||||
|
"lyricsModeExternalSubtitle": "Separate .lrc file for players like Samsung Music",
|
||||||
|
"@lyricsModeExternalSubtitle": {
|
||||||
|
"description": "Subtitle for external option"
|
||||||
|
},
|
||||||
|
"lyricsModeBoth": "Both",
|
||||||
|
"@lyricsModeBoth": {
|
||||||
|
"description": "Lyrics mode option - embed and external"
|
||||||
|
},
|
||||||
|
"lyricsModeBothSubtitle": "Embed and save .lrc file",
|
||||||
|
"@lyricsModeBothSubtitle": {
|
||||||
|
"description": "Subtitle for both option"
|
||||||
|
},
|
||||||
"sectionColor": "Color",
|
"sectionColor": "Color",
|
||||||
"@sectionColor": {
|
"@sectionColor": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
@@ -1997,6 +2070,18 @@
|
|||||||
"@trackReleaseDate": {
|
"@trackReleaseDate": {
|
||||||
"description": "Metadata label - release date"
|
"description": "Metadata label - release date"
|
||||||
},
|
},
|
||||||
|
"trackGenre": "Genre",
|
||||||
|
"@trackGenre": {
|
||||||
|
"description": "Metadata label - music genre"
|
||||||
|
},
|
||||||
|
"trackLabel": "Label",
|
||||||
|
"@trackLabel": {
|
||||||
|
"description": "Metadata label - record label"
|
||||||
|
},
|
||||||
|
"trackCopyright": "Copyright",
|
||||||
|
"@trackCopyright": {
|
||||||
|
"description": "Metadata label - copyright information"
|
||||||
|
},
|
||||||
"trackDownloaded": "Downloaded",
|
"trackDownloaded": "Downloaded",
|
||||||
"@trackDownloaded": {
|
"@trackDownloaded": {
|
||||||
"description": "Metadata label - download date"
|
"description": "Metadata label - download date"
|
||||||
@@ -2017,6 +2102,18 @@
|
|||||||
"@trackLyricsLoadFailed": {
|
"@trackLyricsLoadFailed": {
|
||||||
"description": "Message when lyrics loading fails"
|
"description": "Message when lyrics loading fails"
|
||||||
},
|
},
|
||||||
|
"trackEmbedLyrics": "Embed Lyrics",
|
||||||
|
"@trackEmbedLyrics": {
|
||||||
|
"description": "Action - embed lyrics into audio file"
|
||||||
|
},
|
||||||
|
"trackLyricsEmbedded": "Lyrics embedded successfully",
|
||||||
|
"@trackLyricsEmbedded": {
|
||||||
|
"description": "Snackbar - lyrics saved to file"
|
||||||
|
},
|
||||||
|
"trackInstrumental": "Instrumental track",
|
||||||
|
"@trackInstrumental": {
|
||||||
|
"description": "Message when track is instrumental (no lyrics)"
|
||||||
|
},
|
||||||
"trackCopiedToClipboard": "Copied to clipboard",
|
"trackCopiedToClipboard": "Copied to clipboard",
|
||||||
"@trackCopiedToClipboard": {
|
"@trackCopiedToClipboard": {
|
||||||
"description": "Snackbar - content copied"
|
"description": "Snackbar - content copied"
|
||||||
@@ -2328,6 +2425,26 @@
|
|||||||
"@qualityHiResFlacMaxSubtitle": {
|
"@qualityHiResFlacMaxSubtitle": {
|
||||||
"description": "Technical spec for hi-res max"
|
"description": "Technical spec for hi-res max"
|
||||||
},
|
},
|
||||||
|
"qualityMp3": "MP3",
|
||||||
|
"@qualityMp3": {
|
||||||
|
"description": "Quality option - MP3 lossy format"
|
||||||
|
},
|
||||||
|
"qualityMp3Subtitle": "320kbps (converted from FLAC)",
|
||||||
|
"@qualityMp3Subtitle": {
|
||||||
|
"description": "Technical spec for MP3"
|
||||||
|
},
|
||||||
|
"enableMp3Option": "Enable MP3 Option",
|
||||||
|
"@enableMp3Option": {
|
||||||
|
"description": "Setting - enable MP3 quality option"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOn": "MP3 quality option is available",
|
||||||
|
"@enableMp3OptionSubtitleOn": {
|
||||||
|
"description": "Subtitle when MP3 is enabled"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOff": "Downloads FLAC then converts to 320kbps MP3",
|
||||||
|
"@enableMp3OptionSubtitleOff": {
|
||||||
|
"description": "Subtitle when MP3 is disabled"
|
||||||
|
},
|
||||||
"qualityNote": "Actual quality depends on track availability from the service",
|
"qualityNote": "Actual quality depends on track availability from the service",
|
||||||
"@qualityNote": {
|
"@qualityNote": {
|
||||||
"description": "Note about quality availability"
|
"description": "Note about quality availability"
|
||||||
@@ -2516,6 +2633,14 @@
|
|||||||
"@albumFolderYearAlbumSubtitle": {
|
"@albumFolderYearAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
|
"albumFolderArtistAlbumSingles": "Artist / Album + Singles",
|
||||||
|
"@albumFolderArtistAlbumSingles": {
|
||||||
|
"description": "Album folder option with singles inside artist"
|
||||||
|
},
|
||||||
|
"albumFolderArtistAlbumSinglesSubtitle": "Artist/Album/ and Artist/Singles/",
|
||||||
|
"@albumFolderArtistAlbumSinglesSubtitle": {
|
||||||
|
"description": "Folder structure example"
|
||||||
|
},
|
||||||
"downloadedAlbumDeleteSelected": "Delete Selected",
|
"downloadedAlbumDeleteSelected": "Delete Selected",
|
||||||
"@downloadedAlbumDeleteSelected": {
|
"@downloadedAlbumDeleteSelected": {
|
||||||
"description": "Button - delete selected tracks"
|
"description": "Button - delete selected tracks"
|
||||||
@@ -2572,6 +2697,16 @@
|
|||||||
"@downloadedAlbumSelectToDelete": {
|
"@downloadedAlbumSelectToDelete": {
|
||||||
"description": "Placeholder when nothing selected"
|
"description": "Placeholder when nothing selected"
|
||||||
},
|
},
|
||||||
|
"downloadedAlbumDiscHeader": "Disc {discNumber}",
|
||||||
|
"@downloadedAlbumDiscHeader": {
|
||||||
|
"description": "Header for disc separator in multi-disc albums",
|
||||||
|
"placeholders": {
|
||||||
|
"discNumber": {
|
||||||
|
"type": "int",
|
||||||
|
"example": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"utilityFunctions": "Utility Functions",
|
"utilityFunctions": "Utility Functions",
|
||||||
"@utilityFunctions": {
|
"@utilityFunctions": {
|
||||||
"description": "Extension capability - utility functions"
|
"description": "Extension capability - utility functions"
|
||||||
@@ -2611,5 +2746,123 @@
|
|||||||
"description": "Error message"
|
"description": "Error message"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"discographyDownload": "Download Discography",
|
||||||
|
"@discographyDownload": {
|
||||||
|
"description": "Button - download artist discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAll": "Download All",
|
||||||
|
"@discographyDownloadAll": {
|
||||||
|
"description": "Option - download entire discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAllSubtitle": "{count} tracks from {albumCount} releases",
|
||||||
|
"@discographyDownloadAllSubtitle": {
|
||||||
|
"description": "Subtitle showing total tracks and albums",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnly": "Albums Only",
|
||||||
|
"@discographyAlbumsOnly": {
|
||||||
|
"description": "Option - download only albums"
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnlySubtitle": "{count} tracks from {albumCount} albums",
|
||||||
|
"@discographyAlbumsOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing album tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySinglesOnly": "Singles & EPs Only",
|
||||||
|
"@discographySinglesOnly": {
|
||||||
|
"description": "Option - download only singles"
|
||||||
|
},
|
||||||
|
"discographySinglesOnlySubtitle": "{count} tracks from {albumCount} singles",
|
||||||
|
"@discographySinglesOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing singles tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectAlbums": "Select Albums...",
|
||||||
|
"@discographySelectAlbums": {
|
||||||
|
"description": "Option - manually select albums to download"
|
||||||
|
},
|
||||||
|
"discographySelectAlbumsSubtitle": "Choose specific albums or singles",
|
||||||
|
"@discographySelectAlbumsSubtitle": {
|
||||||
|
"description": "Subtitle for select albums option"
|
||||||
|
},
|
||||||
|
"discographyFetchingTracks": "Fetching tracks...",
|
||||||
|
"@discographyFetchingTracks": {
|
||||||
|
"description": "Progress - fetching album tracks"
|
||||||
|
},
|
||||||
|
"discographyFetchingAlbum": "Fetching {current} of {total}...",
|
||||||
|
"@discographyFetchingAlbum": {
|
||||||
|
"description": "Progress - fetching specific album",
|
||||||
|
"placeholders": {
|
||||||
|
"current": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"total": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectedCount": "{count} selected",
|
||||||
|
"@discographySelectedCount": {
|
||||||
|
"description": "Selection count badge",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyDownloadSelected": "Download Selected",
|
||||||
|
"@discographyDownloadSelected": {
|
||||||
|
"description": "Button - download selected albums"
|
||||||
|
},
|
||||||
|
"discographyAddedToQueue": "Added {count} tracks to queue",
|
||||||
|
"@discographyAddedToQueue": {
|
||||||
|
"description": "Snackbar - tracks added from discography",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySkippedDownloaded": "{added} added, {skipped} already downloaded",
|
||||||
|
"@discographySkippedDownloaded": {
|
||||||
|
"description": "Snackbar - with skipped tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"added": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"skipped": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyNoAlbums": "No albums available",
|
||||||
|
"@discographyNoAlbums": {
|
||||||
|
"description": "Error - no albums found for artist"
|
||||||
|
},
|
||||||
|
"discographyFailedToFetch": "Failed to fetch some albums",
|
||||||
|
"@discographyFailedToFetch": {
|
||||||
|
"description": "Error - some albums failed to load"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
+141
-141
@@ -835,19 +835,19 @@
|
|||||||
"@setupIosEmptyFolderWarning": {
|
"@setupIosEmptyFolderWarning": {
|
||||||
"description": "iOS folder selection warning"
|
"description": "iOS folder selection warning"
|
||||||
},
|
},
|
||||||
"setupDownloadInFlac": "Download Spotify tracks in FLAC",
|
"setupDownloadInFlac": "Baixe faixas do Spotify em FLAC",
|
||||||
"@setupDownloadInFlac": {
|
"@setupDownloadInFlac": {
|
||||||
"description": "App tagline in setup"
|
"description": "App tagline in setup"
|
||||||
},
|
},
|
||||||
"setupStepStorage": "Storage",
|
"setupStepStorage": "Armazenamento",
|
||||||
"@setupStepStorage": {
|
"@setupStepStorage": {
|
||||||
"description": "Setup step indicator - storage"
|
"description": "Setup step indicator - storage"
|
||||||
},
|
},
|
||||||
"setupStepNotification": "Notification",
|
"setupStepNotification": "Notificação",
|
||||||
"@setupStepNotification": {
|
"@setupStepNotification": {
|
||||||
"description": "Setup step indicator - notification"
|
"description": "Setup step indicator - notification"
|
||||||
},
|
},
|
||||||
"setupStepFolder": "Folder",
|
"setupStepFolder": "Pasta",
|
||||||
"@setupStepFolder": {
|
"@setupStepFolder": {
|
||||||
"description": "Setup step indicator - folder"
|
"description": "Setup step indicator - folder"
|
||||||
},
|
},
|
||||||
@@ -855,19 +855,19 @@
|
|||||||
"@setupStepSpotify": {
|
"@setupStepSpotify": {
|
||||||
"description": "Setup step indicator - Spotify API"
|
"description": "Setup step indicator - Spotify API"
|
||||||
},
|
},
|
||||||
"setupStepPermission": "Permission",
|
"setupStepPermission": "Permissão",
|
||||||
"@setupStepPermission": {
|
"@setupStepPermission": {
|
||||||
"description": "Setup step indicator - permission"
|
"description": "Setup step indicator - permission"
|
||||||
},
|
},
|
||||||
"setupStorageGranted": "Storage Permission Granted!",
|
"setupStorageGranted": "Permissão de Armazenamento Concedida!",
|
||||||
"@setupStorageGranted": {
|
"@setupStorageGranted": {
|
||||||
"description": "Success message for storage permission"
|
"description": "Success message for storage permission"
|
||||||
},
|
},
|
||||||
"setupStorageRequired": "Storage Permission Required",
|
"setupStorageRequired": "Permissão de Armazenamento Necessária",
|
||||||
"@setupStorageRequired": {
|
"@setupStorageRequired": {
|
||||||
"description": "Title when storage permission needed"
|
"description": "Title when storage permission needed"
|
||||||
},
|
},
|
||||||
"setupStorageDescription": "SpotiFLAC needs storage permission to save your downloaded music files.",
|
"setupStorageDescription": "O SpotiFLAC precisa de permissão de armazenamento para salvar os seus arquivos de música baixados.",
|
||||||
"@setupStorageDescription": {
|
"@setupStorageDescription": {
|
||||||
"description": "Explanation for storage permission"
|
"description": "Explanation for storage permission"
|
||||||
},
|
},
|
||||||
@@ -1071,23 +1071,23 @@
|
|||||||
"@dialogClearAllDownloads": {
|
"@dialogClearAllDownloads": {
|
||||||
"description": "Dialog message - clear downloads confirmation"
|
"description": "Dialog message - clear downloads confirmation"
|
||||||
},
|
},
|
||||||
"dialogRemoveFromDevice": "Remove from device?",
|
"dialogRemoveFromDevice": "Remover do dispositivo?",
|
||||||
"@dialogRemoveFromDevice": {
|
"@dialogRemoveFromDevice": {
|
||||||
"description": "Dialog title - delete file confirmation"
|
"description": "Dialog title - delete file confirmation"
|
||||||
},
|
},
|
||||||
"dialogRemoveExtension": "Remove Extension",
|
"dialogRemoveExtension": "Remover Extensão",
|
||||||
"@dialogRemoveExtension": {
|
"@dialogRemoveExtension": {
|
||||||
"description": "Dialog title - uninstall extension"
|
"description": "Dialog title - uninstall extension"
|
||||||
},
|
},
|
||||||
"dialogRemoveExtensionMessage": "Are you sure you want to remove this extension? This cannot be undone.",
|
"dialogRemoveExtensionMessage": "Tem certeza de que deseja remover esta extensão? Isso não pode ser desfeito.",
|
||||||
"@dialogRemoveExtensionMessage": {
|
"@dialogRemoveExtensionMessage": {
|
||||||
"description": "Dialog message - uninstall confirmation"
|
"description": "Dialog message - uninstall confirmation"
|
||||||
},
|
},
|
||||||
"dialogUninstallExtension": "Uninstall Extension?",
|
"dialogUninstallExtension": "Desinstalar Extensão?",
|
||||||
"@dialogUninstallExtension": {
|
"@dialogUninstallExtension": {
|
||||||
"description": "Dialog title - uninstall extension"
|
"description": "Dialog title - uninstall extension"
|
||||||
},
|
},
|
||||||
"dialogUninstallExtensionMessage": "Are you sure you want to remove {extensionName}?",
|
"dialogUninstallExtensionMessage": "Tem certeza de que deseja remover {extensionName}?",
|
||||||
"@dialogUninstallExtensionMessage": {
|
"@dialogUninstallExtensionMessage": {
|
||||||
"description": "Dialog message - uninstall specific extension",
|
"description": "Dialog message - uninstall specific extension",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1096,19 +1096,19 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"dialogClearHistoryTitle": "Clear History",
|
"dialogClearHistoryTitle": "Limpar Histórico",
|
||||||
"@dialogClearHistoryTitle": {
|
"@dialogClearHistoryTitle": {
|
||||||
"description": "Dialog title - clear download history"
|
"description": "Dialog title - clear download history"
|
||||||
},
|
},
|
||||||
"dialogClearHistoryMessage": "Are you sure you want to clear all download history? This cannot be undone.",
|
"dialogClearHistoryMessage": "Tem certeza de que deseja limpar todo o histórico de downloads? Isso não pode ser desfeito.",
|
||||||
"@dialogClearHistoryMessage": {
|
"@dialogClearHistoryMessage": {
|
||||||
"description": "Dialog message - clear history confirmation"
|
"description": "Dialog message - clear history confirmation"
|
||||||
},
|
},
|
||||||
"dialogDeleteSelectedTitle": "Delete Selected",
|
"dialogDeleteSelectedTitle": "Apagar Selecionados",
|
||||||
"@dialogDeleteSelectedTitle": {
|
"@dialogDeleteSelectedTitle": {
|
||||||
"description": "Dialog title - delete selected items"
|
"description": "Dialog title - delete selected items"
|
||||||
},
|
},
|
||||||
"dialogDeleteSelectedMessage": "Delete {count} {count, plural, =1{track} other{tracks}} from history?\n\nThis will also delete the files from storage.",
|
"dialogDeleteSelectedMessage": "Apagar {count} {count, plural, =1{faixa} other{faixas}} do histórico?\n\nIsso também apagará os arquivos do armazenamento.",
|
||||||
"@dialogDeleteSelectedMessage": {
|
"@dialogDeleteSelectedMessage": {
|
||||||
"description": "Dialog message - delete selected tracks",
|
"description": "Dialog message - delete selected tracks",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1117,11 +1117,11 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"dialogImportPlaylistTitle": "Import Playlist",
|
"dialogImportPlaylistTitle": "Importar Playlist",
|
||||||
"@dialogImportPlaylistTitle": {
|
"@dialogImportPlaylistTitle": {
|
||||||
"description": "Dialog title - import CSV playlist"
|
"description": "Dialog title - import CSV playlist"
|
||||||
},
|
},
|
||||||
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
"dialogImportPlaylistMessage": "Encontradas {count} faixas no CSV. Adicionar à fila de download?",
|
||||||
"@dialogImportPlaylistMessage": {
|
"@dialogImportPlaylistMessage": {
|
||||||
"description": "Dialog message - import playlist confirmation",
|
"description": "Dialog message - import playlist confirmation",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1130,7 +1130,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"snackbarAddedToQueue": "Added \"{trackName}\" to queue",
|
"snackbarAddedToQueue": "\"{trackName}\" adicionada à fila",
|
||||||
"@snackbarAddedToQueue": {
|
"@snackbarAddedToQueue": {
|
||||||
"description": "Snackbar - track added to download queue",
|
"description": "Snackbar - track added to download queue",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1139,7 +1139,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"snackbarAddedTracksToQueue": "Added {count} tracks to queue",
|
"snackbarAddedTracksToQueue": "{count} faixas adicionadas à fila",
|
||||||
"@snackbarAddedTracksToQueue": {
|
"@snackbarAddedTracksToQueue": {
|
||||||
"description": "Snackbar - multiple tracks added to queue",
|
"description": "Snackbar - multiple tracks added to queue",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1148,7 +1148,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"snackbarAlreadyDownloaded": "\"{trackName}\" already downloaded",
|
"snackbarAlreadyDownloaded": "\"{trackName}\" já foi baixada",
|
||||||
"@snackbarAlreadyDownloaded": {
|
"@snackbarAlreadyDownloaded": {
|
||||||
"description": "Snackbar - track already exists",
|
"description": "Snackbar - track already exists",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1157,19 +1157,19 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"snackbarHistoryCleared": "History cleared",
|
"snackbarHistoryCleared": "Histórico limpo",
|
||||||
"@snackbarHistoryCleared": {
|
"@snackbarHistoryCleared": {
|
||||||
"description": "Snackbar - history deleted"
|
"description": "Snackbar - history deleted"
|
||||||
},
|
},
|
||||||
"snackbarCredentialsSaved": "Credentials saved",
|
"snackbarCredentialsSaved": "Credenciais salvas",
|
||||||
"@snackbarCredentialsSaved": {
|
"@snackbarCredentialsSaved": {
|
||||||
"description": "Snackbar - Spotify credentials saved"
|
"description": "Snackbar - Spotify credentials saved"
|
||||||
},
|
},
|
||||||
"snackbarCredentialsCleared": "Credentials cleared",
|
"snackbarCredentialsCleared": "Credenciais removidas",
|
||||||
"@snackbarCredentialsCleared": {
|
"@snackbarCredentialsCleared": {
|
||||||
"description": "Snackbar - Spotify credentials removed"
|
"description": "Snackbar - Spotify credentials removed"
|
||||||
},
|
},
|
||||||
"snackbarDeletedTracks": "Deleted {count} {count, plural, =1{track} other{tracks}}",
|
"snackbarDeletedTracks": "{count} {count, plural, =1{faixa apagada} other{faixas apagadas}}",
|
||||||
"@snackbarDeletedTracks": {
|
"@snackbarDeletedTracks": {
|
||||||
"description": "Snackbar - tracks deleted",
|
"description": "Snackbar - tracks deleted",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1178,7 +1178,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"snackbarCannotOpenFile": "Cannot open file: {error}",
|
"snackbarCannotOpenFile": "Não foi possível abrir o arquivo: {error}",
|
||||||
"@snackbarCannotOpenFile": {
|
"@snackbarCannotOpenFile": {
|
||||||
"description": "Snackbar - file open error",
|
"description": "Snackbar - file open error",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1187,15 +1187,15 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"snackbarFillAllFields": "Please fill all fields",
|
"snackbarFillAllFields": "Por favor, preencha todos os campos",
|
||||||
"@snackbarFillAllFields": {
|
"@snackbarFillAllFields": {
|
||||||
"description": "Snackbar - validation error"
|
"description": "Snackbar - validation error"
|
||||||
},
|
},
|
||||||
"snackbarViewQueue": "View Queue",
|
"snackbarViewQueue": "Ver Fila",
|
||||||
"@snackbarViewQueue": {
|
"@snackbarViewQueue": {
|
||||||
"description": "Snackbar action - view download queue"
|
"description": "Snackbar action - view download queue"
|
||||||
},
|
},
|
||||||
"snackbarFailedToLoad": "Failed to load: {error}",
|
"snackbarFailedToLoad": "Falha ao carregar: {error}",
|
||||||
"@snackbarFailedToLoad": {
|
"@snackbarFailedToLoad": {
|
||||||
"description": "Snackbar - loading error",
|
"description": "Snackbar - loading error",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1204,7 +1204,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"snackbarUrlCopied": "{platform} URL copied to clipboard",
|
"snackbarUrlCopied": "URL do {platform} copiada para a área de transferência",
|
||||||
"@snackbarUrlCopied": {
|
"@snackbarUrlCopied": {
|
||||||
"description": "Snackbar - URL copied",
|
"description": "Snackbar - URL copied",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1214,23 +1214,23 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"snackbarFileNotFound": "File not found",
|
"snackbarFileNotFound": "Arquivo não encontrado",
|
||||||
"@snackbarFileNotFound": {
|
"@snackbarFileNotFound": {
|
||||||
"description": "Snackbar - file doesn't exist"
|
"description": "Snackbar - file doesn't exist"
|
||||||
},
|
},
|
||||||
"snackbarSelectExtFile": "Please select a .spotiflac-ext file",
|
"snackbarSelectExtFile": "Por favor, selecione um arquivo .spotiflac-ext",
|
||||||
"@snackbarSelectExtFile": {
|
"@snackbarSelectExtFile": {
|
||||||
"description": "Snackbar - wrong file type selected"
|
"description": "Snackbar - wrong file type selected"
|
||||||
},
|
},
|
||||||
"snackbarProviderPrioritySaved": "Provider priority saved",
|
"snackbarProviderPrioritySaved": "Prioridade de provedor salva",
|
||||||
"@snackbarProviderPrioritySaved": {
|
"@snackbarProviderPrioritySaved": {
|
||||||
"description": "Snackbar - provider order saved"
|
"description": "Snackbar - provider order saved"
|
||||||
},
|
},
|
||||||
"snackbarMetadataProviderSaved": "Metadata provider priority saved",
|
"snackbarMetadataProviderSaved": "Prioridade de provedor de metadados salva",
|
||||||
"@snackbarMetadataProviderSaved": {
|
"@snackbarMetadataProviderSaved": {
|
||||||
"description": "Snackbar - metadata provider order saved"
|
"description": "Snackbar - metadata provider order saved"
|
||||||
},
|
},
|
||||||
"snackbarExtensionInstalled": "{extensionName} installed.",
|
"snackbarExtensionInstalled": "{extensionName} instalada.",
|
||||||
"@snackbarExtensionInstalled": {
|
"@snackbarExtensionInstalled": {
|
||||||
"description": "Snackbar - extension installed successfully",
|
"description": "Snackbar - extension installed successfully",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1239,7 +1239,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"snackbarExtensionUpdated": "{extensionName} updated.",
|
"snackbarExtensionUpdated": "{extensionName} atualizada.",
|
||||||
"@snackbarExtensionUpdated": {
|
"@snackbarExtensionUpdated": {
|
||||||
"description": "Snackbar - extension updated successfully",
|
"description": "Snackbar - extension updated successfully",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1248,23 +1248,23 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"snackbarFailedToInstall": "Failed to install extension",
|
"snackbarFailedToInstall": "Falha ao instalar extensão",
|
||||||
"@snackbarFailedToInstall": {
|
"@snackbarFailedToInstall": {
|
||||||
"description": "Snackbar - extension install error"
|
"description": "Snackbar - extension install error"
|
||||||
},
|
},
|
||||||
"snackbarFailedToUpdate": "Failed to update extension",
|
"snackbarFailedToUpdate": "Falha ao atualizar extensão",
|
||||||
"@snackbarFailedToUpdate": {
|
"@snackbarFailedToUpdate": {
|
||||||
"description": "Snackbar - extension update error"
|
"description": "Snackbar - extension update error"
|
||||||
},
|
},
|
||||||
"errorRateLimited": "Rate Limited",
|
"errorRateLimited": "Taxa Limitada",
|
||||||
"@errorRateLimited": {
|
"@errorRateLimited": {
|
||||||
"description": "Error title - too many requests"
|
"description": "Error title - too many requests"
|
||||||
},
|
},
|
||||||
"errorRateLimitedMessage": "Too many requests. Please wait a moment before searching again.",
|
"errorRateLimitedMessage": "Muitas solicitações. Por favor, aguarde um momento antes de pesquisar novamente.",
|
||||||
"@errorRateLimitedMessage": {
|
"@errorRateLimitedMessage": {
|
||||||
"description": "Error message - rate limit explanation"
|
"description": "Error message - rate limit explanation"
|
||||||
},
|
},
|
||||||
"errorFailedToLoad": "Failed to load {item}",
|
"errorFailedToLoad": "Falha ao carregar {item}",
|
||||||
"@errorFailedToLoad": {
|
"@errorFailedToLoad": {
|
||||||
"description": "Error message - loading failed",
|
"description": "Error message - loading failed",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1274,11 +1274,11 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"errorNoTracksFound": "No tracks found",
|
"errorNoTracksFound": "Nenhuma faixa encontrada",
|
||||||
"@errorNoTracksFound": {
|
"@errorNoTracksFound": {
|
||||||
"description": "Error - search returned no results"
|
"description": "Error - search returned no results"
|
||||||
},
|
},
|
||||||
"errorMissingExtensionSource": "Cannot load {item}: missing extension source",
|
"errorMissingExtensionSource": "Não foi possível carregar {item}: fonte de extensão ausente",
|
||||||
"@errorMissingExtensionSource": {
|
"@errorMissingExtensionSource": {
|
||||||
"description": "Error - extension source not available",
|
"description": "Error - extension source not available",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1287,23 +1287,23 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"statusQueued": "Queued",
|
"statusQueued": "Na Fila",
|
||||||
"@statusQueued": {
|
"@statusQueued": {
|
||||||
"description": "Download status - waiting in queue"
|
"description": "Download status - waiting in queue"
|
||||||
},
|
},
|
||||||
"statusDownloading": "Downloading",
|
"statusDownloading": "Baixando",
|
||||||
"@statusDownloading": {
|
"@statusDownloading": {
|
||||||
"description": "Download status - in progress"
|
"description": "Download status - in progress"
|
||||||
},
|
},
|
||||||
"statusFinalizing": "Finalizing",
|
"statusFinalizing": "Finalizando",
|
||||||
"@statusFinalizing": {
|
"@statusFinalizing": {
|
||||||
"description": "Download status - writing metadata"
|
"description": "Download status - writing metadata"
|
||||||
},
|
},
|
||||||
"statusCompleted": "Completed",
|
"statusCompleted": "Concluído",
|
||||||
"@statusCompleted": {
|
"@statusCompleted": {
|
||||||
"description": "Download status - finished"
|
"description": "Download status - finished"
|
||||||
},
|
},
|
||||||
"statusFailed": "Failed",
|
"statusFailed": "Falhou",
|
||||||
"@statusFailed": {
|
"@statusFailed": {
|
||||||
"description": "Download status - error occurred"
|
"description": "Download status - error occurred"
|
||||||
},
|
},
|
||||||
@@ -1735,19 +1735,19 @@
|
|||||||
"@logNetworkErrorDescription": {
|
"@logNetworkErrorDescription": {
|
||||||
"description": "Network error explanation"
|
"description": "Network error explanation"
|
||||||
},
|
},
|
||||||
"logNetworkErrorSuggestion": "Check your internet connection",
|
"logNetworkErrorSuggestion": "Verifique a sua conexão com a internet",
|
||||||
"@logNetworkErrorSuggestion": {
|
"@logNetworkErrorSuggestion": {
|
||||||
"description": "Network error fix suggestion"
|
"description": "Network error fix suggestion"
|
||||||
},
|
},
|
||||||
"logTrackNotFoundDescription": "Some tracks could not be found on download services",
|
"logTrackNotFoundDescription": "Algumas faixas não foram encontradas nos serviços de download",
|
||||||
"@logTrackNotFoundDescription": {
|
"@logTrackNotFoundDescription": {
|
||||||
"description": "Track not found explanation"
|
"description": "Track not found explanation"
|
||||||
},
|
},
|
||||||
"logTrackNotFoundSuggestion": "The track may not be available in lossless quality",
|
"logTrackNotFoundSuggestion": "A faixa pode não estar disponível em qualidade lossless",
|
||||||
"@logTrackNotFoundSuggestion": {
|
"@logTrackNotFoundSuggestion": {
|
||||||
"description": "Track not found explanation"
|
"description": "Track not found explanation"
|
||||||
},
|
},
|
||||||
"logTotalErrors": "Total errors: {count}",
|
"logTotalErrors": "Total de erros: {count}",
|
||||||
"@logTotalErrors": {
|
"@logTotalErrors": {
|
||||||
"description": "Error count display",
|
"description": "Error count display",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1756,7 +1756,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"logAffected": "Affected: {domains}",
|
"logAffected": "Afetados: {domains}",
|
||||||
"@logAffected": {
|
"@logAffected": {
|
||||||
"description": "Affected domains display",
|
"description": "Affected domains display",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1765,7 +1765,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"logEntriesFiltered": "Entries ({count} filtered)",
|
"logEntriesFiltered": "Entradas ({count} filtradas)",
|
||||||
"@logEntriesFiltered": {
|
"@logEntriesFiltered": {
|
||||||
"description": "Log count with filter active",
|
"description": "Log count with filter active",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1774,7 +1774,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"logEntries": "Entries ({count})",
|
"logEntries": "Entradas ({count})",
|
||||||
"@logEntries": {
|
"@logEntries": {
|
||||||
"description": "Total log count",
|
"description": "Total log count",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1783,11 +1783,11 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"credentialsTitle": "Spotify Credentials",
|
"credentialsTitle": "Credenciais do Spotify",
|
||||||
"@credentialsTitle": {
|
"@credentialsTitle": {
|
||||||
"description": "Credentials dialog title"
|
"description": "Credentials dialog title"
|
||||||
},
|
},
|
||||||
"credentialsDescription": "Enter your Client ID and Secret to use your own Spotify application quota.",
|
"credentialsDescription": "Insira o seu Client ID e Secret para usar a sua própria cota de aplicativo do Spotify.",
|
||||||
"@credentialsDescription": {
|
"@credentialsDescription": {
|
||||||
"description": "Credentials dialog explanation"
|
"description": "Credentials dialog explanation"
|
||||||
},
|
},
|
||||||
@@ -2001,35 +2001,35 @@
|
|||||||
"@trackDownloaded": {
|
"@trackDownloaded": {
|
||||||
"description": "Metadata label - download date"
|
"description": "Metadata label - download date"
|
||||||
},
|
},
|
||||||
"trackCopyLyrics": "Copy lyrics",
|
"trackCopyLyrics": "Copiar letras",
|
||||||
"@trackCopyLyrics": {
|
"@trackCopyLyrics": {
|
||||||
"description": "Action - copy lyrics to clipboard"
|
"description": "Action - copy lyrics to clipboard"
|
||||||
},
|
},
|
||||||
"trackLyricsNotAvailable": "Lyrics not available for this track",
|
"trackLyricsNotAvailable": "Letras não disponíveis para esta faixa",
|
||||||
"@trackLyricsNotAvailable": {
|
"@trackLyricsNotAvailable": {
|
||||||
"description": "Message when lyrics not found"
|
"description": "Message when lyrics not found"
|
||||||
},
|
},
|
||||||
"trackLyricsTimeout": "Request timed out. Try again later.",
|
"trackLyricsTimeout": "A solicitação expirou. Tente novamente mais tarde.",
|
||||||
"@trackLyricsTimeout": {
|
"@trackLyricsTimeout": {
|
||||||
"description": "Message when lyrics request times out"
|
"description": "Message when lyrics request times out"
|
||||||
},
|
},
|
||||||
"trackLyricsLoadFailed": "Failed to load lyrics",
|
"trackLyricsLoadFailed": "Falha ao carregar letras",
|
||||||
"@trackLyricsLoadFailed": {
|
"@trackLyricsLoadFailed": {
|
||||||
"description": "Message when lyrics loading fails"
|
"description": "Message when lyrics loading fails"
|
||||||
},
|
},
|
||||||
"trackCopiedToClipboard": "Copied to clipboard",
|
"trackCopiedToClipboard": "Copiado para a área de transferência",
|
||||||
"@trackCopiedToClipboard": {
|
"@trackCopiedToClipboard": {
|
||||||
"description": "Snackbar - content copied"
|
"description": "Snackbar - content copied"
|
||||||
},
|
},
|
||||||
"trackDeleteConfirmTitle": "Remove from device?",
|
"trackDeleteConfirmTitle": "Remover do dispositivo?",
|
||||||
"@trackDeleteConfirmTitle": {
|
"@trackDeleteConfirmTitle": {
|
||||||
"description": "Delete confirmation title"
|
"description": "Delete confirmation title"
|
||||||
},
|
},
|
||||||
"trackDeleteConfirmMessage": "This will permanently delete the downloaded file and remove it from your history.",
|
"trackDeleteConfirmMessage": "Isso apagará permanentemente o arquivo baixado e o removerá do seu histórico.",
|
||||||
"@trackDeleteConfirmMessage": {
|
"@trackDeleteConfirmMessage": {
|
||||||
"description": "Delete confirmation message"
|
"description": "Delete confirmation message"
|
||||||
},
|
},
|
||||||
"trackCannotOpen": "Cannot open: {message}",
|
"trackCannotOpen": "Não foi possível abrir: {message}",
|
||||||
"@trackCannotOpen": {
|
"@trackCannotOpen": {
|
||||||
"description": "Error opening file",
|
"description": "Error opening file",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -2038,15 +2038,15 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"dateToday": "Today",
|
"dateToday": "Hoje",
|
||||||
"@dateToday": {
|
"@dateToday": {
|
||||||
"description": "Relative date - today"
|
"description": "Relative date - today"
|
||||||
},
|
},
|
||||||
"dateYesterday": "Yesterday",
|
"dateYesterday": "Ontem",
|
||||||
"@dateYesterday": {
|
"@dateYesterday": {
|
||||||
"description": "Relative date - yesterday"
|
"description": "Relative date - yesterday"
|
||||||
},
|
},
|
||||||
"dateDaysAgo": "{count} days ago",
|
"dateDaysAgo": "Há {count} dias",
|
||||||
"@dateDaysAgo": {
|
"@dateDaysAgo": {
|
||||||
"description": "Relative date - days ago",
|
"description": "Relative date - days ago",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -2055,7 +2055,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"dateWeeksAgo": "{count} weeks ago",
|
"dateWeeksAgo": "Há {count} semanas",
|
||||||
"@dateWeeksAgo": {
|
"@dateWeeksAgo": {
|
||||||
"description": "Relative date - weeks ago",
|
"description": "Relative date - weeks ago",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -2064,7 +2064,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"dateMonthsAgo": "{count} months ago",
|
"dateMonthsAgo": "Há {count} meses",
|
||||||
"@dateMonthsAgo": {
|
"@dateMonthsAgo": {
|
||||||
"description": "Relative date - months ago",
|
"description": "Relative date - months ago",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -2073,27 +2073,27 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"concurrentSequential": "Sequential",
|
"concurrentSequential": "Sequencial",
|
||||||
"@concurrentSequential": {
|
"@concurrentSequential": {
|
||||||
"description": "Download mode - one at a time"
|
"description": "Download mode - one at a time"
|
||||||
},
|
},
|
||||||
"concurrentParallel2": "2 Parallel",
|
"concurrentParallel2": "2 Paralelos",
|
||||||
"@concurrentParallel2": {
|
"@concurrentParallel2": {
|
||||||
"description": "Download mode - 2 simultaneous"
|
"description": "Download mode - 2 simultaneous"
|
||||||
},
|
},
|
||||||
"concurrentParallel3": "3 Parallel",
|
"concurrentParallel3": "3 Paralelos",
|
||||||
"@concurrentParallel3": {
|
"@concurrentParallel3": {
|
||||||
"description": "Download mode - 3 simultaneous"
|
"description": "Download mode - 3 simultaneous"
|
||||||
},
|
},
|
||||||
"tapToSeeError": "Tap to see error details",
|
"tapToSeeError": "Toque para ver detalhes do erro",
|
||||||
"@tapToSeeError": {
|
"@tapToSeeError": {
|
||||||
"description": "Tooltip for failed download"
|
"description": "Tooltip for failed download"
|
||||||
},
|
},
|
||||||
"storeFilterAll": "All",
|
"storeFilterAll": "Todos",
|
||||||
"@storeFilterAll": {
|
"@storeFilterAll": {
|
||||||
"description": "Store filter - all extensions"
|
"description": "Store filter - all extensions"
|
||||||
},
|
},
|
||||||
"storeFilterMetadata": "Metadata",
|
"storeFilterMetadata": "Metadados",
|
||||||
"@storeFilterMetadata": {
|
"@storeFilterMetadata": {
|
||||||
"description": "Store filter - metadata providers"
|
"description": "Store filter - metadata providers"
|
||||||
},
|
},
|
||||||
@@ -2101,43 +2101,43 @@
|
|||||||
"@storeFilterDownload": {
|
"@storeFilterDownload": {
|
||||||
"description": "Store filter - download providers"
|
"description": "Store filter - download providers"
|
||||||
},
|
},
|
||||||
"storeFilterUtility": "Utility",
|
"storeFilterUtility": "Utilitário",
|
||||||
"@storeFilterUtility": {
|
"@storeFilterUtility": {
|
||||||
"description": "Store filter - utility extensions"
|
"description": "Store filter - utility extensions"
|
||||||
},
|
},
|
||||||
"storeFilterLyrics": "Lyrics",
|
"storeFilterLyrics": "Letras",
|
||||||
"@storeFilterLyrics": {
|
"@storeFilterLyrics": {
|
||||||
"description": "Store filter - lyrics providers"
|
"description": "Store filter - lyrics providers"
|
||||||
},
|
},
|
||||||
"storeFilterIntegration": "Integration",
|
"storeFilterIntegration": "Integração",
|
||||||
"@storeFilterIntegration": {
|
"@storeFilterIntegration": {
|
||||||
"description": "Store filter - integrations"
|
"description": "Store filter - integrations"
|
||||||
},
|
},
|
||||||
"storeClearFilters": "Clear filters",
|
"storeClearFilters": "Limpar filtros",
|
||||||
"@storeClearFilters": {
|
"@storeClearFilters": {
|
||||||
"description": "Button to clear all filters"
|
"description": "Button to clear all filters"
|
||||||
},
|
},
|
||||||
"storeNoResults": "No extensions found",
|
"storeNoResults": "Nenhuma extensão encontrada",
|
||||||
"@storeNoResults": {
|
"@storeNoResults": {
|
||||||
"description": "Empty state when no extensions match filters"
|
"description": "Empty state when no extensions match filters"
|
||||||
},
|
},
|
||||||
"extensionProviderPriority": "Provider Priority",
|
"extensionProviderPriority": "Prioridade de Provedor",
|
||||||
"@extensionProviderPriority": {
|
"@extensionProviderPriority": {
|
||||||
"description": "Extension capability - provider priority"
|
"description": "Extension capability - provider priority"
|
||||||
},
|
},
|
||||||
"extensionInstallButton": "Install Extension",
|
"extensionInstallButton": "Instalar Extensão",
|
||||||
"@extensionInstallButton": {
|
"@extensionInstallButton": {
|
||||||
"description": "Button to install extension"
|
"description": "Button to install extension"
|
||||||
},
|
},
|
||||||
"extensionDefaultProvider": "Default (Deezer/Spotify)",
|
"extensionDefaultProvider": "Padrão (Deezer/Spotify)",
|
||||||
"@extensionDefaultProvider": {
|
"@extensionDefaultProvider": {
|
||||||
"description": "Default search provider option"
|
"description": "Default search provider option"
|
||||||
},
|
},
|
||||||
"extensionDefaultProviderSubtitle": "Use built-in search",
|
"extensionDefaultProviderSubtitle": "Usar pesquisa integrada",
|
||||||
"@extensionDefaultProviderSubtitle": {
|
"@extensionDefaultProviderSubtitle": {
|
||||||
"description": "Subtitle for default provider"
|
"description": "Subtitle for default provider"
|
||||||
},
|
},
|
||||||
"extensionAuthor": "Author",
|
"extensionAuthor": "Autor",
|
||||||
"@extensionAuthor": {
|
"@extensionAuthor": {
|
||||||
"description": "Extension detail - author"
|
"description": "Extension detail - author"
|
||||||
},
|
},
|
||||||
@@ -2145,43 +2145,43 @@
|
|||||||
"@extensionId": {
|
"@extensionId": {
|
||||||
"description": "Extension detail - unique ID"
|
"description": "Extension detail - unique ID"
|
||||||
},
|
},
|
||||||
"extensionError": "Error",
|
"extensionError": "Erro",
|
||||||
"@extensionError": {
|
"@extensionError": {
|
||||||
"description": "Extension detail - error message"
|
"description": "Extension detail - error message"
|
||||||
},
|
},
|
||||||
"extensionCapabilities": "Capabilities",
|
"extensionCapabilities": "Capacidades",
|
||||||
"@extensionCapabilities": {
|
"@extensionCapabilities": {
|
||||||
"description": "Section header - extension features"
|
"description": "Section header - extension features"
|
||||||
},
|
},
|
||||||
"extensionMetadataProvider": "Metadata Provider",
|
"extensionMetadataProvider": "Provedor de Metadados",
|
||||||
"@extensionMetadataProvider": {
|
"@extensionMetadataProvider": {
|
||||||
"description": "Capability - provides metadata"
|
"description": "Capability - provides metadata"
|
||||||
},
|
},
|
||||||
"extensionDownloadProvider": "Download Provider",
|
"extensionDownloadProvider": "Provedor de Download",
|
||||||
"@extensionDownloadProvider": {
|
"@extensionDownloadProvider": {
|
||||||
"description": "Capability - provides downloads"
|
"description": "Capability - provides downloads"
|
||||||
},
|
},
|
||||||
"extensionLyricsProvider": "Lyrics Provider",
|
"extensionLyricsProvider": "Provedor de Letras",
|
||||||
"@extensionLyricsProvider": {
|
"@extensionLyricsProvider": {
|
||||||
"description": "Capability - provides lyrics"
|
"description": "Capability - provides lyrics"
|
||||||
},
|
},
|
||||||
"extensionUrlHandler": "URL Handler",
|
"extensionUrlHandler": "Manipulador de URL",
|
||||||
"@extensionUrlHandler": {
|
"@extensionUrlHandler": {
|
||||||
"description": "Capability - handles URLs"
|
"description": "Capability - handles URLs"
|
||||||
},
|
},
|
||||||
"extensionQualityOptions": "Quality Options",
|
"extensionQualityOptions": "Opções de Qualidade",
|
||||||
"@extensionQualityOptions": {
|
"@extensionQualityOptions": {
|
||||||
"description": "Capability - quality selection"
|
"description": "Capability - quality selection"
|
||||||
},
|
},
|
||||||
"extensionPostProcessingHooks": "Post-Processing Hooks",
|
"extensionPostProcessingHooks": "Ganchos de Pós-Processamento",
|
||||||
"@extensionPostProcessingHooks": {
|
"@extensionPostProcessingHooks": {
|
||||||
"description": "Capability - post-processing"
|
"description": "Capability - post-processing"
|
||||||
},
|
},
|
||||||
"extensionPermissions": "Permissions",
|
"extensionPermissions": "Permissões",
|
||||||
"@extensionPermissions": {
|
"@extensionPermissions": {
|
||||||
"description": "Section header - required permissions"
|
"description": "Section header - required permissions"
|
||||||
},
|
},
|
||||||
"extensionSettings": "Settings",
|
"extensionSettings": "Configurações",
|
||||||
"@extensionSettings": {
|
"@extensionSettings": {
|
||||||
"description": "Section header - extension settings"
|
"description": "Section header - extension settings"
|
||||||
},
|
},
|
||||||
@@ -2376,31 +2376,31 @@
|
|||||||
"@folderNone": {
|
"@folderNone": {
|
||||||
"description": "Folder option - no organization"
|
"description": "Folder option - no organization"
|
||||||
},
|
},
|
||||||
"folderNoneSubtitle": "Save all files directly to download folder",
|
"folderNoneSubtitle": "Salvar todos os arquivos diretamente na pasta de download",
|
||||||
"@folderNoneSubtitle": {
|
"@folderNoneSubtitle": {
|
||||||
"description": "Subtitle for no folder organization"
|
"description": "Subtitle for no folder organization"
|
||||||
},
|
},
|
||||||
"folderArtist": "Artist",
|
"folderArtist": "Artista",
|
||||||
"@folderArtist": {
|
"@folderArtist": {
|
||||||
"description": "Folder option - by artist"
|
"description": "Folder option - by artist"
|
||||||
},
|
},
|
||||||
"folderArtistSubtitle": "Artist Name/filename",
|
"folderArtistSubtitle": "Nome do Artista/nome do arquivo",
|
||||||
"@folderArtistSubtitle": {
|
"@folderArtistSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
"folderAlbum": "Album",
|
"folderAlbum": "Álbum",
|
||||||
"@folderAlbum": {
|
"@folderAlbum": {
|
||||||
"description": "Folder option - by album"
|
"description": "Folder option - by album"
|
||||||
},
|
},
|
||||||
"folderAlbumSubtitle": "Album Name/filename",
|
"folderAlbumSubtitle": "Nome do Álbum/nome do arquivo",
|
||||||
"@folderAlbumSubtitle": {
|
"@folderAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
"folderArtistAlbum": "Artist/Album",
|
"folderArtistAlbum": "Artista/Álbum",
|
||||||
"@folderArtistAlbum": {
|
"@folderArtistAlbum": {
|
||||||
"description": "Folder option - nested"
|
"description": "Folder option - nested"
|
||||||
},
|
},
|
||||||
"folderArtistAlbumSubtitle": "Artist Name/Album Name/filename",
|
"folderArtistAlbumSubtitle": "Nome do Artista/Nome do Álbum/nome do arquivo",
|
||||||
"@folderArtistAlbumSubtitle": {
|
"@folderArtistAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
@@ -2424,103 +2424,103 @@
|
|||||||
"@serviceSpotify": {
|
"@serviceSpotify": {
|
||||||
"description": "Service name - DO NOT TRANSLATE"
|
"description": "Service name - DO NOT TRANSLATE"
|
||||||
},
|
},
|
||||||
"appearanceAmoledDark": "AMOLED Dark",
|
"appearanceAmoledDark": "AMOLED Escuro",
|
||||||
"@appearanceAmoledDark": {
|
"@appearanceAmoledDark": {
|
||||||
"description": "Theme option - pure black"
|
"description": "Theme option - pure black"
|
||||||
},
|
},
|
||||||
"appearanceAmoledDarkSubtitle": "Pure black background",
|
"appearanceAmoledDarkSubtitle": "Fundo preto puro",
|
||||||
"@appearanceAmoledDarkSubtitle": {
|
"@appearanceAmoledDarkSubtitle": {
|
||||||
"description": "Subtitle for AMOLED dark"
|
"description": "Subtitle for AMOLED dark"
|
||||||
},
|
},
|
||||||
"appearanceChooseAccentColor": "Choose Accent Color",
|
"appearanceChooseAccentColor": "Escolher Cor de Destaque",
|
||||||
"@appearanceChooseAccentColor": {
|
"@appearanceChooseAccentColor": {
|
||||||
"description": "Color picker dialog title"
|
"description": "Color picker dialog title"
|
||||||
},
|
},
|
||||||
"appearanceChooseTheme": "Theme Mode",
|
"appearanceChooseTheme": "Modo de Tema",
|
||||||
"@appearanceChooseTheme": {
|
"@appearanceChooseTheme": {
|
||||||
"description": "Theme picker dialog title"
|
"description": "Theme picker dialog title"
|
||||||
},
|
},
|
||||||
"queueTitle": "Download Queue",
|
"queueTitle": "Fila de Download",
|
||||||
"@queueTitle": {
|
"@queueTitle": {
|
||||||
"description": "Queue screen title"
|
"description": "Queue screen title"
|
||||||
},
|
},
|
||||||
"queueClearAll": "Clear All",
|
"queueClearAll": "Limpar Tudo",
|
||||||
"@queueClearAll": {
|
"@queueClearAll": {
|
||||||
"description": "Button - clear all queue items"
|
"description": "Button - clear all queue items"
|
||||||
},
|
},
|
||||||
"queueClearAllMessage": "Are you sure you want to clear all downloads?",
|
"queueClearAllMessage": "Tem certeza de que deseja limpar todos os downloads?",
|
||||||
"@queueClearAllMessage": {
|
"@queueClearAllMessage": {
|
||||||
"description": "Clear queue confirmation"
|
"description": "Clear queue confirmation"
|
||||||
},
|
},
|
||||||
"queueEmpty": "No downloads in queue",
|
"queueEmpty": "Nenhum download na fila",
|
||||||
"@queueEmpty": {
|
"@queueEmpty": {
|
||||||
"description": "Empty queue state title"
|
"description": "Empty queue state title"
|
||||||
},
|
},
|
||||||
"queueEmptySubtitle": "Add tracks from the home screen",
|
"queueEmptySubtitle": "Adicione faixas a partir da tela inicial",
|
||||||
"@queueEmptySubtitle": {
|
"@queueEmptySubtitle": {
|
||||||
"description": "Empty queue state subtitle"
|
"description": "Empty queue state subtitle"
|
||||||
},
|
},
|
||||||
"queueClearCompleted": "Clear completed",
|
"queueClearCompleted": "Limpar concluídos",
|
||||||
"@queueClearCompleted": {
|
"@queueClearCompleted": {
|
||||||
"description": "Button - clear finished downloads"
|
"description": "Button - clear finished downloads"
|
||||||
},
|
},
|
||||||
"queueDownloadFailed": "Download Failed",
|
"queueDownloadFailed": "Download Falhou",
|
||||||
"@queueDownloadFailed": {
|
"@queueDownloadFailed": {
|
||||||
"description": "Error dialog title"
|
"description": "Error dialog title"
|
||||||
},
|
},
|
||||||
"queueTrackLabel": "Track:",
|
"queueTrackLabel": "Faixa:",
|
||||||
"@queueTrackLabel": {
|
"@queueTrackLabel": {
|
||||||
"description": "Label in error dialog"
|
"description": "Label in error dialog"
|
||||||
},
|
},
|
||||||
"queueArtistLabel": "Artist:",
|
"queueArtistLabel": "Artista:",
|
||||||
"@queueArtistLabel": {
|
"@queueArtistLabel": {
|
||||||
"description": "Label in error dialog"
|
"description": "Label in error dialog"
|
||||||
},
|
},
|
||||||
"queueErrorLabel": "Error:",
|
"queueErrorLabel": "Erro:",
|
||||||
"@queueErrorLabel": {
|
"@queueErrorLabel": {
|
||||||
"description": "Label in error dialog"
|
"description": "Label in error dialog"
|
||||||
},
|
},
|
||||||
"queueUnknownError": "Unknown error",
|
"queueUnknownError": "Erro desconhecido",
|
||||||
"@queueUnknownError": {
|
"@queueUnknownError": {
|
||||||
"description": "Fallback error message"
|
"description": "Fallback error message"
|
||||||
},
|
},
|
||||||
"albumFolderArtistAlbum": "Artist / Album",
|
"albumFolderArtistAlbum": "Artista / Álbum",
|
||||||
"@albumFolderArtistAlbum": {
|
"@albumFolderArtistAlbum": {
|
||||||
"description": "Album folder option"
|
"description": "Album folder option"
|
||||||
},
|
},
|
||||||
"albumFolderArtistAlbumSubtitle": "Albums/Artist Name/Album Name/",
|
"albumFolderArtistAlbumSubtitle": "Álbuns/Nome do Artista/Nome do Álbum/",
|
||||||
"@albumFolderArtistAlbumSubtitle": {
|
"@albumFolderArtistAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
"albumFolderArtistYearAlbum": "Artist / [Year] Album",
|
"albumFolderArtistYearAlbum": "Artista / [Ano] Álbum",
|
||||||
"@albumFolderArtistYearAlbum": {
|
"@albumFolderArtistYearAlbum": {
|
||||||
"description": "Album folder option with year"
|
"description": "Album folder option with year"
|
||||||
},
|
},
|
||||||
"albumFolderArtistYearAlbumSubtitle": "Albums/Artist Name/[2005] Album Name/",
|
"albumFolderArtistYearAlbumSubtitle": "Álbuns/Nome do Artista/[2005] Nome do Álbum/",
|
||||||
"@albumFolderArtistYearAlbumSubtitle": {
|
"@albumFolderArtistYearAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
"albumFolderAlbumOnly": "Album Only",
|
"albumFolderAlbumOnly": "Apenas Álbum",
|
||||||
"@albumFolderAlbumOnly": {
|
"@albumFolderAlbumOnly": {
|
||||||
"description": "Album folder option"
|
"description": "Album folder option"
|
||||||
},
|
},
|
||||||
"albumFolderAlbumOnlySubtitle": "Albums/Album Name/",
|
"albumFolderAlbumOnlySubtitle": "Álbuns/Nome do Álbum/",
|
||||||
"@albumFolderAlbumOnlySubtitle": {
|
"@albumFolderAlbumOnlySubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
"albumFolderYearAlbum": "[Year] Album",
|
"albumFolderYearAlbum": "[Ano] Álbum",
|
||||||
"@albumFolderYearAlbum": {
|
"@albumFolderYearAlbum": {
|
||||||
"description": "Album folder option with year"
|
"description": "Album folder option with year"
|
||||||
},
|
},
|
||||||
"albumFolderYearAlbumSubtitle": "Albums/[2005] Album Name/",
|
"albumFolderYearAlbumSubtitle": "Álbuns/[2005] Nome do Álbum/",
|
||||||
"@albumFolderYearAlbumSubtitle": {
|
"@albumFolderYearAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
"downloadedAlbumDeleteSelected": "Delete Selected",
|
"downloadedAlbumDeleteSelected": "Apagar Selecionados",
|
||||||
"@downloadedAlbumDeleteSelected": {
|
"@downloadedAlbumDeleteSelected": {
|
||||||
"description": "Button - delete selected tracks"
|
"description": "Button - delete selected tracks"
|
||||||
},
|
},
|
||||||
"downloadedAlbumDeleteMessage": "Delete {count} {count, plural, =1{track} other{tracks}} from this album?\n\nThis will also delete the files from storage.",
|
"downloadedAlbumDeleteMessage": "Apagar {count} {count, plural, =1{faixa} other{faixas}} deste álbum?\n\nIsso também apagará os arquivos do armazenamento.",
|
||||||
"@downloadedAlbumDeleteMessage": {
|
"@downloadedAlbumDeleteMessage": {
|
||||||
"description": "Delete confirmation with count",
|
"description": "Delete confirmation with count",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -2529,11 +2529,11 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"downloadedAlbumTracksHeader": "Tracks",
|
"downloadedAlbumTracksHeader": "Faixas",
|
||||||
"@downloadedAlbumTracksHeader": {
|
"@downloadedAlbumTracksHeader": {
|
||||||
"description": "Section header for tracks"
|
"description": "Section header for tracks"
|
||||||
},
|
},
|
||||||
"downloadedAlbumDownloadedCount": "{count} downloaded",
|
"downloadedAlbumDownloadedCount": "{count} baixadas",
|
||||||
"@downloadedAlbumDownloadedCount": {
|
"@downloadedAlbumDownloadedCount": {
|
||||||
"description": "Downloaded tracks count badge",
|
"description": "Downloaded tracks count badge",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -2542,7 +2542,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"downloadedAlbumSelectedCount": "{count} selected",
|
"downloadedAlbumSelectedCount": "{count} selecionadas",
|
||||||
"@downloadedAlbumSelectedCount": {
|
"@downloadedAlbumSelectedCount": {
|
||||||
"description": "Selection count indicator",
|
"description": "Selection count indicator",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -2551,15 +2551,15 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"downloadedAlbumAllSelected": "All tracks selected",
|
"downloadedAlbumAllSelected": "Todas as faixas selecionadas",
|
||||||
"@downloadedAlbumAllSelected": {
|
"@downloadedAlbumAllSelected": {
|
||||||
"description": "Status - all items selected"
|
"description": "Status - all items selected"
|
||||||
},
|
},
|
||||||
"downloadedAlbumTapToSelect": "Tap tracks to select",
|
"downloadedAlbumTapToSelect": "Toque nas faixas para selecionar",
|
||||||
"@downloadedAlbumTapToSelect": {
|
"@downloadedAlbumTapToSelect": {
|
||||||
"description": "Selection hint"
|
"description": "Selection hint"
|
||||||
},
|
},
|
||||||
"downloadedAlbumDeleteCount": "Delete {count} {count, plural, =1{track} other{tracks}}",
|
"downloadedAlbumDeleteCount": "Apagar {count} {count, plural, =1{faixa} other{faixas}}",
|
||||||
"@downloadedAlbumDeleteCount": {
|
"@downloadedAlbumDeleteCount": {
|
||||||
"description": "Delete button text with count",
|
"description": "Delete button text with count",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -2568,23 +2568,23 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"downloadedAlbumSelectToDelete": "Select tracks to delete",
|
"downloadedAlbumSelectToDelete": "Selecione faixas para apagar",
|
||||||
"@downloadedAlbumSelectToDelete": {
|
"@downloadedAlbumSelectToDelete": {
|
||||||
"description": "Placeholder when nothing selected"
|
"description": "Placeholder when nothing selected"
|
||||||
},
|
},
|
||||||
"utilityFunctions": "Utility Functions",
|
"utilityFunctions": "Funções Utilitárias",
|
||||||
"@utilityFunctions": {
|
"@utilityFunctions": {
|
||||||
"description": "Extension capability - utility functions"
|
"description": "Extension capability - utility functions"
|
||||||
},
|
},
|
||||||
"recentTypeArtist": "Artist",
|
"recentTypeArtist": "Artista",
|
||||||
"@recentTypeArtist": {
|
"@recentTypeArtist": {
|
||||||
"description": "Recent access item type - artist"
|
"description": "Recent access item type - artist"
|
||||||
},
|
},
|
||||||
"recentTypeAlbum": "Album",
|
"recentTypeAlbum": "Álbum",
|
||||||
"@recentTypeAlbum": {
|
"@recentTypeAlbum": {
|
||||||
"description": "Recent access item type - album"
|
"description": "Recent access item type - album"
|
||||||
},
|
},
|
||||||
"recentTypeSong": "Song",
|
"recentTypeSong": "Música",
|
||||||
"@recentTypeSong": {
|
"@recentTypeSong": {
|
||||||
"description": "Recent access item type - song/track"
|
"description": "Recent access item type - song/track"
|
||||||
},
|
},
|
||||||
@@ -2602,7 +2602,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"errorGeneric": "Error: {message}",
|
"errorGeneric": "Erro: {message}",
|
||||||
"@errorGeneric": {
|
"@errorGeneric": {
|
||||||
"description": "Generic error message format",
|
"description": "Generic error message format",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
|
|||||||
+263
-10
@@ -85,7 +85,7 @@
|
|||||||
"@historyFilterSingles": {
|
"@historyFilterSingles": {
|
||||||
"description": "Filter chip - show singles only"
|
"description": "Filter chip - show singles only"
|
||||||
},
|
},
|
||||||
"historyTracksCount": "{count, plural, one {{count} трек} few {{count} трека} many {{count} треков} other {{count} треков}}",
|
"historyTracksCount": "{count, plural, one {{count} трек} few {{count} трека} many {{count} треков} =1 {1 трек} other {{count} треков}}",
|
||||||
"@historyTracksCount": {
|
"@historyTracksCount": {
|
||||||
"description": "Track count with plural form",
|
"description": "Track count with plural form",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -94,7 +94,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"historyAlbumsCount": "{count, plural, one {{count} альбом} few {{count} альбома} many {{count} альбомов} other {{count} альбомов}}",
|
"historyAlbumsCount": "{count, plural, one {{count} альбом} few {{count} альбома} many {{count} альбомов} =1 {1 альбом} other {{count} альбомов}}",
|
||||||
"@historyAlbumsCount": {
|
"@historyAlbumsCount": {
|
||||||
"description": "Album count with plural form",
|
"description": "Album count with plural form",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -127,6 +127,10 @@
|
|||||||
"@historyNoSinglesSubtitle": {
|
"@historyNoSinglesSubtitle": {
|
||||||
"description": "Empty state subtitle for singles filter"
|
"description": "Empty state subtitle for singles filter"
|
||||||
},
|
},
|
||||||
|
"historySearchHint": "Поиск в истории...",
|
||||||
|
"@historySearchHint": {
|
||||||
|
"description": "Search bar placeholder in history"
|
||||||
|
},
|
||||||
"settingsTitle": "Настройки",
|
"settingsTitle": "Настройки",
|
||||||
"@settingsTitle": {
|
"@settingsTitle": {
|
||||||
"description": "Settings screen title"
|
"description": "Settings screen title"
|
||||||
@@ -512,6 +516,10 @@
|
|||||||
"@aboutLogoArtist": {
|
"@aboutLogoArtist": {
|
||||||
"description": "Role description for logo artist"
|
"description": "Role description for logo artist"
|
||||||
},
|
},
|
||||||
|
"aboutTranslators": "Переводчики",
|
||||||
|
"@aboutTranslators": {
|
||||||
|
"description": "Section for translators"
|
||||||
|
},
|
||||||
"aboutSpecialThanks": "Особая благодарность",
|
"aboutSpecialThanks": "Особая благодарность",
|
||||||
"@aboutSpecialThanks": {
|
"@aboutSpecialThanks": {
|
||||||
"description": "Section for special thanks"
|
"description": "Section for special thanks"
|
||||||
@@ -544,6 +552,26 @@
|
|||||||
"@aboutFeatureRequestSubtitle": {
|
"@aboutFeatureRequestSubtitle": {
|
||||||
"description": "Subtitle for feature request"
|
"description": "Subtitle for feature request"
|
||||||
},
|
},
|
||||||
|
"aboutTelegramChannel": "Telegram канал",
|
||||||
|
"@aboutTelegramChannel": {
|
||||||
|
"description": "Link to Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChannelSubtitle": "Объявления и обновления",
|
||||||
|
"@aboutTelegramChannelSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChat": "Сообщество в Telegram",
|
||||||
|
"@aboutTelegramChat": {
|
||||||
|
"description": "Link to Telegram chat group"
|
||||||
|
},
|
||||||
|
"aboutTelegramChatSubtitle": "Чат с другими пользователями",
|
||||||
|
"@aboutTelegramChatSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram chat"
|
||||||
|
},
|
||||||
|
"aboutSocial": "Соцсети",
|
||||||
|
"@aboutSocial": {
|
||||||
|
"description": "Section for social links"
|
||||||
|
},
|
||||||
"aboutSupport": "Поддержка",
|
"aboutSupport": "Поддержка",
|
||||||
"@aboutSupport": {
|
"@aboutSupport": {
|
||||||
"description": "Section for support/donation links"
|
"description": "Section for support/donation links"
|
||||||
@@ -596,7 +624,7 @@
|
|||||||
"@albumTitle": {
|
"@albumTitle": {
|
||||||
"description": "Album screen title"
|
"description": "Album screen title"
|
||||||
},
|
},
|
||||||
"albumTracks": "{count, plural, one {{count} трек} few {{count} трека} many {{count} треков} other {{count} треков}}",
|
"albumTracks": "{count, plural, one {{count} трек} few {{count} трека} many {{count} треков} =1 {1 трек} other {{count} треков}}",
|
||||||
"@albumTracks": {
|
"@albumTracks": {
|
||||||
"description": "Album track count",
|
"description": "Album track count",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -633,7 +661,7 @@
|
|||||||
"@artistCompilations": {
|
"@artistCompilations": {
|
||||||
"description": "Section header for compilations"
|
"description": "Section header for compilations"
|
||||||
},
|
},
|
||||||
"artistReleases": "{count, plural, one {{count} релиз} few {{count} релиза} many {{count} релизов} other {{count} релизов}}",
|
"artistReleases": "{count, plural, one {{count} релиз} few {{count} релиза} many {{count} релизов} =1 {1 релиз} other {{count} релизов}}",
|
||||||
"@artistReleases": {
|
"@artistReleases": {
|
||||||
"description": "Artist release count",
|
"description": "Artist release count",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1108,7 +1136,7 @@
|
|||||||
"@dialogDeleteSelectedTitle": {
|
"@dialogDeleteSelectedTitle": {
|
||||||
"description": "Dialog title - delete selected items"
|
"description": "Dialog title - delete selected items"
|
||||||
},
|
},
|
||||||
"dialogDeleteSelectedMessage": "Удалить {count} {count, plural, one {трек} few {трека} many {треков} other{треков}} из истории?\n\nЭто также удалит файлы из хранилища.",
|
"dialogDeleteSelectedMessage": "Удалить {count} {count, plural, one {трек} few {трека} many {треков} =1{трек} other{треков}} из истории?\n\nЭто также удалит файлы из хранилища.",
|
||||||
"@dialogDeleteSelectedMessage": {
|
"@dialogDeleteSelectedMessage": {
|
||||||
"description": "Dialog message - delete selected tracks",
|
"description": "Dialog message - delete selected tracks",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1122,6 +1150,15 @@
|
|||||||
"description": "Dialog title - import CSV playlist"
|
"description": "Dialog title - import CSV playlist"
|
||||||
},
|
},
|
||||||
"dialogImportPlaylistMessage": "Найдено {count} треков в CSV. Добавить их в очередь загрузки?",
|
"dialogImportPlaylistMessage": "Найдено {count} треков в CSV. Добавить их в очередь загрузки?",
|
||||||
|
"csvImportTracks": "{count} треков из CSV",
|
||||||
|
"@csvImportTracks": {
|
||||||
|
"description": "Label shown in quality picker for CSV import",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"@dialogImportPlaylistMessage": {
|
"@dialogImportPlaylistMessage": {
|
||||||
"description": "Dialog message - import playlist confirmation",
|
"description": "Dialog message - import playlist confirmation",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1169,7 +1206,7 @@
|
|||||||
"@snackbarCredentialsCleared": {
|
"@snackbarCredentialsCleared": {
|
||||||
"description": "Snackbar - Spotify credentials removed"
|
"description": "Snackbar - Spotify credentials removed"
|
||||||
},
|
},
|
||||||
"snackbarDeletedTracks": "Удалено {count} {count, plural, one {трек} few {трека} many {треков} other{треков}}",
|
"snackbarDeletedTracks": "Удалено {count} {count, plural, one {трек} few {трека} many {треков} =1{трек} other{треков}}",
|
||||||
"@snackbarDeletedTracks": {
|
"@snackbarDeletedTracks": {
|
||||||
"description": "Snackbar - tracks deleted",
|
"description": "Snackbar - tracks deleted",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1376,7 +1413,7 @@
|
|||||||
"@selectionTapToSelect": {
|
"@selectionTapToSelect": {
|
||||||
"description": "Hint - how to select items"
|
"description": "Hint - how to select items"
|
||||||
},
|
},
|
||||||
"selectionDeleteTracks": "Удалить {count} {count, plural, one {трек} few {трека} many {треков} other{треков}}",
|
"selectionDeleteTracks": "Удалить {count} {count, plural, one {трек} few {трека} many {треков} =1{трек} other{треков}}",
|
||||||
"@selectionDeleteTracks": {
|
"@selectionDeleteTracks": {
|
||||||
"description": "Delete button with count",
|
"description": "Delete button with count",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1851,6 +1888,42 @@
|
|||||||
"@sectionFileSettings": {
|
"@sectionFileSettings": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
},
|
},
|
||||||
|
"sectionLyrics": "Тексты песен",
|
||||||
|
"@sectionLyrics": {
|
||||||
|
"description": "Settings section header"
|
||||||
|
},
|
||||||
|
"lyricsMode": "Режим текстов песен",
|
||||||
|
"@lyricsMode": {
|
||||||
|
"description": "Setting - how to save lyrics"
|
||||||
|
},
|
||||||
|
"lyricsModeDescription": "Выберите как сохранить тексты песен при скачивании",
|
||||||
|
"@lyricsModeDescription": {
|
||||||
|
"description": "Lyrics mode picker description"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbed": "Вставить в файл",
|
||||||
|
"@lyricsModeEmbed": {
|
||||||
|
"description": "Lyrics mode option - embed in audio file"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbedSubtitle": "Встроить текст в метаданные FLAC",
|
||||||
|
"@lyricsModeEmbedSubtitle": {
|
||||||
|
"description": "Subtitle for embed option"
|
||||||
|
},
|
||||||
|
"lyricsModeExternal": "Внешний файл .lrc",
|
||||||
|
"@lyricsModeExternal": {
|
||||||
|
"description": "Lyrics mode option - separate LRC file"
|
||||||
|
},
|
||||||
|
"lyricsModeExternalSubtitle": "Отдельный файл .lrc для плееров, таких, как Samsung Music",
|
||||||
|
"@lyricsModeExternalSubtitle": {
|
||||||
|
"description": "Subtitle for external option"
|
||||||
|
},
|
||||||
|
"lyricsModeBoth": "Оба варианта",
|
||||||
|
"@lyricsModeBoth": {
|
||||||
|
"description": "Lyrics mode option - embed and external"
|
||||||
|
},
|
||||||
|
"lyricsModeBothSubtitle": "Вставить и сохранить файл .lrc",
|
||||||
|
"@lyricsModeBothSubtitle": {
|
||||||
|
"description": "Subtitle for both option"
|
||||||
|
},
|
||||||
"sectionColor": "Цвет",
|
"sectionColor": "Цвет",
|
||||||
"@sectionColor": {
|
"@sectionColor": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
@@ -1916,7 +1989,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"tracksCount": "{count, plural, one {{count} трек} few {{count} трека} many {{count} треков} other {{count} треков}}",
|
"tracksCount": "{count, plural, one {{count} трек} few {{count} трека} many {{count} треков} =1 {1 трек} other {{count} треков}}",
|
||||||
"@tracksCount": {
|
"@tracksCount": {
|
||||||
"description": "Track count display",
|
"description": "Track count display",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1997,6 +2070,18 @@
|
|||||||
"@trackReleaseDate": {
|
"@trackReleaseDate": {
|
||||||
"description": "Metadata label - release date"
|
"description": "Metadata label - release date"
|
||||||
},
|
},
|
||||||
|
"trackGenre": "Жанр",
|
||||||
|
"@trackGenre": {
|
||||||
|
"description": "Metadata label - music genre"
|
||||||
|
},
|
||||||
|
"trackLabel": "Заголовок",
|
||||||
|
"@trackLabel": {
|
||||||
|
"description": "Metadata label - record label"
|
||||||
|
},
|
||||||
|
"trackCopyright": "Авторские права",
|
||||||
|
"@trackCopyright": {
|
||||||
|
"description": "Metadata label - copyright information"
|
||||||
|
},
|
||||||
"trackDownloaded": "Скачано",
|
"trackDownloaded": "Скачано",
|
||||||
"@trackDownloaded": {
|
"@trackDownloaded": {
|
||||||
"description": "Metadata label - download date"
|
"description": "Metadata label - download date"
|
||||||
@@ -2017,6 +2102,18 @@
|
|||||||
"@trackLyricsLoadFailed": {
|
"@trackLyricsLoadFailed": {
|
||||||
"description": "Message when lyrics loading fails"
|
"description": "Message when lyrics loading fails"
|
||||||
},
|
},
|
||||||
|
"trackEmbedLyrics": "Вставить текст песни",
|
||||||
|
"@trackEmbedLyrics": {
|
||||||
|
"description": "Action - embed lyrics into audio file"
|
||||||
|
},
|
||||||
|
"trackLyricsEmbedded": "Текст успешно добавлен",
|
||||||
|
"@trackLyricsEmbedded": {
|
||||||
|
"description": "Snackbar - lyrics saved to file"
|
||||||
|
},
|
||||||
|
"trackInstrumental": "Инструментальный трек",
|
||||||
|
"@trackInstrumental": {
|
||||||
|
"description": "Message when track is instrumental (no lyrics)"
|
||||||
|
},
|
||||||
"trackCopiedToClipboard": "Скопировано в буфер обмена",
|
"trackCopiedToClipboard": "Скопировано в буфер обмена",
|
||||||
"@trackCopiedToClipboard": {
|
"@trackCopiedToClipboard": {
|
||||||
"description": "Snackbar - content copied"
|
"description": "Snackbar - content copied"
|
||||||
@@ -2328,6 +2425,26 @@
|
|||||||
"@qualityHiResFlacMaxSubtitle": {
|
"@qualityHiResFlacMaxSubtitle": {
|
||||||
"description": "Technical spec for hi-res max"
|
"description": "Technical spec for hi-res max"
|
||||||
},
|
},
|
||||||
|
"qualityMp3": "MP3",
|
||||||
|
"@qualityMp3": {
|
||||||
|
"description": "Quality option - MP3 lossy format"
|
||||||
|
},
|
||||||
|
"qualityMp3Subtitle": "320 кбит/с (Конвертировано из FLAC)",
|
||||||
|
"@qualityMp3Subtitle": {
|
||||||
|
"description": "Technical spec for MP3"
|
||||||
|
},
|
||||||
|
"enableMp3Option": "Скачивние в MP3",
|
||||||
|
"@enableMp3Option": {
|
||||||
|
"description": "Setting - enable MP3 quality option"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOn": "MP3 качество доступно",
|
||||||
|
"@enableMp3OptionSubtitleOn": {
|
||||||
|
"description": "Subtitle when MP3 is enabled"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOff": "Скачивать FLAC и конвертировать в MP3 320 кбит/с",
|
||||||
|
"@enableMp3OptionSubtitleOff": {
|
||||||
|
"description": "Subtitle when MP3 is disabled"
|
||||||
|
},
|
||||||
"qualityNote": "Фактическое качество зависит от доступности треков в сервисе",
|
"qualityNote": "Фактическое качество зависит от доступности треков в сервисе",
|
||||||
"@qualityNote": {
|
"@qualityNote": {
|
||||||
"description": "Note about quality availability"
|
"description": "Note about quality availability"
|
||||||
@@ -2516,11 +2633,19 @@
|
|||||||
"@albumFolderYearAlbumSubtitle": {
|
"@albumFolderYearAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
|
"albumFolderArtistAlbumSingles": "Исполнитель / Альбом + Синглы",
|
||||||
|
"@albumFolderArtistAlbumSingles": {
|
||||||
|
"description": "Album folder option with singles inside artist"
|
||||||
|
},
|
||||||
|
"albumFolderArtistAlbumSinglesSubtitle": "Исполнитель/Альбом и Исполнитель/Сингл/",
|
||||||
|
"@albumFolderArtistAlbumSinglesSubtitle": {
|
||||||
|
"description": "Folder structure example"
|
||||||
|
},
|
||||||
"downloadedAlbumDeleteSelected": "Удалить выбранные",
|
"downloadedAlbumDeleteSelected": "Удалить выбранные",
|
||||||
"@downloadedAlbumDeleteSelected": {
|
"@downloadedAlbumDeleteSelected": {
|
||||||
"description": "Button - delete selected tracks"
|
"description": "Button - delete selected tracks"
|
||||||
},
|
},
|
||||||
"downloadedAlbumDeleteMessage": "Удалить {count} {count, plural, one {трек} few {трека} many {треков} other{треков}} из этого альбома?\n\nЭто также удалит файлы из хранилища.",
|
"downloadedAlbumDeleteMessage": "Удалить {count} {count, plural, one {трек} few {трека} many {треков} =1{трек} other{треков}} из этого альбома?\n\nЭто также удалит файлы из хранилища.",
|
||||||
"@downloadedAlbumDeleteMessage": {
|
"@downloadedAlbumDeleteMessage": {
|
||||||
"description": "Delete confirmation with count",
|
"description": "Delete confirmation with count",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -2559,7 +2684,7 @@
|
|||||||
"@downloadedAlbumTapToSelect": {
|
"@downloadedAlbumTapToSelect": {
|
||||||
"description": "Selection hint"
|
"description": "Selection hint"
|
||||||
},
|
},
|
||||||
"downloadedAlbumDeleteCount": "Удалить {count} {count, plural, one {трек} few {трека} many {треков} other{треков}}",
|
"downloadedAlbumDeleteCount": "Удалить {count} {count, plural, one {трек} few {трека} many {треков} =1{трек} other{треков}}",
|
||||||
"@downloadedAlbumDeleteCount": {
|
"@downloadedAlbumDeleteCount": {
|
||||||
"description": "Delete button text with count",
|
"description": "Delete button text with count",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -2572,6 +2697,16 @@
|
|||||||
"@downloadedAlbumSelectToDelete": {
|
"@downloadedAlbumSelectToDelete": {
|
||||||
"description": "Placeholder when nothing selected"
|
"description": "Placeholder when nothing selected"
|
||||||
},
|
},
|
||||||
|
"downloadedAlbumDiscHeader": "Диск {discNumber}",
|
||||||
|
"@downloadedAlbumDiscHeader": {
|
||||||
|
"description": "Header for disc separator in multi-disc albums",
|
||||||
|
"placeholders": {
|
||||||
|
"discNumber": {
|
||||||
|
"type": "int",
|
||||||
|
"example": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"utilityFunctions": "Функции утилиты",
|
"utilityFunctions": "Функции утилиты",
|
||||||
"@utilityFunctions": {
|
"@utilityFunctions": {
|
||||||
"description": "Extension capability - utility functions"
|
"description": "Extension capability - utility functions"
|
||||||
@@ -2611,5 +2746,123 @@
|
|||||||
"description": "Error message"
|
"description": "Error message"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"discographyDownload": "Скачать дискографию",
|
||||||
|
"@discographyDownload": {
|
||||||
|
"description": "Button - download artist discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAll": "Скачать всё",
|
||||||
|
"@discographyDownloadAll": {
|
||||||
|
"description": "Option - download entire discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAllSubtitle": "{count} треков из {albumCount} релизов",
|
||||||
|
"@discographyDownloadAllSubtitle": {
|
||||||
|
"description": "Subtitle showing total tracks and albums",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnly": "Только альбомы",
|
||||||
|
"@discographyAlbumsOnly": {
|
||||||
|
"description": "Option - download only albums"
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnlySubtitle": "{count} треков из {albumCount} альбомов",
|
||||||
|
"@discographyAlbumsOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing album tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySinglesOnly": "Только синглы и EP",
|
||||||
|
"@discographySinglesOnly": {
|
||||||
|
"description": "Option - download only singles"
|
||||||
|
},
|
||||||
|
"discographySinglesOnlySubtitle": "{count} треков из {albumCount} синглов",
|
||||||
|
"@discographySinglesOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing singles tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectAlbums": "Выбрать альбомы...",
|
||||||
|
"@discographySelectAlbums": {
|
||||||
|
"description": "Option - manually select albums to download"
|
||||||
|
},
|
||||||
|
"discographySelectAlbumsSubtitle": "Выберите конкретные альбомы или синглы",
|
||||||
|
"@discographySelectAlbumsSubtitle": {
|
||||||
|
"description": "Subtitle for select albums option"
|
||||||
|
},
|
||||||
|
"discographyFetchingTracks": "Получение треков...",
|
||||||
|
"@discographyFetchingTracks": {
|
||||||
|
"description": "Progress - fetching album tracks"
|
||||||
|
},
|
||||||
|
"discographyFetchingAlbum": "Получение {current} из {total}...",
|
||||||
|
"@discographyFetchingAlbum": {
|
||||||
|
"description": "Progress - fetching specific album",
|
||||||
|
"placeholders": {
|
||||||
|
"current": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"total": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectedCount": "{count} выбрано",
|
||||||
|
"@discographySelectedCount": {
|
||||||
|
"description": "Selection count badge",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyDownloadSelected": "Скачать выбранное",
|
||||||
|
"@discographyDownloadSelected": {
|
||||||
|
"description": "Button - download selected albums"
|
||||||
|
},
|
||||||
|
"discographyAddedToQueue": "Добавлено {count} треков в очередь",
|
||||||
|
"@discographyAddedToQueue": {
|
||||||
|
"description": "Snackbar - tracks added from discography",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySkippedDownloaded": "{added} добавлено, {skipped} уже скачано",
|
||||||
|
"@discographySkippedDownloaded": {
|
||||||
|
"description": "Snackbar - with skipped tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"added": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"skipped": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyNoAlbums": "Нет доступных альбомов",
|
||||||
|
"@discographyNoAlbums": {
|
||||||
|
"description": "Error - no albums found for artist"
|
||||||
|
},
|
||||||
|
"discographyFailedToFetch": "Не удалось получить некоторые альбомы",
|
||||||
|
"@discographyFailedToFetch": {
|
||||||
|
"description": "Error - some albums failed to load"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
+2865
-4
File diff suppressed because it is too large
Load Diff
@@ -127,6 +127,10 @@
|
|||||||
"@historyNoSinglesSubtitle": {
|
"@historyNoSinglesSubtitle": {
|
||||||
"description": "Empty state subtitle for singles filter"
|
"description": "Empty state subtitle for singles filter"
|
||||||
},
|
},
|
||||||
|
"historySearchHint": "Search history...",
|
||||||
|
"@historySearchHint": {
|
||||||
|
"description": "Search bar placeholder in history"
|
||||||
|
},
|
||||||
"settingsTitle": "Settings",
|
"settingsTitle": "Settings",
|
||||||
"@settingsTitle": {
|
"@settingsTitle": {
|
||||||
"description": "Settings screen title"
|
"description": "Settings screen title"
|
||||||
@@ -512,6 +516,10 @@
|
|||||||
"@aboutLogoArtist": {
|
"@aboutLogoArtist": {
|
||||||
"description": "Role description for logo artist"
|
"description": "Role description for logo artist"
|
||||||
},
|
},
|
||||||
|
"aboutTranslators": "Translators",
|
||||||
|
"@aboutTranslators": {
|
||||||
|
"description": "Section for translators"
|
||||||
|
},
|
||||||
"aboutSpecialThanks": "Special Thanks",
|
"aboutSpecialThanks": "Special Thanks",
|
||||||
"@aboutSpecialThanks": {
|
"@aboutSpecialThanks": {
|
||||||
"description": "Section for special thanks"
|
"description": "Section for special thanks"
|
||||||
@@ -544,6 +552,26 @@
|
|||||||
"@aboutFeatureRequestSubtitle": {
|
"@aboutFeatureRequestSubtitle": {
|
||||||
"description": "Subtitle for feature request"
|
"description": "Subtitle for feature request"
|
||||||
},
|
},
|
||||||
|
"aboutTelegramChannel": "Telegram Channel",
|
||||||
|
"@aboutTelegramChannel": {
|
||||||
|
"description": "Link to Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChannelSubtitle": "Announcements and updates",
|
||||||
|
"@aboutTelegramChannelSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChat": "Telegram Community",
|
||||||
|
"@aboutTelegramChat": {
|
||||||
|
"description": "Link to Telegram chat group"
|
||||||
|
},
|
||||||
|
"aboutTelegramChatSubtitle": "Chat with other users",
|
||||||
|
"@aboutTelegramChatSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram chat"
|
||||||
|
},
|
||||||
|
"aboutSocial": "Social",
|
||||||
|
"@aboutSocial": {
|
||||||
|
"description": "Section for social links"
|
||||||
|
},
|
||||||
"aboutSupport": "Support",
|
"aboutSupport": "Support",
|
||||||
"@aboutSupport": {
|
"@aboutSupport": {
|
||||||
"description": "Section for support/donation links"
|
"description": "Section for support/donation links"
|
||||||
@@ -1122,6 +1150,15 @@
|
|||||||
"description": "Dialog title - import CSV playlist"
|
"description": "Dialog title - import CSV playlist"
|
||||||
},
|
},
|
||||||
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
||||||
|
"csvImportTracks": "{count} tracks from CSV",
|
||||||
|
"@csvImportTracks": {
|
||||||
|
"description": "Label shown in quality picker for CSV import",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"@dialogImportPlaylistMessage": {
|
"@dialogImportPlaylistMessage": {
|
||||||
"description": "Dialog message - import playlist confirmation",
|
"description": "Dialog message - import playlist confirmation",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1851,6 +1888,42 @@
|
|||||||
"@sectionFileSettings": {
|
"@sectionFileSettings": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
},
|
},
|
||||||
|
"sectionLyrics": "Lyrics",
|
||||||
|
"@sectionLyrics": {
|
||||||
|
"description": "Settings section header"
|
||||||
|
},
|
||||||
|
"lyricsMode": "Lyrics Mode",
|
||||||
|
"@lyricsMode": {
|
||||||
|
"description": "Setting - how to save lyrics"
|
||||||
|
},
|
||||||
|
"lyricsModeDescription": "Choose how lyrics are saved with your downloads",
|
||||||
|
"@lyricsModeDescription": {
|
||||||
|
"description": "Lyrics mode picker description"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbed": "Embed in file",
|
||||||
|
"@lyricsModeEmbed": {
|
||||||
|
"description": "Lyrics mode option - embed in audio file"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbedSubtitle": "Lyrics stored inside FLAC metadata",
|
||||||
|
"@lyricsModeEmbedSubtitle": {
|
||||||
|
"description": "Subtitle for embed option"
|
||||||
|
},
|
||||||
|
"lyricsModeExternal": "External .lrc file",
|
||||||
|
"@lyricsModeExternal": {
|
||||||
|
"description": "Lyrics mode option - separate LRC file"
|
||||||
|
},
|
||||||
|
"lyricsModeExternalSubtitle": "Separate .lrc file for players like Samsung Music",
|
||||||
|
"@lyricsModeExternalSubtitle": {
|
||||||
|
"description": "Subtitle for external option"
|
||||||
|
},
|
||||||
|
"lyricsModeBoth": "Both",
|
||||||
|
"@lyricsModeBoth": {
|
||||||
|
"description": "Lyrics mode option - embed and external"
|
||||||
|
},
|
||||||
|
"lyricsModeBothSubtitle": "Embed and save .lrc file",
|
||||||
|
"@lyricsModeBothSubtitle": {
|
||||||
|
"description": "Subtitle for both option"
|
||||||
|
},
|
||||||
"sectionColor": "Color",
|
"sectionColor": "Color",
|
||||||
"@sectionColor": {
|
"@sectionColor": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
@@ -1997,6 +2070,18 @@
|
|||||||
"@trackReleaseDate": {
|
"@trackReleaseDate": {
|
||||||
"description": "Metadata label - release date"
|
"description": "Metadata label - release date"
|
||||||
},
|
},
|
||||||
|
"trackGenre": "Genre",
|
||||||
|
"@trackGenre": {
|
||||||
|
"description": "Metadata label - music genre"
|
||||||
|
},
|
||||||
|
"trackLabel": "Label",
|
||||||
|
"@trackLabel": {
|
||||||
|
"description": "Metadata label - record label"
|
||||||
|
},
|
||||||
|
"trackCopyright": "Copyright",
|
||||||
|
"@trackCopyright": {
|
||||||
|
"description": "Metadata label - copyright information"
|
||||||
|
},
|
||||||
"trackDownloaded": "Downloaded",
|
"trackDownloaded": "Downloaded",
|
||||||
"@trackDownloaded": {
|
"@trackDownloaded": {
|
||||||
"description": "Metadata label - download date"
|
"description": "Metadata label - download date"
|
||||||
@@ -2017,6 +2102,18 @@
|
|||||||
"@trackLyricsLoadFailed": {
|
"@trackLyricsLoadFailed": {
|
||||||
"description": "Message when lyrics loading fails"
|
"description": "Message when lyrics loading fails"
|
||||||
},
|
},
|
||||||
|
"trackEmbedLyrics": "Embed Lyrics",
|
||||||
|
"@trackEmbedLyrics": {
|
||||||
|
"description": "Action - embed lyrics into audio file"
|
||||||
|
},
|
||||||
|
"trackLyricsEmbedded": "Lyrics embedded successfully",
|
||||||
|
"@trackLyricsEmbedded": {
|
||||||
|
"description": "Snackbar - lyrics saved to file"
|
||||||
|
},
|
||||||
|
"trackInstrumental": "Instrumental track",
|
||||||
|
"@trackInstrumental": {
|
||||||
|
"description": "Message when track is instrumental (no lyrics)"
|
||||||
|
},
|
||||||
"trackCopiedToClipboard": "Copied to clipboard",
|
"trackCopiedToClipboard": "Copied to clipboard",
|
||||||
"@trackCopiedToClipboard": {
|
"@trackCopiedToClipboard": {
|
||||||
"description": "Snackbar - content copied"
|
"description": "Snackbar - content copied"
|
||||||
@@ -2328,6 +2425,26 @@
|
|||||||
"@qualityHiResFlacMaxSubtitle": {
|
"@qualityHiResFlacMaxSubtitle": {
|
||||||
"description": "Technical spec for hi-res max"
|
"description": "Technical spec for hi-res max"
|
||||||
},
|
},
|
||||||
|
"qualityMp3": "MP3",
|
||||||
|
"@qualityMp3": {
|
||||||
|
"description": "Quality option - MP3 lossy format"
|
||||||
|
},
|
||||||
|
"qualityMp3Subtitle": "320kbps (converted from FLAC)",
|
||||||
|
"@qualityMp3Subtitle": {
|
||||||
|
"description": "Technical spec for MP3"
|
||||||
|
},
|
||||||
|
"enableMp3Option": "Enable MP3 Option",
|
||||||
|
"@enableMp3Option": {
|
||||||
|
"description": "Setting - enable MP3 quality option"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOn": "MP3 quality option is available",
|
||||||
|
"@enableMp3OptionSubtitleOn": {
|
||||||
|
"description": "Subtitle when MP3 is enabled"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOff": "Downloads FLAC then converts to 320kbps MP3",
|
||||||
|
"@enableMp3OptionSubtitleOff": {
|
||||||
|
"description": "Subtitle when MP3 is disabled"
|
||||||
|
},
|
||||||
"qualityNote": "Actual quality depends on track availability from the service",
|
"qualityNote": "Actual quality depends on track availability from the service",
|
||||||
"@qualityNote": {
|
"@qualityNote": {
|
||||||
"description": "Note about quality availability"
|
"description": "Note about quality availability"
|
||||||
@@ -2516,6 +2633,14 @@
|
|||||||
"@albumFolderYearAlbumSubtitle": {
|
"@albumFolderYearAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
|
"albumFolderArtistAlbumSingles": "Artist / Album + Singles",
|
||||||
|
"@albumFolderArtistAlbumSingles": {
|
||||||
|
"description": "Album folder option with singles inside artist"
|
||||||
|
},
|
||||||
|
"albumFolderArtistAlbumSinglesSubtitle": "Artist/Album/ and Artist/Singles/",
|
||||||
|
"@albumFolderArtistAlbumSinglesSubtitle": {
|
||||||
|
"description": "Folder structure example"
|
||||||
|
},
|
||||||
"downloadedAlbumDeleteSelected": "Delete Selected",
|
"downloadedAlbumDeleteSelected": "Delete Selected",
|
||||||
"@downloadedAlbumDeleteSelected": {
|
"@downloadedAlbumDeleteSelected": {
|
||||||
"description": "Button - delete selected tracks"
|
"description": "Button - delete selected tracks"
|
||||||
@@ -2572,6 +2697,16 @@
|
|||||||
"@downloadedAlbumSelectToDelete": {
|
"@downloadedAlbumSelectToDelete": {
|
||||||
"description": "Placeholder when nothing selected"
|
"description": "Placeholder when nothing selected"
|
||||||
},
|
},
|
||||||
|
"downloadedAlbumDiscHeader": "Disc {discNumber}",
|
||||||
|
"@downloadedAlbumDiscHeader": {
|
||||||
|
"description": "Header for disc separator in multi-disc albums",
|
||||||
|
"placeholders": {
|
||||||
|
"discNumber": {
|
||||||
|
"type": "int",
|
||||||
|
"example": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"utilityFunctions": "Utility Functions",
|
"utilityFunctions": "Utility Functions",
|
||||||
"@utilityFunctions": {
|
"@utilityFunctions": {
|
||||||
"description": "Extension capability - utility functions"
|
"description": "Extension capability - utility functions"
|
||||||
@@ -2611,5 +2746,123 @@
|
|||||||
"description": "Error message"
|
"description": "Error message"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"discographyDownload": "Download Discography",
|
||||||
|
"@discographyDownload": {
|
||||||
|
"description": "Button - download artist discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAll": "Download All",
|
||||||
|
"@discographyDownloadAll": {
|
||||||
|
"description": "Option - download entire discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAllSubtitle": "{count} tracks from {albumCount} releases",
|
||||||
|
"@discographyDownloadAllSubtitle": {
|
||||||
|
"description": "Subtitle showing total tracks and albums",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnly": "Albums Only",
|
||||||
|
"@discographyAlbumsOnly": {
|
||||||
|
"description": "Option - download only albums"
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnlySubtitle": "{count} tracks from {albumCount} albums",
|
||||||
|
"@discographyAlbumsOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing album tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySinglesOnly": "Singles & EPs Only",
|
||||||
|
"@discographySinglesOnly": {
|
||||||
|
"description": "Option - download only singles"
|
||||||
|
},
|
||||||
|
"discographySinglesOnlySubtitle": "{count} tracks from {albumCount} singles",
|
||||||
|
"@discographySinglesOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing singles tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectAlbums": "Select Albums...",
|
||||||
|
"@discographySelectAlbums": {
|
||||||
|
"description": "Option - manually select albums to download"
|
||||||
|
},
|
||||||
|
"discographySelectAlbumsSubtitle": "Choose specific albums or singles",
|
||||||
|
"@discographySelectAlbumsSubtitle": {
|
||||||
|
"description": "Subtitle for select albums option"
|
||||||
|
},
|
||||||
|
"discographyFetchingTracks": "Fetching tracks...",
|
||||||
|
"@discographyFetchingTracks": {
|
||||||
|
"description": "Progress - fetching album tracks"
|
||||||
|
},
|
||||||
|
"discographyFetchingAlbum": "Fetching {current} of {total}...",
|
||||||
|
"@discographyFetchingAlbum": {
|
||||||
|
"description": "Progress - fetching specific album",
|
||||||
|
"placeholders": {
|
||||||
|
"current": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"total": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectedCount": "{count} selected",
|
||||||
|
"@discographySelectedCount": {
|
||||||
|
"description": "Selection count badge",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyDownloadSelected": "Download Selected",
|
||||||
|
"@discographyDownloadSelected": {
|
||||||
|
"description": "Button - download selected albums"
|
||||||
|
},
|
||||||
|
"discographyAddedToQueue": "Added {count} tracks to queue",
|
||||||
|
"@discographyAddedToQueue": {
|
||||||
|
"description": "Snackbar - tracks added from discography",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySkippedDownloaded": "{added} added, {skipped} already downloaded",
|
||||||
|
"@discographySkippedDownloaded": {
|
||||||
|
"description": "Snackbar - with skipped tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"added": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"skipped": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyNoAlbums": "No albums available",
|
||||||
|
"@discographyNoAlbums": {
|
||||||
|
"description": "Error - no albums found for artist"
|
||||||
|
},
|
||||||
|
"discographyFailedToFetch": "Failed to fetch some albums",
|
||||||
|
"@discographyFailedToFetch": {
|
||||||
|
"description": "Error - some albums failed to load"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -127,6 +127,10 @@
|
|||||||
"@historyNoSinglesSubtitle": {
|
"@historyNoSinglesSubtitle": {
|
||||||
"description": "Empty state subtitle for singles filter"
|
"description": "Empty state subtitle for singles filter"
|
||||||
},
|
},
|
||||||
|
"historySearchHint": "Search history...",
|
||||||
|
"@historySearchHint": {
|
||||||
|
"description": "Search bar placeholder in history"
|
||||||
|
},
|
||||||
"settingsTitle": "Settings",
|
"settingsTitle": "Settings",
|
||||||
"@settingsTitle": {
|
"@settingsTitle": {
|
||||||
"description": "Settings screen title"
|
"description": "Settings screen title"
|
||||||
@@ -512,6 +516,10 @@
|
|||||||
"@aboutLogoArtist": {
|
"@aboutLogoArtist": {
|
||||||
"description": "Role description for logo artist"
|
"description": "Role description for logo artist"
|
||||||
},
|
},
|
||||||
|
"aboutTranslators": "Translators",
|
||||||
|
"@aboutTranslators": {
|
||||||
|
"description": "Section for translators"
|
||||||
|
},
|
||||||
"aboutSpecialThanks": "Special Thanks",
|
"aboutSpecialThanks": "Special Thanks",
|
||||||
"@aboutSpecialThanks": {
|
"@aboutSpecialThanks": {
|
||||||
"description": "Section for special thanks"
|
"description": "Section for special thanks"
|
||||||
@@ -544,6 +552,26 @@
|
|||||||
"@aboutFeatureRequestSubtitle": {
|
"@aboutFeatureRequestSubtitle": {
|
||||||
"description": "Subtitle for feature request"
|
"description": "Subtitle for feature request"
|
||||||
},
|
},
|
||||||
|
"aboutTelegramChannel": "Telegram Channel",
|
||||||
|
"@aboutTelegramChannel": {
|
||||||
|
"description": "Link to Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChannelSubtitle": "Announcements and updates",
|
||||||
|
"@aboutTelegramChannelSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram channel"
|
||||||
|
},
|
||||||
|
"aboutTelegramChat": "Telegram Community",
|
||||||
|
"@aboutTelegramChat": {
|
||||||
|
"description": "Link to Telegram chat group"
|
||||||
|
},
|
||||||
|
"aboutTelegramChatSubtitle": "Chat with other users",
|
||||||
|
"@aboutTelegramChatSubtitle": {
|
||||||
|
"description": "Subtitle for Telegram chat"
|
||||||
|
},
|
||||||
|
"aboutSocial": "Social",
|
||||||
|
"@aboutSocial": {
|
||||||
|
"description": "Section for social links"
|
||||||
|
},
|
||||||
"aboutSupport": "Support",
|
"aboutSupport": "Support",
|
||||||
"@aboutSupport": {
|
"@aboutSupport": {
|
||||||
"description": "Section for support/donation links"
|
"description": "Section for support/donation links"
|
||||||
@@ -1122,6 +1150,15 @@
|
|||||||
"description": "Dialog title - import CSV playlist"
|
"description": "Dialog title - import CSV playlist"
|
||||||
},
|
},
|
||||||
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
"dialogImportPlaylistMessage": "Found {count} tracks in CSV. Add them to download queue?",
|
||||||
|
"csvImportTracks": "{count} tracks from CSV",
|
||||||
|
"@csvImportTracks": {
|
||||||
|
"description": "Label shown in quality picker for CSV import",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"@dialogImportPlaylistMessage": {
|
"@dialogImportPlaylistMessage": {
|
||||||
"description": "Dialog message - import playlist confirmation",
|
"description": "Dialog message - import playlist confirmation",
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
@@ -1851,6 +1888,42 @@
|
|||||||
"@sectionFileSettings": {
|
"@sectionFileSettings": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
},
|
},
|
||||||
|
"sectionLyrics": "Lyrics",
|
||||||
|
"@sectionLyrics": {
|
||||||
|
"description": "Settings section header"
|
||||||
|
},
|
||||||
|
"lyricsMode": "Lyrics Mode",
|
||||||
|
"@lyricsMode": {
|
||||||
|
"description": "Setting - how to save lyrics"
|
||||||
|
},
|
||||||
|
"lyricsModeDescription": "Choose how lyrics are saved with your downloads",
|
||||||
|
"@lyricsModeDescription": {
|
||||||
|
"description": "Lyrics mode picker description"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbed": "Embed in file",
|
||||||
|
"@lyricsModeEmbed": {
|
||||||
|
"description": "Lyrics mode option - embed in audio file"
|
||||||
|
},
|
||||||
|
"lyricsModeEmbedSubtitle": "Lyrics stored inside FLAC metadata",
|
||||||
|
"@lyricsModeEmbedSubtitle": {
|
||||||
|
"description": "Subtitle for embed option"
|
||||||
|
},
|
||||||
|
"lyricsModeExternal": "External .lrc file",
|
||||||
|
"@lyricsModeExternal": {
|
||||||
|
"description": "Lyrics mode option - separate LRC file"
|
||||||
|
},
|
||||||
|
"lyricsModeExternalSubtitle": "Separate .lrc file for players like Samsung Music",
|
||||||
|
"@lyricsModeExternalSubtitle": {
|
||||||
|
"description": "Subtitle for external option"
|
||||||
|
},
|
||||||
|
"lyricsModeBoth": "Both",
|
||||||
|
"@lyricsModeBoth": {
|
||||||
|
"description": "Lyrics mode option - embed and external"
|
||||||
|
},
|
||||||
|
"lyricsModeBothSubtitle": "Embed and save .lrc file",
|
||||||
|
"@lyricsModeBothSubtitle": {
|
||||||
|
"description": "Subtitle for both option"
|
||||||
|
},
|
||||||
"sectionColor": "Color",
|
"sectionColor": "Color",
|
||||||
"@sectionColor": {
|
"@sectionColor": {
|
||||||
"description": "Settings section header"
|
"description": "Settings section header"
|
||||||
@@ -1997,6 +2070,18 @@
|
|||||||
"@trackReleaseDate": {
|
"@trackReleaseDate": {
|
||||||
"description": "Metadata label - release date"
|
"description": "Metadata label - release date"
|
||||||
},
|
},
|
||||||
|
"trackGenre": "Genre",
|
||||||
|
"@trackGenre": {
|
||||||
|
"description": "Metadata label - music genre"
|
||||||
|
},
|
||||||
|
"trackLabel": "Label",
|
||||||
|
"@trackLabel": {
|
||||||
|
"description": "Metadata label - record label"
|
||||||
|
},
|
||||||
|
"trackCopyright": "Copyright",
|
||||||
|
"@trackCopyright": {
|
||||||
|
"description": "Metadata label - copyright information"
|
||||||
|
},
|
||||||
"trackDownloaded": "Downloaded",
|
"trackDownloaded": "Downloaded",
|
||||||
"@trackDownloaded": {
|
"@trackDownloaded": {
|
||||||
"description": "Metadata label - download date"
|
"description": "Metadata label - download date"
|
||||||
@@ -2017,6 +2102,18 @@
|
|||||||
"@trackLyricsLoadFailed": {
|
"@trackLyricsLoadFailed": {
|
||||||
"description": "Message when lyrics loading fails"
|
"description": "Message when lyrics loading fails"
|
||||||
},
|
},
|
||||||
|
"trackEmbedLyrics": "Embed Lyrics",
|
||||||
|
"@trackEmbedLyrics": {
|
||||||
|
"description": "Action - embed lyrics into audio file"
|
||||||
|
},
|
||||||
|
"trackLyricsEmbedded": "Lyrics embedded successfully",
|
||||||
|
"@trackLyricsEmbedded": {
|
||||||
|
"description": "Snackbar - lyrics saved to file"
|
||||||
|
},
|
||||||
|
"trackInstrumental": "Instrumental track",
|
||||||
|
"@trackInstrumental": {
|
||||||
|
"description": "Message when track is instrumental (no lyrics)"
|
||||||
|
},
|
||||||
"trackCopiedToClipboard": "Copied to clipboard",
|
"trackCopiedToClipboard": "Copied to clipboard",
|
||||||
"@trackCopiedToClipboard": {
|
"@trackCopiedToClipboard": {
|
||||||
"description": "Snackbar - content copied"
|
"description": "Snackbar - content copied"
|
||||||
@@ -2328,6 +2425,26 @@
|
|||||||
"@qualityHiResFlacMaxSubtitle": {
|
"@qualityHiResFlacMaxSubtitle": {
|
||||||
"description": "Technical spec for hi-res max"
|
"description": "Technical spec for hi-res max"
|
||||||
},
|
},
|
||||||
|
"qualityMp3": "MP3",
|
||||||
|
"@qualityMp3": {
|
||||||
|
"description": "Quality option - MP3 lossy format"
|
||||||
|
},
|
||||||
|
"qualityMp3Subtitle": "320kbps (converted from FLAC)",
|
||||||
|
"@qualityMp3Subtitle": {
|
||||||
|
"description": "Technical spec for MP3"
|
||||||
|
},
|
||||||
|
"enableMp3Option": "Enable MP3 Option",
|
||||||
|
"@enableMp3Option": {
|
||||||
|
"description": "Setting - enable MP3 quality option"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOn": "MP3 quality option is available",
|
||||||
|
"@enableMp3OptionSubtitleOn": {
|
||||||
|
"description": "Subtitle when MP3 is enabled"
|
||||||
|
},
|
||||||
|
"enableMp3OptionSubtitleOff": "Downloads FLAC then converts to 320kbps MP3",
|
||||||
|
"@enableMp3OptionSubtitleOff": {
|
||||||
|
"description": "Subtitle when MP3 is disabled"
|
||||||
|
},
|
||||||
"qualityNote": "Actual quality depends on track availability from the service",
|
"qualityNote": "Actual quality depends on track availability from the service",
|
||||||
"@qualityNote": {
|
"@qualityNote": {
|
||||||
"description": "Note about quality availability"
|
"description": "Note about quality availability"
|
||||||
@@ -2516,6 +2633,14 @@
|
|||||||
"@albumFolderYearAlbumSubtitle": {
|
"@albumFolderYearAlbumSubtitle": {
|
||||||
"description": "Folder structure example"
|
"description": "Folder structure example"
|
||||||
},
|
},
|
||||||
|
"albumFolderArtistAlbumSingles": "Artist / Album + Singles",
|
||||||
|
"@albumFolderArtistAlbumSingles": {
|
||||||
|
"description": "Album folder option with singles inside artist"
|
||||||
|
},
|
||||||
|
"albumFolderArtistAlbumSinglesSubtitle": "Artist/Album/ and Artist/Singles/",
|
||||||
|
"@albumFolderArtistAlbumSinglesSubtitle": {
|
||||||
|
"description": "Folder structure example"
|
||||||
|
},
|
||||||
"downloadedAlbumDeleteSelected": "Delete Selected",
|
"downloadedAlbumDeleteSelected": "Delete Selected",
|
||||||
"@downloadedAlbumDeleteSelected": {
|
"@downloadedAlbumDeleteSelected": {
|
||||||
"description": "Button - delete selected tracks"
|
"description": "Button - delete selected tracks"
|
||||||
@@ -2572,6 +2697,16 @@
|
|||||||
"@downloadedAlbumSelectToDelete": {
|
"@downloadedAlbumSelectToDelete": {
|
||||||
"description": "Placeholder when nothing selected"
|
"description": "Placeholder when nothing selected"
|
||||||
},
|
},
|
||||||
|
"downloadedAlbumDiscHeader": "Disc {discNumber}",
|
||||||
|
"@downloadedAlbumDiscHeader": {
|
||||||
|
"description": "Header for disc separator in multi-disc albums",
|
||||||
|
"placeholders": {
|
||||||
|
"discNumber": {
|
||||||
|
"type": "int",
|
||||||
|
"example": "1"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
"utilityFunctions": "Utility Functions",
|
"utilityFunctions": "Utility Functions",
|
||||||
"@utilityFunctions": {
|
"@utilityFunctions": {
|
||||||
"description": "Extension capability - utility functions"
|
"description": "Extension capability - utility functions"
|
||||||
@@ -2611,5 +2746,123 @@
|
|||||||
"description": "Error message"
|
"description": "Error message"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"discographyDownload": "Download Discography",
|
||||||
|
"@discographyDownload": {
|
||||||
|
"description": "Button - download artist discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAll": "Download All",
|
||||||
|
"@discographyDownloadAll": {
|
||||||
|
"description": "Option - download entire discography"
|
||||||
|
},
|
||||||
|
"discographyDownloadAllSubtitle": "{count} tracks from {albumCount} releases",
|
||||||
|
"@discographyDownloadAllSubtitle": {
|
||||||
|
"description": "Subtitle showing total tracks and albums",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnly": "Albums Only",
|
||||||
|
"@discographyAlbumsOnly": {
|
||||||
|
"description": "Option - download only albums"
|
||||||
|
},
|
||||||
|
"discographyAlbumsOnlySubtitle": "{count} tracks from {albumCount} albums",
|
||||||
|
"@discographyAlbumsOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing album tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySinglesOnly": "Singles & EPs Only",
|
||||||
|
"@discographySinglesOnly": {
|
||||||
|
"description": "Option - download only singles"
|
||||||
|
},
|
||||||
|
"discographySinglesOnlySubtitle": "{count} tracks from {albumCount} singles",
|
||||||
|
"@discographySinglesOnlySubtitle": {
|
||||||
|
"description": "Subtitle showing singles tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"albumCount": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectAlbums": "Select Albums...",
|
||||||
|
"@discographySelectAlbums": {
|
||||||
|
"description": "Option - manually select albums to download"
|
||||||
|
},
|
||||||
|
"discographySelectAlbumsSubtitle": "Choose specific albums or singles",
|
||||||
|
"@discographySelectAlbumsSubtitle": {
|
||||||
|
"description": "Subtitle for select albums option"
|
||||||
|
},
|
||||||
|
"discographyFetchingTracks": "Fetching tracks...",
|
||||||
|
"@discographyFetchingTracks": {
|
||||||
|
"description": "Progress - fetching album tracks"
|
||||||
|
},
|
||||||
|
"discographyFetchingAlbum": "Fetching {current} of {total}...",
|
||||||
|
"@discographyFetchingAlbum": {
|
||||||
|
"description": "Progress - fetching specific album",
|
||||||
|
"placeholders": {
|
||||||
|
"current": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"total": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySelectedCount": "{count} selected",
|
||||||
|
"@discographySelectedCount": {
|
||||||
|
"description": "Selection count badge",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyDownloadSelected": "Download Selected",
|
||||||
|
"@discographyDownloadSelected": {
|
||||||
|
"description": "Button - download selected albums"
|
||||||
|
},
|
||||||
|
"discographyAddedToQueue": "Added {count} tracks to queue",
|
||||||
|
"@discographyAddedToQueue": {
|
||||||
|
"description": "Snackbar - tracks added from discography",
|
||||||
|
"placeholders": {
|
||||||
|
"count": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographySkippedDownloaded": "{added} added, {skipped} already downloaded",
|
||||||
|
"@discographySkippedDownloaded": {
|
||||||
|
"description": "Snackbar - with skipped tracks count",
|
||||||
|
"placeholders": {
|
||||||
|
"added": {
|
||||||
|
"type": "int"
|
||||||
|
},
|
||||||
|
"skipped": {
|
||||||
|
"type": "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"discographyNoAlbums": "No albums available",
|
||||||
|
"@discographyNoAlbums": {
|
||||||
|
"description": "Error - no albums found for artist"
|
||||||
|
},
|
||||||
|
"discographyFailedToFetch": "Failed to fetch some albums",
|
||||||
|
"@discographyFailedToFetch": {
|
||||||
|
"description": "Error - some albums failed to load"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -18,6 +18,8 @@ const List<Locale> filteredSupportedLocales = <Locale>[
|
|||||||
Locale('es', 'ES'),
|
Locale('es', 'ES'),
|
||||||
Locale('id'),
|
Locale('id'),
|
||||||
Locale('pt', 'PT'),
|
Locale('pt', 'PT'),
|
||||||
|
Locale('ja'),
|
||||||
|
Locale('tr'),
|
||||||
];
|
];
|
||||||
|
|
||||||
/// Set of locale codes for quick lookup.
|
/// Set of locale codes for quick lookup.
|
||||||
@@ -27,4 +29,6 @@ const Set<String> filteredLocaleCodes = <String>{
|
|||||||
'es_ES',
|
'es_ES',
|
||||||
'id',
|
'id',
|
||||||
'pt_PT',
|
'pt_PT',
|
||||||
|
'ja',
|
||||||
|
'tr',
|
||||||
};
|
};
|
||||||
|
|||||||
+2
-1
@@ -43,6 +43,8 @@ class _EagerInitializationState extends ConsumerState<_EagerInitialization> {
|
|||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
_initializeExtensions();
|
_initializeExtensions();
|
||||||
|
// Trigger history provider initialization without subscribing to updates.
|
||||||
|
ref.read(downloadHistoryProvider);
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _initializeExtensions() async {
|
Future<void> _initializeExtensions() async {
|
||||||
@@ -62,7 +64,6 @@ class _EagerInitializationState extends ConsumerState<_EagerInitialization> {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
ref.watch(downloadHistoryProvider);
|
|
||||||
return widget.child;
|
return widget.child;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -31,8 +31,11 @@ class AppSettings {
|
|||||||
final String albumFolderStructure;
|
final String albumFolderStructure;
|
||||||
final bool showExtensionStore;
|
final bool showExtensionStore;
|
||||||
final String locale;
|
final String locale;
|
||||||
final bool enableMp3Option;
|
final bool enableLossyOption;
|
||||||
|
final String lossyFormat;
|
||||||
|
final String lossyBitrate; // e.g., 'mp3_320', 'mp3_256', 'mp3_192', 'mp3_128', 'opus_128', 'opus_96', 'opus_64'
|
||||||
final String lyricsMode;
|
final String lyricsMode;
|
||||||
|
final bool useAllFilesAccess; // Android 13+ only: enable MANAGE_EXTERNAL_STORAGE
|
||||||
|
|
||||||
const AppSettings({
|
const AppSettings({
|
||||||
this.defaultService = 'tidal',
|
this.defaultService = 'tidal',
|
||||||
@@ -62,8 +65,11 @@ class AppSettings {
|
|||||||
this.albumFolderStructure = 'artist_album',
|
this.albumFolderStructure = 'artist_album',
|
||||||
this.showExtensionStore = true,
|
this.showExtensionStore = true,
|
||||||
this.locale = 'system',
|
this.locale = 'system',
|
||||||
this.enableMp3Option = false,
|
this.enableLossyOption = false,
|
||||||
|
this.lossyFormat = 'mp3',
|
||||||
|
this.lossyBitrate = 'mp3_320',
|
||||||
this.lyricsMode = 'embed',
|
this.lyricsMode = 'embed',
|
||||||
|
this.useAllFilesAccess = false,
|
||||||
});
|
});
|
||||||
|
|
||||||
AppSettings copyWith({
|
AppSettings copyWith({
|
||||||
@@ -95,8 +101,11 @@ class AppSettings {
|
|||||||
String? albumFolderStructure,
|
String? albumFolderStructure,
|
||||||
bool? showExtensionStore,
|
bool? showExtensionStore,
|
||||||
String? locale,
|
String? locale,
|
||||||
bool? enableMp3Option,
|
bool? enableLossyOption,
|
||||||
|
String? lossyFormat,
|
||||||
|
String? lossyBitrate,
|
||||||
String? lyricsMode,
|
String? lyricsMode,
|
||||||
|
bool? useAllFilesAccess,
|
||||||
}) {
|
}) {
|
||||||
return AppSettings(
|
return AppSettings(
|
||||||
defaultService: defaultService ?? this.defaultService,
|
defaultService: defaultService ?? this.defaultService,
|
||||||
@@ -126,8 +135,11 @@ class AppSettings {
|
|||||||
albumFolderStructure: albumFolderStructure ?? this.albumFolderStructure,
|
albumFolderStructure: albumFolderStructure ?? this.albumFolderStructure,
|
||||||
showExtensionStore: showExtensionStore ?? this.showExtensionStore,
|
showExtensionStore: showExtensionStore ?? this.showExtensionStore,
|
||||||
locale: locale ?? this.locale,
|
locale: locale ?? this.locale,
|
||||||
enableMp3Option: enableMp3Option ?? this.enableMp3Option,
|
enableLossyOption: enableLossyOption ?? this.enableLossyOption,
|
||||||
|
lossyFormat: lossyFormat ?? this.lossyFormat,
|
||||||
|
lossyBitrate: lossyBitrate ?? this.lossyBitrate,
|
||||||
lyricsMode: lyricsMode ?? this.lyricsMode,
|
lyricsMode: lyricsMode ?? this.lyricsMode,
|
||||||
|
useAllFilesAccess: useAllFilesAccess ?? this.useAllFilesAccess,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -36,8 +36,11 @@ AppSettings _$AppSettingsFromJson(Map<String, dynamic> json) => AppSettings(
|
|||||||
json['albumFolderStructure'] as String? ?? 'artist_album',
|
json['albumFolderStructure'] as String? ?? 'artist_album',
|
||||||
showExtensionStore: json['showExtensionStore'] as bool? ?? true,
|
showExtensionStore: json['showExtensionStore'] as bool? ?? true,
|
||||||
locale: json['locale'] as String? ?? 'system',
|
locale: json['locale'] as String? ?? 'system',
|
||||||
enableMp3Option: json['enableMp3Option'] as bool? ?? false,
|
enableLossyOption: json['enableLossyOption'] as bool? ?? false,
|
||||||
|
lossyFormat: json['lossyFormat'] as String? ?? 'mp3',
|
||||||
|
lossyBitrate: json['lossyBitrate'] as String? ?? 'mp3_320',
|
||||||
lyricsMode: json['lyricsMode'] as String? ?? 'embed',
|
lyricsMode: json['lyricsMode'] as String? ?? 'embed',
|
||||||
|
useAllFilesAccess: json['useAllFilesAccess'] as bool? ?? false,
|
||||||
);
|
);
|
||||||
|
|
||||||
Map<String, dynamic> _$AppSettingsToJson(AppSettings instance) =>
|
Map<String, dynamic> _$AppSettingsToJson(AppSettings instance) =>
|
||||||
@@ -69,6 +72,9 @@ Map<String, dynamic> _$AppSettingsToJson(AppSettings instance) =>
|
|||||||
'albumFolderStructure': instance.albumFolderStructure,
|
'albumFolderStructure': instance.albumFolderStructure,
|
||||||
'showExtensionStore': instance.showExtensionStore,
|
'showExtensionStore': instance.showExtensionStore,
|
||||||
'locale': instance.locale,
|
'locale': instance.locale,
|
||||||
'enableMp3Option': instance.enableMp3Option,
|
'enableLossyOption': instance.enableLossyOption,
|
||||||
|
'lossyFormat': instance.lossyFormat,
|
||||||
|
'lossyBitrate': instance.lossyBitrate,
|
||||||
'lyricsMode': instance.lyricsMode,
|
'lyricsMode': instance.lyricsMode,
|
||||||
|
'useAllFilesAccess': instance.useAllFilesAccess,
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1376,6 +1376,143 @@ class DownloadQueueNotifier extends Notifier<DownloadQueueState> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<void> _embedMetadataToOpus(
|
||||||
|
String opusPath,
|
||||||
|
Track track, {
|
||||||
|
String? genre,
|
||||||
|
String? label,
|
||||||
|
String? copyright,
|
||||||
|
}) async {
|
||||||
|
final settings = ref.read(settingsProvider);
|
||||||
|
|
||||||
|
String? coverPath;
|
||||||
|
var coverUrl = track.coverUrl;
|
||||||
|
if (coverUrl != null && coverUrl.isNotEmpty) {
|
||||||
|
try {
|
||||||
|
if (settings.maxQualityCover) {
|
||||||
|
coverUrl = _upgradeToMaxQualityCover(coverUrl);
|
||||||
|
_log.d('Cover URL upgraded to max quality for Opus: $coverUrl');
|
||||||
|
}
|
||||||
|
|
||||||
|
final tempDir = await getTemporaryDirectory();
|
||||||
|
final uniqueId =
|
||||||
|
'${DateTime.now().millisecondsSinceEpoch}_${Random().nextInt(10000)}';
|
||||||
|
coverPath = '${tempDir.path}/cover_opus_$uniqueId.jpg';
|
||||||
|
|
||||||
|
final httpClient = HttpClient();
|
||||||
|
final request = await httpClient.getUrl(Uri.parse(coverUrl));
|
||||||
|
final response = await request.close();
|
||||||
|
if (response.statusCode == 200) {
|
||||||
|
final file = File(coverPath);
|
||||||
|
final sink = file.openWrite();
|
||||||
|
await response.pipe(sink);
|
||||||
|
await sink.close();
|
||||||
|
_log.d('Cover downloaded for Opus: $coverPath');
|
||||||
|
} else {
|
||||||
|
_log.w('Failed to download cover for Opus: HTTP ${response.statusCode}');
|
||||||
|
coverPath = null;
|
||||||
|
}
|
||||||
|
httpClient.close();
|
||||||
|
} catch (e) {
|
||||||
|
_log.e('Failed to download cover for Opus: $e');
|
||||||
|
coverPath = null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
final metadata = <String, String>{
|
||||||
|
'TITLE': track.name,
|
||||||
|
'ARTIST': track.artistName,
|
||||||
|
'ALBUM': track.albumName,
|
||||||
|
};
|
||||||
|
|
||||||
|
final albumArtist = _normalizeOptionalString(track.albumArtist) ??
|
||||||
|
track.artistName;
|
||||||
|
metadata['ALBUMARTIST'] = albumArtist;
|
||||||
|
|
||||||
|
if (track.trackNumber != null) {
|
||||||
|
metadata['TRACKNUMBER'] = track.trackNumber.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (track.discNumber != null) {
|
||||||
|
metadata['DISCNUMBER'] = track.discNumber.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (track.releaseDate != null) {
|
||||||
|
metadata['DATE'] = track.releaseDate!;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (track.isrc != null) {
|
||||||
|
metadata['ISRC'] = track.isrc!;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (genre != null && genre.isNotEmpty) {
|
||||||
|
metadata['GENRE'] = genre;
|
||||||
|
_log.d('Adding GENRE to Opus: $genre');
|
||||||
|
}
|
||||||
|
if (label != null && label.isNotEmpty) {
|
||||||
|
metadata['ORGANIZATION'] = label;
|
||||||
|
_log.d('Adding ORGANIZATION (label) to Opus: $label');
|
||||||
|
}
|
||||||
|
if (copyright != null && copyright.isNotEmpty) {
|
||||||
|
metadata['COPYRIGHT'] = copyright;
|
||||||
|
_log.d('Adding COPYRIGHT to Opus: $copyright');
|
||||||
|
}
|
||||||
|
|
||||||
|
_log.d('Opus Metadata map content: $metadata');
|
||||||
|
|
||||||
|
if (settings.embedLyrics) {
|
||||||
|
try {
|
||||||
|
final durationMs = track.duration * 1000;
|
||||||
|
|
||||||
|
final lrcContent = await PlatformBridge.getLyricsLRC(
|
||||||
|
track.id,
|
||||||
|
track.name,
|
||||||
|
track.artistName,
|
||||||
|
filePath: '',
|
||||||
|
durationMs: durationMs,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (lrcContent.isNotEmpty) {
|
||||||
|
metadata['LYRICS'] = lrcContent;
|
||||||
|
_log.d('Lyrics fetched for Opus embedding (${lrcContent.length} chars)');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
_log.w('Failed to fetch lyrics for Opus embedding: $e');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_log.d('Embedding tags to Opus: $metadata');
|
||||||
|
|
||||||
|
final result = await FFmpegService.embedMetadataToOpus(
|
||||||
|
opusPath: opusPath,
|
||||||
|
coverPath: coverPath != null && await File(coverPath).exists()
|
||||||
|
? coverPath
|
||||||
|
: null,
|
||||||
|
metadata: metadata,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (result != null) {
|
||||||
|
_log.d('Metadata, lyrics, and cover embedded to Opus via FFmpeg');
|
||||||
|
} else {
|
||||||
|
_log.w('FFmpeg Opus metadata/cover embed failed');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (coverPath != null) {
|
||||||
|
try {
|
||||||
|
final coverFile = File(coverPath);
|
||||||
|
if (await coverFile.exists()) {
|
||||||
|
await coverFile.delete();
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
_log.w('Failed to cleanup Opus cover file: $e');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
_log.e('Failed to embed metadata to Opus: $e');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Future<void> _processQueue() async {
|
Future<void> _processQueue() async {
|
||||||
if (state.isProcessing) return;
|
if (state.isProcessing) return;
|
||||||
|
|
||||||
@@ -1667,6 +1804,10 @@ class DownloadQueueNotifier extends Notifier<DownloadQueueState> {
|
|||||||
);
|
);
|
||||||
|
|
||||||
final quality = item.qualityOverride ?? state.audioQuality;
|
final quality = item.qualityOverride ?? state.audioQuality;
|
||||||
|
|
||||||
|
// For LOSSY, we need to download FLAC first then convert
|
||||||
|
// Servers don't support lossy quality directly
|
||||||
|
final downloadQuality = quality == 'LOSSY' ? 'LOSSLESS' : quality;
|
||||||
|
|
||||||
// Fetch extended metadata (genre, label) from Deezer if available
|
// Fetch extended metadata (genre, label) from Deezer if available
|
||||||
String? genre;
|
String? genre;
|
||||||
@@ -1717,7 +1858,7 @@ class DownloadQueueNotifier extends Notifier<DownloadQueueState> {
|
|||||||
if (useExtensions) {
|
if (useExtensions) {
|
||||||
_log.d('Using extension providers for download');
|
_log.d('Using extension providers for download');
|
||||||
_log.d(
|
_log.d(
|
||||||
'Quality: $quality${item.qualityOverride != null ? ' (override)' : ''}',
|
'Quality: $quality${item.qualityOverride != null ? ' (override)' : ''}${quality == 'LOSSY' ? ' (downloading as LOSSLESS for conversion)' : ''}',
|
||||||
);
|
);
|
||||||
_log.d('Output dir: $outputDir');
|
_log.d('Output dir: $outputDir');
|
||||||
result = await PlatformBridge.downloadWithExtensions(
|
result = await PlatformBridge.downloadWithExtensions(
|
||||||
@@ -1730,7 +1871,7 @@ class DownloadQueueNotifier extends Notifier<DownloadQueueState> {
|
|||||||
coverUrl: trackToDownload.coverUrl,
|
coverUrl: trackToDownload.coverUrl,
|
||||||
outputDir: outputDir,
|
outputDir: outputDir,
|
||||||
filenameFormat: state.filenameFormat,
|
filenameFormat: state.filenameFormat,
|
||||||
quality: quality,
|
quality: downloadQuality,
|
||||||
trackNumber: trackToDownload.trackNumber ?? 1,
|
trackNumber: trackToDownload.trackNumber ?? 1,
|
||||||
discNumber: trackToDownload.discNumber ?? 1,
|
discNumber: trackToDownload.discNumber ?? 1,
|
||||||
releaseDate: trackToDownload.releaseDate,
|
releaseDate: trackToDownload.releaseDate,
|
||||||
@@ -1744,7 +1885,7 @@ class DownloadQueueNotifier extends Notifier<DownloadQueueState> {
|
|||||||
} else if (state.autoFallback) {
|
} else if (state.autoFallback) {
|
||||||
_log.d('Using auto-fallback mode');
|
_log.d('Using auto-fallback mode');
|
||||||
_log.d(
|
_log.d(
|
||||||
'Quality: $quality${item.qualityOverride != null ? ' (override)' : ''}',
|
'Quality: $quality${item.qualityOverride != null ? ' (override)' : ''}${quality == 'LOSSY' ? ' (downloading as LOSSLESS for conversion)' : ''}',
|
||||||
);
|
);
|
||||||
_log.d('Output dir: $outputDir');
|
_log.d('Output dir: $outputDir');
|
||||||
result = await PlatformBridge.downloadWithFallback(
|
result = await PlatformBridge.downloadWithFallback(
|
||||||
@@ -1757,7 +1898,7 @@ class DownloadQueueNotifier extends Notifier<DownloadQueueState> {
|
|||||||
coverUrl: trackToDownload.coverUrl,
|
coverUrl: trackToDownload.coverUrl,
|
||||||
outputDir: outputDir,
|
outputDir: outputDir,
|
||||||
filenameFormat: state.filenameFormat,
|
filenameFormat: state.filenameFormat,
|
||||||
quality: quality,
|
quality: downloadQuality,
|
||||||
trackNumber: trackToDownload.trackNumber ?? 1,
|
trackNumber: trackToDownload.trackNumber ?? 1,
|
||||||
discNumber: trackToDownload.discNumber ?? 1,
|
discNumber: trackToDownload.discNumber ?? 1,
|
||||||
releaseDate: trackToDownload.releaseDate,
|
releaseDate: trackToDownload.releaseDate,
|
||||||
@@ -1780,7 +1921,7 @@ class DownloadQueueNotifier extends Notifier<DownloadQueueState> {
|
|||||||
coverUrl: trackToDownload.coverUrl,
|
coverUrl: trackToDownload.coverUrl,
|
||||||
outputDir: outputDir,
|
outputDir: outputDir,
|
||||||
filenameFormat: state.filenameFormat,
|
filenameFormat: state.filenameFormat,
|
||||||
quality: quality,
|
quality: downloadQuality,
|
||||||
trackNumber: trackToDownload.trackNumber ?? 1,
|
trackNumber: trackToDownload.trackNumber ?? 1,
|
||||||
discNumber: trackToDownload.discNumber ?? 1,
|
discNumber: trackToDownload.discNumber ?? 1,
|
||||||
releaseDate: trackToDownload.releaseDate,
|
releaseDate: trackToDownload.releaseDate,
|
||||||
@@ -1965,11 +2106,13 @@ class DownloadQueueNotifier extends Notifier<DownloadQueueState> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (quality == 'MP3' && filePath != null && filePath.endsWith('.flac')) {
|
if (quality == 'LOSSY' && filePath != null && filePath.endsWith('.flac')) {
|
||||||
if (wasExisting) {
|
if (wasExisting) {
|
||||||
_log.i('MP3 requested but existing FLAC found - skipping conversion to preserve original file');
|
_log.i('Lossy requested but existing FLAC found - skipping conversion to preserve original file');
|
||||||
} else {
|
} else {
|
||||||
_log.i('MP3 quality selected, converting FLAC to MP3...');
|
final lossyFormat = settings.lossyFormat;
|
||||||
|
final lossyBitrate = settings.lossyBitrate;
|
||||||
|
_log.i('Lossy quality selected, converting FLAC to $lossyFormat ($lossyBitrate)...');
|
||||||
updateItemStatus(
|
updateItemStatus(
|
||||||
item.id,
|
item.id,
|
||||||
DownloadStatus.downloading,
|
DownloadStatus.downloading,
|
||||||
@@ -1977,40 +2120,56 @@ class DownloadQueueNotifier extends Notifier<DownloadQueueState> {
|
|||||||
);
|
);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
final mp3Path = await FFmpegService.convertFlacToMp3(
|
final convertedPath = await FFmpegService.convertFlacToLossy(
|
||||||
filePath,
|
filePath,
|
||||||
bitrate: '320k',
|
format: lossyFormat,
|
||||||
|
bitrate: lossyBitrate,
|
||||||
deleteOriginal: true,
|
deleteOriginal: true,
|
||||||
);
|
);
|
||||||
|
|
||||||
if (mp3Path != null) {
|
if (convertedPath != null) {
|
||||||
filePath = mp3Path;
|
filePath = convertedPath;
|
||||||
actualQuality = 'MP3 320kbps';
|
// Extract bitrate for display (e.g., 'mp3_320' -> '320kbps')
|
||||||
_log.i('Successfully converted to MP3: $mp3Path');
|
final bitrateDisplay = lossyBitrate.contains('_')
|
||||||
|
? '${lossyBitrate.split('_').last}kbps'
|
||||||
|
: (lossyFormat == 'opus' ? '128kbps' : '320kbps');
|
||||||
|
actualQuality = '${lossyFormat.toUpperCase()} $bitrateDisplay';
|
||||||
|
_log.i('Successfully converted to $lossyFormat ($bitrateDisplay): $convertedPath');
|
||||||
|
|
||||||
_log.i('Embedding metadata to MP3...');
|
// Embed metadata and cover for both MP3 and Opus
|
||||||
|
_log.i('Embedding metadata to $lossyFormat...');
|
||||||
updateItemStatus(
|
updateItemStatus(
|
||||||
item.id,
|
item.id,
|
||||||
DownloadStatus.downloading,
|
DownloadStatus.downloading,
|
||||||
progress: 0.99,
|
progress: 0.99,
|
||||||
);
|
);
|
||||||
|
|
||||||
final mp3BackendGenre = result['genre'] as String?;
|
final lossyBackendGenre = result['genre'] as String?;
|
||||||
final mp3BackendLabel = result['label'] as String?;
|
final lossyBackendLabel = result['label'] as String?;
|
||||||
final mp3BackendCopyright = result['copyright'] as String?;
|
final lossyBackendCopyright = result['copyright'] as String?;
|
||||||
|
|
||||||
await _embedMetadataToMp3(
|
if (lossyFormat == 'mp3') {
|
||||||
mp3Path,
|
await _embedMetadataToMp3(
|
||||||
trackToDownload,
|
convertedPath,
|
||||||
genre: mp3BackendGenre ?? genre,
|
trackToDownload,
|
||||||
label: mp3BackendLabel ?? label,
|
genre: lossyBackendGenre ?? genre,
|
||||||
copyright: mp3BackendCopyright,
|
label: lossyBackendLabel ?? label,
|
||||||
);
|
copyright: lossyBackendCopyright,
|
||||||
|
);
|
||||||
|
} else if (lossyFormat == 'opus') {
|
||||||
|
await _embedMetadataToOpus(
|
||||||
|
convertedPath,
|
||||||
|
trackToDownload,
|
||||||
|
genre: lossyBackendGenre ?? genre,
|
||||||
|
label: lossyBackendLabel ?? label,
|
||||||
|
copyright: lossyBackendCopyright,
|
||||||
|
);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
_log.w('MP3 conversion failed, keeping FLAC file');
|
_log.w('$lossyFormat conversion failed, keeping FLAC file');
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
_log.e('MP3 conversion error: $e, keeping FLAC file');
|
_log.e('Lossy conversion error: $e, keeping FLAC file');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -146,6 +146,26 @@ class Extension {
|
|||||||
bool get hasBrowseCategories => capabilities['browseCategories'] == true;
|
bool get hasBrowseCategories => capabilities['browseCategories'] == true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class SearchFilter {
|
||||||
|
final String id;
|
||||||
|
final String? label;
|
||||||
|
final String? icon;
|
||||||
|
|
||||||
|
const SearchFilter({
|
||||||
|
required this.id,
|
||||||
|
this.label,
|
||||||
|
this.icon,
|
||||||
|
});
|
||||||
|
|
||||||
|
factory SearchFilter.fromJson(Map<String, dynamic> json) {
|
||||||
|
return SearchFilter(
|
||||||
|
id: json['id'] as String? ?? '',
|
||||||
|
label: json['label'] as String?,
|
||||||
|
icon: json['icon'] as String?,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class SearchBehavior {
|
class SearchBehavior {
|
||||||
final bool enabled;
|
final bool enabled;
|
||||||
final String? placeholder;
|
final String? placeholder;
|
||||||
@@ -154,6 +174,7 @@ class SearchBehavior {
|
|||||||
final String? thumbnailRatio; // "square" (1:1), "wide" (16:9), "portrait" (2:3)
|
final String? thumbnailRatio; // "square" (1:1), "wide" (16:9), "portrait" (2:3)
|
||||||
final int? thumbnailWidth;
|
final int? thumbnailWidth;
|
||||||
final int? thumbnailHeight;
|
final int? thumbnailHeight;
|
||||||
|
final List<SearchFilter> filters; // Available search filters (e.g., track, album, artist, playlist)
|
||||||
|
|
||||||
const SearchBehavior({
|
const SearchBehavior({
|
||||||
required this.enabled,
|
required this.enabled,
|
||||||
@@ -163,6 +184,7 @@ class SearchBehavior {
|
|||||||
this.thumbnailRatio,
|
this.thumbnailRatio,
|
||||||
this.thumbnailWidth,
|
this.thumbnailWidth,
|
||||||
this.thumbnailHeight,
|
this.thumbnailHeight,
|
||||||
|
this.filters = const [],
|
||||||
});
|
});
|
||||||
|
|
||||||
factory SearchBehavior.fromJson(Map<String, dynamic> json) {
|
factory SearchBehavior.fromJson(Map<String, dynamic> json) {
|
||||||
@@ -174,6 +196,9 @@ class SearchBehavior {
|
|||||||
thumbnailRatio: json['thumbnailRatio'] as String?,
|
thumbnailRatio: json['thumbnailRatio'] as String?,
|
||||||
thumbnailWidth: json['thumbnailWidth'] as int?,
|
thumbnailWidth: json['thumbnailWidth'] as int?,
|
||||||
thumbnailHeight: json['thumbnailHeight'] as int?,
|
thumbnailHeight: json['thumbnailHeight'] as int?,
|
||||||
|
filters: (json['filters'] as List<dynamic>?)
|
||||||
|
?.map((f) => SearchFilter.fromJson(f as Map<String, dynamic>))
|
||||||
|
.toList() ?? [],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -122,7 +122,8 @@ class RecentAccessNotifier extends Notifier<RecentAccessState> {
|
|||||||
items = decoded
|
items = decoded
|
||||||
.map((e) => RecentAccessItem.fromJson(e as Map<String, dynamic>))
|
.map((e) => RecentAccessItem.fromJson(e as Map<String, dynamic>))
|
||||||
.toList();
|
.toList();
|
||||||
} catch (e) {
|
} catch (_) {
|
||||||
|
// Ignore JSON parse errors, use empty list
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -231,14 +231,31 @@ class SettingsNotifier extends Notifier<AppSettings> {
|
|||||||
_saveSettings();
|
_saveSettings();
|
||||||
}
|
}
|
||||||
|
|
||||||
void setEnableMp3Option(bool enabled) {
|
void setEnableLossyOption(bool enabled) {
|
||||||
state = state.copyWith(enableMp3Option: enabled);
|
state = state.copyWith(enableLossyOption: enabled);
|
||||||
// If MP3 is disabled and current quality is MP3, reset to LOSSLESS
|
// If Lossy is disabled and current quality is LOSSY, reset to LOSSLESS
|
||||||
if (!enabled && state.audioQuality == 'MP3') {
|
if (!enabled && state.audioQuality == 'LOSSY') {
|
||||||
state = state.copyWith(audioQuality: 'LOSSLESS');
|
state = state.copyWith(audioQuality: 'LOSSLESS');
|
||||||
}
|
}
|
||||||
_saveSettings();
|
_saveSettings();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void setLossyFormat(String format) {
|
||||||
|
state = state.copyWith(lossyFormat: format);
|
||||||
|
_saveSettings();
|
||||||
|
}
|
||||||
|
|
||||||
|
void setLossyBitrate(String bitrate) {
|
||||||
|
// Extract format from bitrate (e.g., 'mp3_320' -> 'mp3')
|
||||||
|
final format = bitrate.split('_').first;
|
||||||
|
state = state.copyWith(lossyBitrate: bitrate, lossyFormat: format);
|
||||||
|
_saveSettings();
|
||||||
|
}
|
||||||
|
|
||||||
|
void setUseAllFilesAccess(bool enabled) {
|
||||||
|
state = state.copyWith(useAllFilesAccess: enabled);
|
||||||
|
_saveSettings();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
final settingsProvider = NotifierProvider<SettingsNotifier, AppSettings>(
|
final settingsProvider = NotifierProvider<SettingsNotifier, AppSettings>(
|
||||||
|
|||||||
@@ -22,9 +22,12 @@ class TrackState {
|
|||||||
final List<ArtistAlbum>? artistAlbums; // For artist page
|
final List<ArtistAlbum>? artistAlbums; // For artist page
|
||||||
final List<Track>? artistTopTracks; // Artist's popular tracks
|
final List<Track>? artistTopTracks; // Artist's popular tracks
|
||||||
final List<SearchArtist>? searchArtists; // For search results
|
final List<SearchArtist>? searchArtists; // For search results
|
||||||
|
final List<SearchAlbum>? searchAlbums; // For search results (albums)
|
||||||
|
final List<SearchPlaylist>? searchPlaylists; // For search results (playlists)
|
||||||
final bool hasSearchText; // For back button handling
|
final bool hasSearchText; // For back button handling
|
||||||
final bool isShowingRecentAccess; // For recent access mode
|
final bool isShowingRecentAccess; // For recent access mode
|
||||||
final String? searchExtensionId; // Extension ID used for current search results
|
final String? searchExtensionId; // Extension ID used for current search results
|
||||||
|
final String? selectedSearchFilter; // Currently selected search filter (e.g., "track", "album", "artist", "playlist")
|
||||||
|
|
||||||
const TrackState({
|
const TrackState({
|
||||||
this.tracks = const [],
|
this.tracks = const [],
|
||||||
@@ -41,12 +44,15 @@ class TrackState {
|
|||||||
this.artistAlbums,
|
this.artistAlbums,
|
||||||
this.artistTopTracks,
|
this.artistTopTracks,
|
||||||
this.searchArtists,
|
this.searchArtists,
|
||||||
|
this.searchAlbums,
|
||||||
|
this.searchPlaylists,
|
||||||
this.hasSearchText = false,
|
this.hasSearchText = false,
|
||||||
this.isShowingRecentAccess = false,
|
this.isShowingRecentAccess = false,
|
||||||
this.searchExtensionId,
|
this.searchExtensionId,
|
||||||
|
this.selectedSearchFilter,
|
||||||
});
|
});
|
||||||
|
|
||||||
bool get hasContent => tracks.isNotEmpty || artistAlbums != null || (searchArtists != null && searchArtists!.isNotEmpty);
|
bool get hasContent => tracks.isNotEmpty || artistAlbums != null || (searchArtists != null && searchArtists!.isNotEmpty) || (searchAlbums != null && searchAlbums!.isNotEmpty) || (searchPlaylists != null && searchPlaylists!.isNotEmpty);
|
||||||
|
|
||||||
TrackState copyWith({
|
TrackState copyWith({
|
||||||
List<Track>? tracks,
|
List<Track>? tracks,
|
||||||
@@ -63,9 +69,13 @@ class TrackState {
|
|||||||
List<ArtistAlbum>? artistAlbums,
|
List<ArtistAlbum>? artistAlbums,
|
||||||
List<Track>? artistTopTracks,
|
List<Track>? artistTopTracks,
|
||||||
List<SearchArtist>? searchArtists,
|
List<SearchArtist>? searchArtists,
|
||||||
|
List<SearchAlbum>? searchAlbums,
|
||||||
|
List<SearchPlaylist>? searchPlaylists,
|
||||||
bool? hasSearchText,
|
bool? hasSearchText,
|
||||||
bool? isShowingRecentAccess,
|
bool? isShowingRecentAccess,
|
||||||
String? searchExtensionId,
|
String? searchExtensionId,
|
||||||
|
String? selectedSearchFilter,
|
||||||
|
bool clearSelectedSearchFilter = false,
|
||||||
}) {
|
}) {
|
||||||
return TrackState(
|
return TrackState(
|
||||||
tracks: tracks ?? this.tracks,
|
tracks: tracks ?? this.tracks,
|
||||||
@@ -82,9 +92,12 @@ class TrackState {
|
|||||||
artistAlbums: artistAlbums ?? this.artistAlbums,
|
artistAlbums: artistAlbums ?? this.artistAlbums,
|
||||||
artistTopTracks: artistTopTracks ?? this.artistTopTracks,
|
artistTopTracks: artistTopTracks ?? this.artistTopTracks,
|
||||||
searchArtists: searchArtists ?? this.searchArtists,
|
searchArtists: searchArtists ?? this.searchArtists,
|
||||||
|
searchAlbums: searchAlbums ?? this.searchAlbums,
|
||||||
|
searchPlaylists: searchPlaylists ?? this.searchPlaylists,
|
||||||
hasSearchText: hasSearchText ?? this.hasSearchText,
|
hasSearchText: hasSearchText ?? this.hasSearchText,
|
||||||
isShowingRecentAccess: isShowingRecentAccess ?? this.isShowingRecentAccess,
|
isShowingRecentAccess: isShowingRecentAccess ?? this.isShowingRecentAccess,
|
||||||
searchExtensionId: searchExtensionId,
|
searchExtensionId: searchExtensionId,
|
||||||
|
selectedSearchFilter: clearSelectedSearchFilter ? null : (selectedSearchFilter ?? this.selectedSearchFilter),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -127,6 +140,42 @@ class SearchArtist {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class SearchAlbum {
|
||||||
|
final String id;
|
||||||
|
final String name;
|
||||||
|
final String artists;
|
||||||
|
final String? imageUrl;
|
||||||
|
final String? releaseDate;
|
||||||
|
final int totalTracks;
|
||||||
|
final String albumType;
|
||||||
|
|
||||||
|
const SearchAlbum({
|
||||||
|
required this.id,
|
||||||
|
required this.name,
|
||||||
|
required this.artists,
|
||||||
|
this.imageUrl,
|
||||||
|
this.releaseDate,
|
||||||
|
required this.totalTracks,
|
||||||
|
required this.albumType,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
class SearchPlaylist {
|
||||||
|
final String id;
|
||||||
|
final String name;
|
||||||
|
final String owner;
|
||||||
|
final String? imageUrl;
|
||||||
|
final int totalTracks;
|
||||||
|
|
||||||
|
const SearchPlaylist({
|
||||||
|
required this.id,
|
||||||
|
required this.name,
|
||||||
|
required this.owner,
|
||||||
|
this.imageUrl,
|
||||||
|
required this.totalTracks,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
class TrackNotifier extends Notifier<TrackState> {
|
class TrackNotifier extends Notifier<TrackState> {
|
||||||
int _currentRequestId = 0;
|
int _currentRequestId = 0;
|
||||||
|
|
||||||
@@ -268,10 +317,13 @@ class TrackNotifier extends Notifier<TrackState> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> search(String query, {String? metadataSource}) async {
|
Future<void> search(String query, {String? metadataSource, String? filterOverride}) async {
|
||||||
final requestId = ++_currentRequestId;
|
final requestId = ++_currentRequestId;
|
||||||
|
|
||||||
|
// Preserve selected filter during loading
|
||||||
|
final currentFilter = filterOverride ?? state.selectedSearchFilter;
|
||||||
|
|
||||||
state = TrackState(isLoading: true, hasSearchText: state.hasSearchText);
|
state = TrackState(isLoading: true, hasSearchText: state.hasSearchText, selectedSearchFilter: currentFilter);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
final settings = ref.read(settingsProvider);
|
final settings = ref.read(settingsProvider);
|
||||||
@@ -289,7 +341,7 @@ class TrackNotifier extends Notifier<TrackState> {
|
|||||||
final source = metadataSource ?? 'deezer';
|
final source = metadataSource ?? 'deezer';
|
||||||
|
|
||||||
_log.i(
|
_log.i(
|
||||||
'Search started: source=$source, query="$query", useExtensions=$useExtensions',
|
'Search started: source=$source, query="$query", useExtensions=$useExtensions, filter=$currentFilter',
|
||||||
);
|
);
|
||||||
|
|
||||||
Map<String, dynamic> results;
|
Map<String, dynamic> results;
|
||||||
@@ -315,11 +367,11 @@ class TrackNotifier extends Notifier<TrackState> {
|
|||||||
|
|
||||||
if (source == 'deezer') {
|
if (source == 'deezer') {
|
||||||
_log.d('Calling Deezer search API...');
|
_log.d('Calling Deezer search API...');
|
||||||
results = await PlatformBridge.searchDeezerAll(query, trackLimit: 20, artistLimit: 5);
|
results = await PlatformBridge.searchDeezerAll(query, trackLimit: 20, artistLimit: 2, filter: currentFilter);
|
||||||
_log.i('Deezer returned ${(results['tracks'] as List?)?.length ?? 0} tracks, ${(results['artists'] as List?)?.length ?? 0} artists');
|
_log.i('Deezer returned ${(results['tracks'] as List?)?.length ?? 0} tracks, ${(results['artists'] as List?)?.length ?? 0} artists, ${(results['albums'] as List?)?.length ?? 0} albums');
|
||||||
} else {
|
} else {
|
||||||
_log.d('Calling Spotify search API...');
|
_log.d('Calling Spotify search API...');
|
||||||
results = await PlatformBridge.searchSpotifyAll(query, trackLimit: 20, artistLimit: 5);
|
results = await PlatformBridge.searchSpotifyAll(query, trackLimit: 20, artistLimit: 2);
|
||||||
_log.i('Spotify returned ${(results['tracks'] as List?)?.length ?? 0} tracks, ${(results['artists'] as List?)?.length ?? 0} artists');
|
_log.i('Spotify returned ${(results['tracks'] as List?)?.length ?? 0} tracks, ${(results['artists'] as List?)?.length ?? 0} artists');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -330,8 +382,9 @@ class TrackNotifier extends Notifier<TrackState> {
|
|||||||
|
|
||||||
final trackList = results['tracks'] as List<dynamic>? ?? [];
|
final trackList = results['tracks'] as List<dynamic>? ?? [];
|
||||||
final artistList = results['artists'] as List<dynamic>? ?? [];
|
final artistList = results['artists'] as List<dynamic>? ?? [];
|
||||||
|
final albumList = results['albums'] as List<dynamic>? ?? [];
|
||||||
|
|
||||||
_log.d('Raw results: ${trackList.length} tracks, ${artistList.length} artists');
|
_log.d('Raw results: ${trackList.length} tracks, ${artistList.length} artists, ${albumList.length} albums');
|
||||||
|
|
||||||
final tracks = <Track>[];
|
final tracks = <Track>[];
|
||||||
|
|
||||||
@@ -373,25 +426,61 @@ class TrackNotifier extends Notifier<TrackState> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_log.i('Search complete: ${tracks.length} tracks (${extensionTracks.length} from extensions), ${artists.length} artists parsed successfully');
|
final albums = <SearchAlbum>[];
|
||||||
|
for (int i = 0; i < albumList.length; i++) {
|
||||||
|
final a = albumList[i];
|
||||||
|
try {
|
||||||
|
if (a is Map<String, dynamic>) {
|
||||||
|
albums.add(_parseSearchAlbum(a));
|
||||||
|
} else {
|
||||||
|
_log.w('Album[$i] is not a Map: ${a.runtimeType}');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
_log.e('Failed to parse album[$i]: $e', e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
final playlistList = results['playlists'] as List<dynamic>? ?? [];
|
||||||
|
final playlists = <SearchPlaylist>[];
|
||||||
|
for (int i = 0; i < playlistList.length; i++) {
|
||||||
|
final p = playlistList[i];
|
||||||
|
try {
|
||||||
|
if (p is Map<String, dynamic>) {
|
||||||
|
playlists.add(_parseSearchPlaylist(p));
|
||||||
|
} else {
|
||||||
|
_log.w('Playlist[$i] is not a Map: ${p.runtimeType}');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
_log.e('Failed to parse playlist[$i]: $e', e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_log.i('Search complete: ${tracks.length} tracks (${extensionTracks.length} from extensions), ${artists.length} artists, ${albums.length} albums, ${playlists.length} playlists parsed successfully');
|
||||||
|
|
||||||
state = TrackState(
|
state = TrackState(
|
||||||
tracks: tracks,
|
tracks: tracks,
|
||||||
searchArtists: artists,
|
searchArtists: artists,
|
||||||
|
searchAlbums: albums,
|
||||||
|
searchPlaylists: playlists,
|
||||||
isLoading: false,
|
isLoading: false,
|
||||||
hasSearchText: state.hasSearchText,
|
hasSearchText: state.hasSearchText,
|
||||||
|
selectedSearchFilter: currentFilter, // Preserve filter in results
|
||||||
);
|
);
|
||||||
} catch (e, stackTrace) {
|
} catch (e, stackTrace) {
|
||||||
if (!_isRequestValid(requestId)) return;
|
if (!_isRequestValid(requestId)) return;
|
||||||
_log.e('Search failed: $e', e, stackTrace);
|
_log.e('Search failed: $e', e, stackTrace);
|
||||||
state = TrackState(isLoading: false, error: e.toString(), hasSearchText: state.hasSearchText);
|
state = TrackState(isLoading: false, error: e.toString(), hasSearchText: state.hasSearchText, selectedSearchFilter: currentFilter);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> customSearch(String extensionId, String query, {Map<String, dynamic>? options}) async {
|
Future<void> customSearch(String extensionId, String query, {Map<String, dynamic>? options}) async {
|
||||||
final requestId = ++_currentRequestId;
|
final requestId = ++_currentRequestId;
|
||||||
|
|
||||||
state = TrackState(isLoading: true, hasSearchText: state.hasSearchText);
|
state = TrackState(
|
||||||
|
isLoading: true,
|
||||||
|
hasSearchText: state.hasSearchText,
|
||||||
|
selectedSearchFilter: state.selectedSearchFilter, // Preserve filter during loading
|
||||||
|
);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
_log.i('Custom search started: extension=$extensionId, query="$query"');
|
_log.i('Custom search started: extension=$extensionId, query="$query"');
|
||||||
@@ -423,6 +512,7 @@ class TrackNotifier extends Notifier<TrackState> {
|
|||||||
isLoading: false,
|
isLoading: false,
|
||||||
hasSearchText: state.hasSearchText,
|
hasSearchText: state.hasSearchText,
|
||||||
searchExtensionId: extensionId, // Store which extension was used
|
searchExtensionId: extensionId, // Store which extension was used
|
||||||
|
selectedSearchFilter: state.selectedSearchFilter, // Preserve selected filter
|
||||||
);
|
);
|
||||||
} catch (e, stackTrace) {
|
} catch (e, stackTrace) {
|
||||||
if (!_isRequestValid(requestId)) return;
|
if (!_isRequestValid(requestId)) return;
|
||||||
@@ -474,6 +564,15 @@ class TrackNotifier extends Notifier<TrackState> {
|
|||||||
state = const TrackState();
|
state = const TrackState();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Set selected search filter for extension search
|
||||||
|
void setSearchFilter(String? filter) {
|
||||||
|
if (state.selectedSearchFilter == filter) return;
|
||||||
|
state = state.copyWith(
|
||||||
|
selectedSearchFilter: filter,
|
||||||
|
clearSelectedSearchFilter: filter == null,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
/// Set search text state for back button handling
|
/// Set search text state for back button handling
|
||||||
void setSearchText(bool hasText) {
|
void setSearchText(bool hasText) {
|
||||||
if (state.hasSearchText == hasText) {
|
if (state.hasSearchText == hasText) {
|
||||||
@@ -571,6 +670,28 @@ class TrackNotifier extends Notifier<TrackState> {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
SearchAlbum _parseSearchAlbum(Map<String, dynamic> data) {
|
||||||
|
return SearchAlbum(
|
||||||
|
id: data['id'] as String? ?? '',
|
||||||
|
name: data['name'] as String? ?? '',
|
||||||
|
artists: data['artists'] as String? ?? '',
|
||||||
|
imageUrl: data['images'] as String?,
|
||||||
|
releaseDate: data['release_date'] as String?,
|
||||||
|
totalTracks: data['total_tracks'] as int? ?? 0,
|
||||||
|
albumType: data['album_type'] as String? ?? 'album',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
SearchPlaylist _parseSearchPlaylist(Map<String, dynamic> data) {
|
||||||
|
return SearchPlaylist(
|
||||||
|
id: data['id'] as String? ?? '',
|
||||||
|
name: data['name'] as String? ?? '',
|
||||||
|
owner: data['owner'] as String? ?? '',
|
||||||
|
imageUrl: data['images'] as String?,
|
||||||
|
totalTracks: data['total_tracks'] as int? ?? 0,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
void _preWarmCacheForTracks(List<Track> tracks) {
|
void _preWarmCacheForTracks(List<Track> tracks) {
|
||||||
final tracksWithIsrc = tracks.where((t) => t.isrc != null && t.isrc!.isNotEmpty).toList();
|
final tracksWithIsrc = tracks.where((t) => t.isrc != null && t.isrc!.isNotEmpty).toList();
|
||||||
if (tracksWithIsrc.isEmpty) return;
|
if (tracksWithIsrc.isEmpty) return;
|
||||||
|
|||||||
+614
-87
@@ -204,12 +204,14 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _performSearch(String query) async {
|
Future<void> _performSearch(String query, {String? filterOverride}) async {
|
||||||
final settings = ref.read(settingsProvider);
|
final settings = ref.read(settingsProvider);
|
||||||
final extState = ref.read(extensionProvider);
|
final extState = ref.read(extensionProvider);
|
||||||
final searchProvider = settings.searchProvider;
|
final searchProvider = settings.searchProvider;
|
||||||
|
// Use filterOverride if provided, otherwise read from state
|
||||||
|
final selectedFilter = filterOverride ?? ref.read(trackProvider).selectedSearchFilter;
|
||||||
|
|
||||||
final searchKey = '${searchProvider ?? 'default'}:$query';
|
final searchKey = '${searchProvider ?? 'default'}:$query:${selectedFilter ?? 'all'}';
|
||||||
if (_lastSearchQuery == searchKey) return;
|
if (_lastSearchQuery == searchKey) return;
|
||||||
_lastSearchQuery = searchKey;
|
_lastSearchQuery = searchKey;
|
||||||
|
|
||||||
@@ -218,12 +220,17 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
extState.extensions.any((e) => e.id == searchProvider && e.enabled);
|
extState.extensions.any((e) => e.id == searchProvider && e.enabled);
|
||||||
|
|
||||||
if (isExtensionEnabled) {
|
if (isExtensionEnabled) {
|
||||||
await ref.read(trackProvider.notifier).customSearch(searchProvider, query);
|
// Build options with filter if selected
|
||||||
|
Map<String, dynamic>? options;
|
||||||
|
if (selectedFilter != null) {
|
||||||
|
options = {'filter': selectedFilter};
|
||||||
|
}
|
||||||
|
await ref.read(trackProvider.notifier).customSearch(searchProvider, query, options: options);
|
||||||
} else {
|
} else {
|
||||||
if (searchProvider != null && searchProvider.isNotEmpty && !isExtensionEnabled) {
|
if (searchProvider != null && searchProvider.isNotEmpty && !isExtensionEnabled) {
|
||||||
ref.read(settingsProvider.notifier).setSearchProvider(null);
|
ref.read(settingsProvider.notifier).setSearchProvider(null);
|
||||||
}
|
}
|
||||||
await ref.read(trackProvider.notifier).search(query, metadataSource: settings.metadataSource);
|
await ref.read(trackProvider.notifier).search(query, metadataSource: settings.metadataSource, filterOverride: selectedFilter);
|
||||||
}
|
}
|
||||||
ref.read(settingsProvider.notifier).setHasSearchedBefore();
|
ref.read(settingsProvider.notifier).setHasSearchedBefore();
|
||||||
}
|
}
|
||||||
@@ -460,6 +467,8 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
|
|
||||||
final tracks = ref.watch(trackProvider.select((s) => s.tracks));
|
final tracks = ref.watch(trackProvider.select((s) => s.tracks));
|
||||||
final searchArtists = ref.watch(trackProvider.select((s) => s.searchArtists));
|
final searchArtists = ref.watch(trackProvider.select((s) => s.searchArtists));
|
||||||
|
final searchAlbums = ref.watch(trackProvider.select((s) => s.searchAlbums));
|
||||||
|
final searchPlaylists = ref.watch(trackProvider.select((s) => s.searchPlaylists));
|
||||||
final isLoading = ref.watch(trackProvider.select((s) => s.isLoading));
|
final isLoading = ref.watch(trackProvider.select((s) => s.isLoading));
|
||||||
final error = ref.watch(trackProvider.select((s) => s.error));
|
final error = ref.watch(trackProvider.select((s) => s.error));
|
||||||
final hasSearchedBefore = ref.watch(settingsProvider.select((s) => s.hasSearchedBefore));
|
final hasSearchedBefore = ref.watch(settingsProvider.select((s) => s.hasSearchedBefore));
|
||||||
@@ -475,7 +484,7 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
));
|
));
|
||||||
|
|
||||||
final colorScheme = Theme.of(context).colorScheme;
|
final colorScheme = Theme.of(context).colorScheme;
|
||||||
final hasActualResults = tracks.isNotEmpty || (searchArtists != null && searchArtists.isNotEmpty);
|
final hasActualResults = tracks.isNotEmpty || (searchArtists != null && searchArtists.isNotEmpty) || (searchAlbums != null && searchAlbums.isNotEmpty) || (searchPlaylists != null && searchPlaylists.isNotEmpty);
|
||||||
final isShowingRecentAccess = ref.watch(trackProvider.select((s) => s.isShowingRecentAccess));
|
final isShowingRecentAccess = ref.watch(trackProvider.select((s) => s.isShowingRecentAccess));
|
||||||
final hasResults = isShowingRecentAccess || hasActualResults || isLoading;
|
final hasResults = isShowingRecentAccess || hasActualResults || isLoading;
|
||||||
final mediaQuery = MediaQuery.of(context);
|
final mediaQuery = MediaQuery.of(context);
|
||||||
@@ -495,6 +504,34 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
final hasExploreContent = exploreSections.isNotEmpty;
|
final hasExploreContent = exploreSections.isNotEmpty;
|
||||||
final showExplore = !hasActualResults && !isLoading && !showRecentAccess && hasHomeFeedExtension && hasExploreContent;
|
final showExplore = !hasActualResults && !isLoading && !showRecentAccess && hasHomeFeedExtension && hasExploreContent;
|
||||||
|
|
||||||
|
// Get current search extension and its filters
|
||||||
|
final settings = ref.watch(settingsProvider);
|
||||||
|
final extState = ref.watch(extensionProvider);
|
||||||
|
final currentSearchProvider = settings.searchProvider;
|
||||||
|
final selectedSearchFilter = ref.watch(trackProvider.select((s) => s.selectedSearchFilter));
|
||||||
|
Extension? currentSearchExtension;
|
||||||
|
List<SearchFilter> searchFilters = [];
|
||||||
|
|
||||||
|
// Check if using extension search provider
|
||||||
|
final isUsingExtensionSearch = currentSearchProvider != null &&
|
||||||
|
currentSearchProvider.isNotEmpty &&
|
||||||
|
extState.extensions.any((e) => e.id == currentSearchProvider && e.enabled);
|
||||||
|
|
||||||
|
if (isUsingExtensionSearch) {
|
||||||
|
currentSearchExtension = extState.extensions.where((e) => e.id == currentSearchProvider && e.enabled).firstOrNull;
|
||||||
|
if (currentSearchExtension?.searchBehavior?.filters.isNotEmpty == true) {
|
||||||
|
searchFilters = currentSearchExtension!.searchBehavior!.filters;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Default Deezer filters
|
||||||
|
searchFilters = const [
|
||||||
|
SearchFilter(id: 'track', label: 'Tracks', icon: 'music'),
|
||||||
|
SearchFilter(id: 'artist', label: 'Artists', icon: 'artist'),
|
||||||
|
SearchFilter(id: 'album', label: 'Albums', icon: 'album'),
|
||||||
|
SearchFilter(id: 'playlist', label: 'Playlists', icon: 'playlist'),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
if (hasActualResults && isShowingRecentAccess) {
|
if (hasActualResults && isShowingRecentAccess) {
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||||
if (mounted) ref.read(trackProvider.notifier).setShowingRecentAccess(false);
|
if (mounted) ref.read(trackProvider.notifier).setShowingRecentAccess(false);
|
||||||
@@ -603,6 +640,16 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
|
// Search filter bar (only shown when has search results or loading search)
|
||||||
|
if (searchFilters.isNotEmpty && (hasActualResults || isLoading))
|
||||||
|
SliverToBoxAdapter(
|
||||||
|
child: _buildSearchFilterBar(
|
||||||
|
searchFilters,
|
||||||
|
selectedSearchFilter,
|
||||||
|
colorScheme,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
if (showRecentAccess)
|
if (showRecentAccess)
|
||||||
SliverToBoxAdapter(
|
SliverToBoxAdapter(
|
||||||
child: _buildRecentAccess(recentAccessView!, colorScheme),
|
child: _buildRecentAccess(recentAccessView!, colorScheme),
|
||||||
@@ -651,6 +698,8 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
..._buildSearchResults(
|
..._buildSearchResults(
|
||||||
tracks: tracks,
|
tracks: tracks,
|
||||||
searchArtists: searchArtists,
|
searchArtists: searchArtists,
|
||||||
|
searchAlbums: searchAlbums,
|
||||||
|
searchPlaylists: searchPlaylists,
|
||||||
isLoading: isLoading,
|
isLoading: isLoading,
|
||||||
error: error,
|
error: error,
|
||||||
colorScheme: colorScheme,
|
colorScheme: colorScheme,
|
||||||
@@ -1529,6 +1578,8 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
List<Widget> _buildSearchResults({
|
List<Widget> _buildSearchResults({
|
||||||
required List<Track> tracks,
|
required List<Track> tracks,
|
||||||
required List<SearchArtist>? searchArtists,
|
required List<SearchArtist>? searchArtists,
|
||||||
|
required List<SearchAlbum>? searchAlbums,
|
||||||
|
required List<SearchPlaylist>? searchPlaylists,
|
||||||
required bool isLoading,
|
required bool isLoading,
|
||||||
required String? error,
|
required String? error,
|
||||||
required ColorScheme colorScheme,
|
required ColorScheme colorScheme,
|
||||||
@@ -1571,9 +1622,42 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
if (isLoading)
|
if (isLoading)
|
||||||
const SliverToBoxAdapter(child: Padding(padding: EdgeInsets.symmetric(horizontal: 16), child: LinearProgressIndicator())),
|
const SliverToBoxAdapter(child: Padding(padding: EdgeInsets.symmetric(horizontal: 16), child: LinearProgressIndicator())),
|
||||||
|
|
||||||
|
// Artists from default search (Deezer/Spotify) - now in vertical list style
|
||||||
if (searchArtists != null && searchArtists.isNotEmpty)
|
if (searchArtists != null && searchArtists.isNotEmpty)
|
||||||
SliverToBoxAdapter(child: _buildArtistSearchResults(searchArtists, colorScheme)),
|
SliverToBoxAdapter(child: Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(16, 8, 16, 8),
|
||||||
|
child: Text(context.l10n.searchArtists, style: Theme.of(context).textTheme.titleSmall?.copyWith(color: colorScheme.onSurfaceVariant)),
|
||||||
|
)),
|
||||||
|
if (searchArtists != null && searchArtists.isNotEmpty)
|
||||||
|
SliverToBoxAdapter(
|
||||||
|
child: Container(
|
||||||
|
margin: const EdgeInsets.symmetric(horizontal: 16),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Theme.of(context).brightness == Brightness.dark
|
||||||
|
? Color.alphaBlend(Colors.white.withValues(alpha: 0.08), colorScheme.surface)
|
||||||
|
: colorScheme.surfaceContainerHighest,
|
||||||
|
borderRadius: BorderRadius.circular(20),
|
||||||
|
),
|
||||||
|
clipBehavior: Clip.antiAlias,
|
||||||
|
child: Material(
|
||||||
|
color: Colors.transparent,
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
for (int i = 0; i < searchArtists.length; i++)
|
||||||
|
_SearchArtistItemWidget(
|
||||||
|
key: ValueKey('search-artist-${searchArtists[i].id}'),
|
||||||
|
artist: searchArtists[i],
|
||||||
|
showDivider: i < searchArtists.length - 1,
|
||||||
|
onTap: () => _navigateToArtist(searchArtists[i].id, searchArtists[i].name, searchArtists[i].imageUrl),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
// Artists from extension search
|
||||||
if (artistItems.isNotEmpty)
|
if (artistItems.isNotEmpty)
|
||||||
SliverToBoxAdapter(child: Padding(
|
SliverToBoxAdapter(child: Padding(
|
||||||
padding: const EdgeInsets.fromLTRB(16, 8, 16, 8),
|
padding: const EdgeInsets.fromLTRB(16, 8, 16, 8),
|
||||||
@@ -1608,6 +1692,42 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
|
// Albums from default search (Deezer/Spotify)
|
||||||
|
if (searchAlbums != null && searchAlbums.isNotEmpty)
|
||||||
|
SliverToBoxAdapter(child: Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(16, 8, 16, 8),
|
||||||
|
child: Text(context.l10n.searchAlbums, style: Theme.of(context).textTheme.titleSmall?.copyWith(color: colorScheme.onSurfaceVariant)),
|
||||||
|
)),
|
||||||
|
if (searchAlbums != null && searchAlbums.isNotEmpty)
|
||||||
|
SliverToBoxAdapter(
|
||||||
|
child: Container(
|
||||||
|
margin: const EdgeInsets.symmetric(horizontal: 16),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Theme.of(context).brightness == Brightness.dark
|
||||||
|
? Color.alphaBlend(Colors.white.withValues(alpha: 0.08), colorScheme.surface)
|
||||||
|
: colorScheme.surfaceContainerHighest,
|
||||||
|
borderRadius: BorderRadius.circular(20),
|
||||||
|
),
|
||||||
|
clipBehavior: Clip.antiAlias,
|
||||||
|
child: Material(
|
||||||
|
color: Colors.transparent,
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
for (int i = 0; i < searchAlbums.length; i++)
|
||||||
|
_SearchAlbumItemWidget(
|
||||||
|
key: ValueKey('search-album-${searchAlbums[i].id}'),
|
||||||
|
album: searchAlbums[i],
|
||||||
|
showDivider: i < searchAlbums.length - 1,
|
||||||
|
onTap: () => _navigateToSearchAlbum(searchAlbums[i]),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
// Albums from extension search
|
||||||
if (albumItems.isNotEmpty)
|
if (albumItems.isNotEmpty)
|
||||||
SliverToBoxAdapter(child: Padding(
|
SliverToBoxAdapter(child: Padding(
|
||||||
padding: const EdgeInsets.fromLTRB(16, 8, 16, 8),
|
padding: const EdgeInsets.fromLTRB(16, 8, 16, 8),
|
||||||
@@ -1642,6 +1762,42 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
|
// Playlists from default search (Deezer/Spotify)
|
||||||
|
if (searchPlaylists != null && searchPlaylists.isNotEmpty)
|
||||||
|
SliverToBoxAdapter(child: Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(16, 8, 16, 8),
|
||||||
|
child: Text(context.l10n.searchPlaylists, style: Theme.of(context).textTheme.titleSmall?.copyWith(color: colorScheme.onSurfaceVariant)),
|
||||||
|
)),
|
||||||
|
if (searchPlaylists != null && searchPlaylists.isNotEmpty)
|
||||||
|
SliverToBoxAdapter(
|
||||||
|
child: Container(
|
||||||
|
margin: const EdgeInsets.symmetric(horizontal: 16),
|
||||||
|
decoration: BoxDecoration(
|
||||||
|
color: Theme.of(context).brightness == Brightness.dark
|
||||||
|
? Color.alphaBlend(Colors.white.withValues(alpha: 0.08), colorScheme.surface)
|
||||||
|
: colorScheme.surfaceContainerHighest,
|
||||||
|
borderRadius: BorderRadius.circular(20),
|
||||||
|
),
|
||||||
|
clipBehavior: Clip.antiAlias,
|
||||||
|
child: Material(
|
||||||
|
color: Colors.transparent,
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
for (int i = 0; i < searchPlaylists.length; i++)
|
||||||
|
_SearchPlaylistItemWidget(
|
||||||
|
key: ValueKey('search-playlist-${searchPlaylists[i].id}'),
|
||||||
|
playlist: searchPlaylists[i],
|
||||||
|
showDivider: i < searchPlaylists.length - 1,
|
||||||
|
onTap: () => _navigateToSearchPlaylist(searchPlaylists[i]),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
|
||||||
|
// Playlists from extension search
|
||||||
if (playlistItems.isNotEmpty)
|
if (playlistItems.isNotEmpty)
|
||||||
SliverToBoxAdapter(child: Padding(
|
SliverToBoxAdapter(child: Padding(
|
||||||
padding: const EdgeInsets.fromLTRB(16, 8, 16, 8),
|
padding: const EdgeInsets.fromLTRB(16, 8, 16, 8),
|
||||||
@@ -1716,83 +1872,6 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildArtistSearchResults(List<SearchArtist> artists, ColorScheme colorScheme) {
|
|
||||||
return Column(
|
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
|
||||||
children: [
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.fromLTRB(16, 8, 16, 8),
|
|
||||||
child: Text(context.l10n.searchArtists, style: Theme.of(context).textTheme.titleSmall?.copyWith(color: colorScheme.onSurfaceVariant)),
|
|
||||||
),
|
|
||||||
SizedBox(
|
|
||||||
height: 160,
|
|
||||||
child: ListView.builder(
|
|
||||||
scrollDirection: Axis.horizontal,
|
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 12),
|
|
||||||
itemCount: artists.length,
|
|
||||||
itemBuilder: (context, index) {
|
|
||||||
final artist = artists[index];
|
|
||||||
return KeyedSubtree(
|
|
||||||
key: ValueKey(artist.id),
|
|
||||||
child: _buildArtistCard(artist, colorScheme),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _buildArtistCard(SearchArtist artist, ColorScheme colorScheme) {
|
|
||||||
final hasValidImage = artist.imageUrl != null &&
|
|
||||||
artist.imageUrl!.isNotEmpty &&
|
|
||||||
Uri.tryParse(artist.imageUrl!)?.hasAuthority == true;
|
|
||||||
|
|
||||||
return GestureDetector(
|
|
||||||
onTap: () => _navigateToArtist(artist.id, artist.name, artist.imageUrl),
|
|
||||||
child: Container(
|
|
||||||
width: 110,
|
|
||||||
margin: const EdgeInsets.symmetric(horizontal: 6),
|
|
||||||
child: Column(
|
|
||||||
children: [
|
|
||||||
Container(
|
|
||||||
width: 100,
|
|
||||||
height: 100,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
shape: BoxShape.circle,
|
|
||||||
color: colorScheme.surfaceContainerHighest,
|
|
||||||
),
|
|
||||||
child: ClipOval(
|
|
||||||
child: hasValidImage
|
|
||||||
? CachedNetworkImage(
|
|
||||||
imageUrl: artist.imageUrl!,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
memCacheWidth: 200,
|
|
||||||
memCacheHeight: 200,
|
|
||||||
cacheManager: CoverCacheManager.instance,
|
|
||||||
errorWidget: (context, url, error) => Icon(
|
|
||||||
Icons.person,
|
|
||||||
color: colorScheme.onSurfaceVariant,
|
|
||||||
size: 44,
|
|
||||||
),
|
|
||||||
)
|
|
||||||
: Icon(Icons.person, color: colorScheme.onSurfaceVariant, size: 44),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
const SizedBox(height: 8),
|
|
||||||
Text(
|
|
||||||
artist.name,
|
|
||||||
style: Theme.of(context).textTheme.bodySmall?.copyWith(fontWeight: FontWeight.w500),
|
|
||||||
maxLines: 2,
|
|
||||||
overflow: TextOverflow.ellipsis,
|
|
||||||
textAlign: TextAlign.center,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
void _navigateToArtist(String artistId, String artistName, String? imageUrl) {
|
void _navigateToArtist(String artistId, String artistName, String? imageUrl) {
|
||||||
ref.read(settingsProvider.notifier).setHasSearchedBefore();
|
ref.read(settingsProvider.notifier).setHasSearchedBefore();
|
||||||
|
|
||||||
@@ -1805,6 +1884,60 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void _navigateToSearchAlbum(SearchAlbum album) {
|
||||||
|
ref.read(settingsProvider.notifier).setHasSearchedBefore();
|
||||||
|
|
||||||
|
// Extract the numeric ID from "deezer:123" format
|
||||||
|
String albumId = album.id;
|
||||||
|
if (albumId.startsWith('deezer:')) {
|
||||||
|
albumId = albumId.substring(7);
|
||||||
|
}
|
||||||
|
|
||||||
|
ref.read(recentAccessProvider.notifier).recordAlbumAccess(
|
||||||
|
id: album.id,
|
||||||
|
name: album.name,
|
||||||
|
artistName: album.artists,
|
||||||
|
imageUrl: album.imageUrl,
|
||||||
|
providerId: 'deezer',
|
||||||
|
);
|
||||||
|
|
||||||
|
Navigator.push(context, MaterialPageRoute(
|
||||||
|
builder: (context) => AlbumScreen(
|
||||||
|
albumId: albumId,
|
||||||
|
albumName: album.name,
|
||||||
|
coverUrl: album.imageUrl,
|
||||||
|
tracks: const [], // Will be fetched by AlbumScreen
|
||||||
|
),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
void _navigateToSearchPlaylist(SearchPlaylist playlist) {
|
||||||
|
ref.read(settingsProvider.notifier).setHasSearchedBefore();
|
||||||
|
|
||||||
|
// Extract the numeric ID from "deezer:123" format
|
||||||
|
String playlistId = playlist.id;
|
||||||
|
if (playlistId.startsWith('deezer:')) {
|
||||||
|
playlistId = playlistId.substring(7);
|
||||||
|
}
|
||||||
|
|
||||||
|
ref.read(recentAccessProvider.notifier).recordPlaylistAccess(
|
||||||
|
id: playlist.id,
|
||||||
|
name: playlist.name,
|
||||||
|
ownerName: playlist.owner,
|
||||||
|
imageUrl: playlist.imageUrl,
|
||||||
|
providerId: 'deezer',
|
||||||
|
);
|
||||||
|
|
||||||
|
Navigator.push(context, MaterialPageRoute(
|
||||||
|
builder: (context) => PlaylistScreen(
|
||||||
|
playlistName: playlist.name,
|
||||||
|
coverUrl: playlist.imageUrl,
|
||||||
|
tracks: const [], // Will be fetched
|
||||||
|
playlistId: playlistId,
|
||||||
|
),
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
void _navigateToExtensionAlbum(Track albumItem) async {
|
void _navigateToExtensionAlbum(Track albumItem) async {
|
||||||
final extensionId = albumItem.source;
|
final extensionId = albumItem.source;
|
||||||
if (extensionId == null || extensionId.isEmpty) {
|
if (extensionId == null || extensionId.isEmpty) {
|
||||||
@@ -1912,6 +2045,106 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
return 'Paste Spotify URL or search...';
|
return 'Paste Spotify URL or search...';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Widget _buildSearchFilterBar(
|
||||||
|
List<SearchFilter> filters,
|
||||||
|
String? selectedFilter,
|
||||||
|
ColorScheme colorScheme,
|
||||||
|
) {
|
||||||
|
return Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 16),
|
||||||
|
child: SingleChildScrollView(
|
||||||
|
scrollDirection: Axis.horizontal,
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
// "All" chip (no filter)
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 8),
|
||||||
|
child: FilterChip(
|
||||||
|
label: const Text('All'),
|
||||||
|
selected: selectedFilter == null,
|
||||||
|
onSelected: (_) {
|
||||||
|
ref.read(trackProvider.notifier).setSearchFilter(null);
|
||||||
|
_triggerSearchWithFilter(null);
|
||||||
|
},
|
||||||
|
showCheckmark: false,
|
||||||
|
selectedColor: colorScheme.primaryContainer,
|
||||||
|
backgroundColor: colorScheme.surfaceContainerHighest,
|
||||||
|
labelStyle: TextStyle(
|
||||||
|
color: selectedFilter == null
|
||||||
|
? colorScheme.onPrimaryContainer
|
||||||
|
: colorScheme.onSurfaceVariant,
|
||||||
|
fontWeight: selectedFilter == null ? FontWeight.w600 : FontWeight.normal,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
// Filter chips from extension
|
||||||
|
...filters.map((filter) {
|
||||||
|
final isSelected = selectedFilter == filter.id;
|
||||||
|
return Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 8),
|
||||||
|
child: FilterChip(
|
||||||
|
label: Text(filter.label ?? filter.id),
|
||||||
|
selected: isSelected,
|
||||||
|
onSelected: (_) {
|
||||||
|
ref.read(trackProvider.notifier).setSearchFilter(filter.id);
|
||||||
|
_triggerSearchWithFilter(filter.id);
|
||||||
|
},
|
||||||
|
showCheckmark: false,
|
||||||
|
selectedColor: colorScheme.primaryContainer,
|
||||||
|
backgroundColor: colorScheme.surfaceContainerHighest,
|
||||||
|
labelStyle: TextStyle(
|
||||||
|
color: isSelected
|
||||||
|
? colorScheme.onPrimaryContainer
|
||||||
|
: colorScheme.onSurfaceVariant,
|
||||||
|
fontWeight: isSelected ? FontWeight.w600 : FontWeight.normal,
|
||||||
|
),
|
||||||
|
avatar: filter.icon != null ? Icon(
|
||||||
|
_getFilterIcon(filter.icon!),
|
||||||
|
size: 18,
|
||||||
|
color: isSelected
|
||||||
|
? colorScheme.onPrimaryContainer
|
||||||
|
: colorScheme.onSurfaceVariant,
|
||||||
|
) : null,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
IconData _getFilterIcon(String iconName) {
|
||||||
|
switch (iconName.toLowerCase()) {
|
||||||
|
case 'music':
|
||||||
|
case 'track':
|
||||||
|
case 'song':
|
||||||
|
return Icons.music_note;
|
||||||
|
case 'album':
|
||||||
|
return Icons.album;
|
||||||
|
case 'artist':
|
||||||
|
return Icons.person;
|
||||||
|
case 'playlist':
|
||||||
|
return Icons.playlist_play;
|
||||||
|
case 'video':
|
||||||
|
return Icons.video_library;
|
||||||
|
case 'podcast':
|
||||||
|
return Icons.podcasts;
|
||||||
|
default:
|
||||||
|
return Icons.search;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _triggerSearchWithFilter(String? filter) {
|
||||||
|
final text = _urlController.text.trim();
|
||||||
|
if (text.isEmpty || text.length < _minLiveSearchChars) return;
|
||||||
|
if (text.startsWith('http') || text.startsWith('spotify:')) return;
|
||||||
|
|
||||||
|
// Reset last search query to force new search
|
||||||
|
_lastSearchQuery = null;
|
||||||
|
_performSearch(text, filterOverride: filter);
|
||||||
|
}
|
||||||
|
|
||||||
Widget _buildSearchBar(ColorScheme colorScheme) {
|
Widget _buildSearchBar(ColorScheme colorScheme) {
|
||||||
final hasText = _urlController.text.isNotEmpty;
|
final hasText = _urlController.text.isNotEmpty;
|
||||||
|
|
||||||
@@ -1938,6 +2171,8 @@ class _HomeTabState extends ConsumerState<HomeTab> with AutomaticKeepAliveClient
|
|||||||
prefixIcon: _SearchProviderDropdown(
|
prefixIcon: _SearchProviderDropdown(
|
||||||
onProviderChanged: () {
|
onProviderChanged: () {
|
||||||
_lastSearchQuery = null;
|
_lastSearchQuery = null;
|
||||||
|
// Reset filter when provider changes
|
||||||
|
ref.read(trackProvider.notifier).setSearchFilter(null);
|
||||||
setState(() {});
|
setState(() {});
|
||||||
final text = _urlController.text.trim();
|
final text = _urlController.text.trim();
|
||||||
if (text.isNotEmpty && text.length >= _minLiveSearchChars) {
|
if (text.isNotEmpty && text.length >= _minLiveSearchChars) {
|
||||||
@@ -2195,12 +2430,16 @@ class _TrackItemWithStatus extends ConsumerWidget {
|
|||||||
double thumbWidth = 56;
|
double thumbWidth = 56;
|
||||||
double thumbHeight = 56;
|
double thumbHeight = 56;
|
||||||
|
|
||||||
final trackState = ref.watch(trackProvider);
|
final searchExtensionId =
|
||||||
final extensionId = track.source ?? trackState.searchExtensionId;
|
ref.watch(trackProvider.select((s) => s.searchExtensionId));
|
||||||
|
final extensionId = track.source ?? searchExtensionId;
|
||||||
|
|
||||||
if (extensionId != null && extensionId.isNotEmpty) {
|
if (extensionId != null && extensionId.isNotEmpty) {
|
||||||
final extState = ref.watch(extensionProvider);
|
final extension = ref.watch(
|
||||||
final extension = extState.extensions.where((e) => e.id == extensionId).firstOrNull;
|
extensionProvider.select(
|
||||||
|
(s) => s.extensions.where((e) => e.id == extensionId).firstOrNull,
|
||||||
|
),
|
||||||
|
);
|
||||||
if (extension?.searchBehavior != null) {
|
if (extension?.searchBehavior != null) {
|
||||||
final size = extension!.searchBehavior!.getThumbnailSize(defaultSize: 56);
|
final size = extension!.searchBehavior!.getThumbnailSize(defaultSize: 56);
|
||||||
thumbWidth = size.$1;
|
thumbWidth = size.$1;
|
||||||
@@ -2477,6 +2716,294 @@ class _CollectionItemWidget extends StatelessWidget {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Widget for displaying artist items from default search (Deezer/Spotify)
|
||||||
|
class _SearchArtistItemWidget extends StatelessWidget {
|
||||||
|
final SearchArtist artist;
|
||||||
|
final bool showDivider;
|
||||||
|
final VoidCallback onTap;
|
||||||
|
|
||||||
|
const _SearchArtistItemWidget({
|
||||||
|
super.key,
|
||||||
|
required this.artist,
|
||||||
|
required this.showDivider,
|
||||||
|
required this.onTap,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final colorScheme = Theme.of(context).colorScheme;
|
||||||
|
final hasValidImage = artist.imageUrl != null &&
|
||||||
|
artist.imageUrl!.isNotEmpty &&
|
||||||
|
Uri.tryParse(artist.imageUrl!)?.hasAuthority == true;
|
||||||
|
|
||||||
|
return Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
InkWell(
|
||||||
|
onTap: onTap,
|
||||||
|
splashColor: colorScheme.primary.withValues(alpha: 0.12),
|
||||||
|
highlightColor: colorScheme.primary.withValues(alpha: 0.08),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 10),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
ClipRRect(
|
||||||
|
borderRadius: BorderRadius.circular(28),
|
||||||
|
child: hasValidImage
|
||||||
|
? CachedNetworkImage(
|
||||||
|
imageUrl: artist.imageUrl!,
|
||||||
|
width: 56,
|
||||||
|
height: 56,
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
memCacheWidth: 112,
|
||||||
|
memCacheHeight: 112,
|
||||||
|
cacheManager: CoverCacheManager.instance,
|
||||||
|
)
|
||||||
|
: Container(
|
||||||
|
width: 56,
|
||||||
|
height: 56,
|
||||||
|
color: colorScheme.surfaceContainerHighest,
|
||||||
|
child: Icon(
|
||||||
|
Icons.person,
|
||||||
|
color: colorScheme.onSurfaceVariant,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 12),
|
||||||
|
Expanded(
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
artist.name,
|
||||||
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(fontWeight: FontWeight.w500),
|
||||||
|
maxLines: 1,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 2),
|
||||||
|
Text(
|
||||||
|
'Artist',
|
||||||
|
style: Theme.of(context).textTheme.bodySmall?.copyWith(color: colorScheme.onSurfaceVariant),
|
||||||
|
maxLines: 1,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Icon(
|
||||||
|
Icons.chevron_right,
|
||||||
|
color: colorScheme.onSurfaceVariant,
|
||||||
|
size: 24,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (showDivider)
|
||||||
|
Divider(
|
||||||
|
height: 1,
|
||||||
|
thickness: 1,
|
||||||
|
indent: 80,
|
||||||
|
endIndent: 12,
|
||||||
|
color: colorScheme.outlineVariant.withValues(alpha: 0.3),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Widget for displaying album items from default search (Deezer/Spotify)
|
||||||
|
class _SearchAlbumItemWidget extends StatelessWidget {
|
||||||
|
final SearchAlbum album;
|
||||||
|
final bool showDivider;
|
||||||
|
final VoidCallback onTap;
|
||||||
|
|
||||||
|
const _SearchAlbumItemWidget({
|
||||||
|
super.key,
|
||||||
|
required this.album,
|
||||||
|
required this.showDivider,
|
||||||
|
required this.onTap,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final colorScheme = Theme.of(context).colorScheme;
|
||||||
|
final hasValidImage = album.imageUrl != null &&
|
||||||
|
album.imageUrl!.isNotEmpty &&
|
||||||
|
Uri.tryParse(album.imageUrl!)?.hasAuthority == true;
|
||||||
|
|
||||||
|
return Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
InkWell(
|
||||||
|
onTap: onTap,
|
||||||
|
splashColor: colorScheme.primary.withValues(alpha: 0.12),
|
||||||
|
highlightColor: colorScheme.primary.withValues(alpha: 0.08),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 10),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
ClipRRect(
|
||||||
|
borderRadius: BorderRadius.circular(10),
|
||||||
|
child: hasValidImage
|
||||||
|
? CachedNetworkImage(
|
||||||
|
imageUrl: album.imageUrl!,
|
||||||
|
width: 56,
|
||||||
|
height: 56,
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
memCacheWidth: 112,
|
||||||
|
memCacheHeight: 112,
|
||||||
|
cacheManager: CoverCacheManager.instance,
|
||||||
|
)
|
||||||
|
: Container(
|
||||||
|
width: 56,
|
||||||
|
height: 56,
|
||||||
|
color: colorScheme.surfaceContainerHighest,
|
||||||
|
child: Icon(
|
||||||
|
Icons.album,
|
||||||
|
color: colorScheme.onSurfaceVariant,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 12),
|
||||||
|
Expanded(
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
album.name,
|
||||||
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(fontWeight: FontWeight.w500),
|
||||||
|
maxLines: 1,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 2),
|
||||||
|
Text(
|
||||||
|
album.artists.isNotEmpty ? album.artists : 'Album',
|
||||||
|
style: Theme.of(context).textTheme.bodySmall?.copyWith(color: colorScheme.onSurfaceVariant),
|
||||||
|
maxLines: 1,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Icon(
|
||||||
|
Icons.chevron_right,
|
||||||
|
color: colorScheme.onSurfaceVariant,
|
||||||
|
size: 24,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (showDivider)
|
||||||
|
Divider(
|
||||||
|
height: 1,
|
||||||
|
thickness: 1,
|
||||||
|
indent: 80,
|
||||||
|
endIndent: 12,
|
||||||
|
color: colorScheme.outlineVariant.withValues(alpha: 0.3),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Widget for displaying playlist items from default search (Deezer/Spotify)
|
||||||
|
class _SearchPlaylistItemWidget extends StatelessWidget {
|
||||||
|
final SearchPlaylist playlist;
|
||||||
|
final bool showDivider;
|
||||||
|
final VoidCallback onTap;
|
||||||
|
|
||||||
|
const _SearchPlaylistItemWidget({
|
||||||
|
super.key,
|
||||||
|
required this.playlist,
|
||||||
|
required this.showDivider,
|
||||||
|
required this.onTap,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
final colorScheme = Theme.of(context).colorScheme;
|
||||||
|
final hasValidImage = playlist.imageUrl != null &&
|
||||||
|
playlist.imageUrl!.isNotEmpty &&
|
||||||
|
Uri.tryParse(playlist.imageUrl!)?.hasAuthority == true;
|
||||||
|
|
||||||
|
return Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
children: [
|
||||||
|
InkWell(
|
||||||
|
onTap: onTap,
|
||||||
|
splashColor: colorScheme.primary.withValues(alpha: 0.12),
|
||||||
|
highlightColor: colorScheme.primary.withValues(alpha: 0.08),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 10),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
ClipRRect(
|
||||||
|
borderRadius: BorderRadius.circular(10),
|
||||||
|
child: hasValidImage
|
||||||
|
? CachedNetworkImage(
|
||||||
|
imageUrl: playlist.imageUrl!,
|
||||||
|
width: 56,
|
||||||
|
height: 56,
|
||||||
|
fit: BoxFit.cover,
|
||||||
|
memCacheWidth: 112,
|
||||||
|
memCacheHeight: 112,
|
||||||
|
cacheManager: CoverCacheManager.instance,
|
||||||
|
)
|
||||||
|
: Container(
|
||||||
|
width: 56,
|
||||||
|
height: 56,
|
||||||
|
color: colorScheme.surfaceContainerHighest,
|
||||||
|
child: Icon(
|
||||||
|
Icons.playlist_play,
|
||||||
|
color: colorScheme.onSurfaceVariant,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 12),
|
||||||
|
Expanded(
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
playlist.name,
|
||||||
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(fontWeight: FontWeight.w500),
|
||||||
|
maxLines: 1,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 2),
|
||||||
|
Text(
|
||||||
|
playlist.owner.isNotEmpty ? playlist.owner : 'Playlist',
|
||||||
|
style: Theme.of(context).textTheme.bodySmall?.copyWith(color: colorScheme.onSurfaceVariant),
|
||||||
|
maxLines: 1,
|
||||||
|
overflow: TextOverflow.ellipsis,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Icon(
|
||||||
|
Icons.chevron_right,
|
||||||
|
color: colorScheme.onSurfaceVariant,
|
||||||
|
size: 24,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (showDivider)
|
||||||
|
Divider(
|
||||||
|
height: 1,
|
||||||
|
thickness: 1,
|
||||||
|
indent: 80,
|
||||||
|
endIndent: 12,
|
||||||
|
color: colorScheme.outlineVariant.withValues(alpha: 0.3),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class ExtensionAlbumScreen extends ConsumerStatefulWidget {
|
class ExtensionAlbumScreen extends ConsumerStatefulWidget {
|
||||||
final String extensionId;
|
final String extensionId;
|
||||||
final String albumId;
|
final String albumId;
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import 'package:flutter_riverpod/flutter_riverpod.dart';
|
|||||||
import 'package:cached_network_image/cached_network_image.dart';
|
import 'package:cached_network_image/cached_network_image.dart';
|
||||||
import 'package:spotiflac_android/services/palette_service.dart';
|
import 'package:spotiflac_android/services/palette_service.dart';
|
||||||
import 'package:spotiflac_android/services/cover_cache_manager.dart';
|
import 'package:spotiflac_android/services/cover_cache_manager.dart';
|
||||||
|
import 'package:spotiflac_android/services/platform_bridge.dart';
|
||||||
import 'package:spotiflac_android/l10n/l10n.dart';
|
import 'package:spotiflac_android/l10n/l10n.dart';
|
||||||
import 'package:spotiflac_android/models/track.dart';
|
import 'package:spotiflac_android/models/track.dart';
|
||||||
import 'package:spotiflac_android/models/download_item.dart';
|
import 'package:spotiflac_android/models/download_item.dart';
|
||||||
@@ -15,12 +16,14 @@ class PlaylistScreen extends ConsumerStatefulWidget {
|
|||||||
final String playlistName;
|
final String playlistName;
|
||||||
final String? coverUrl;
|
final String? coverUrl;
|
||||||
final List<Track> tracks;
|
final List<Track> tracks;
|
||||||
|
final String? playlistId; // Deezer playlist ID for fetching tracks
|
||||||
|
|
||||||
const PlaylistScreen({
|
const PlaylistScreen({
|
||||||
super.key,
|
super.key,
|
||||||
required this.playlistName,
|
required this.playlistName,
|
||||||
this.coverUrl,
|
this.coverUrl,
|
||||||
required this.tracks,
|
required this.tracks,
|
||||||
|
this.playlistId,
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -31,12 +34,18 @@ class _PlaylistScreenState extends ConsumerState<PlaylistScreen> {
|
|||||||
Color? _dominantColor;
|
Color? _dominantColor;
|
||||||
bool _showTitleInAppBar = false;
|
bool _showTitleInAppBar = false;
|
||||||
final ScrollController _scrollController = ScrollController();
|
final ScrollController _scrollController = ScrollController();
|
||||||
|
List<Track>? _fetchedTracks;
|
||||||
|
bool _isLoading = false;
|
||||||
|
String? _error;
|
||||||
|
|
||||||
|
List<Track> get _tracks => _fetchedTracks ?? widget.tracks;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
_scrollController.addListener(_onScroll);
|
_scrollController.addListener(_onScroll);
|
||||||
_extractDominantColor();
|
_extractDominantColor();
|
||||||
|
_fetchTracksIfNeeded();
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -46,6 +55,58 @@ class _PlaylistScreenState extends ConsumerState<PlaylistScreen> {
|
|||||||
super.dispose();
|
super.dispose();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<void> _fetchTracksIfNeeded() async {
|
||||||
|
if (widget.tracks.isNotEmpty || widget.playlistId == null) return;
|
||||||
|
|
||||||
|
setState(() {
|
||||||
|
_isLoading = true;
|
||||||
|
_error = null;
|
||||||
|
});
|
||||||
|
|
||||||
|
try {
|
||||||
|
final result = await PlatformBridge.getDeezerMetadata('playlist', widget.playlistId!);
|
||||||
|
if (!mounted) return;
|
||||||
|
|
||||||
|
final trackList = result['tracks'] as List<dynamic>? ?? [];
|
||||||
|
final tracks = trackList.map((t) => _parseTrack(t as Map<String, dynamic>)).toList();
|
||||||
|
|
||||||
|
setState(() {
|
||||||
|
_fetchedTracks = tracks;
|
||||||
|
_isLoading = false;
|
||||||
|
});
|
||||||
|
} catch (e) {
|
||||||
|
if (!mounted) return;
|
||||||
|
setState(() {
|
||||||
|
_error = e.toString();
|
||||||
|
_isLoading = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Track _parseTrack(Map<String, dynamic> data) {
|
||||||
|
int durationMs = 0;
|
||||||
|
final durationValue = data['duration_ms'];
|
||||||
|
if (durationValue is int) {
|
||||||
|
durationMs = durationValue;
|
||||||
|
} else if (durationValue is double) {
|
||||||
|
durationMs = durationValue.toInt();
|
||||||
|
}
|
||||||
|
|
||||||
|
return Track(
|
||||||
|
id: (data['spotify_id'] ?? data['id'] ?? '').toString(),
|
||||||
|
name: (data['name'] ?? '').toString(),
|
||||||
|
artistName: (data['artists'] ?? data['artist'] ?? '').toString(),
|
||||||
|
albumName: (data['album_name'] ?? data['album'] ?? '').toString(),
|
||||||
|
albumArtist: data['album_artist']?.toString(),
|
||||||
|
coverUrl: (data['cover_url'] ?? data['images'])?.toString(),
|
||||||
|
isrc: data['isrc']?.toString(),
|
||||||
|
duration: (durationMs / 1000).round(),
|
||||||
|
trackNumber: data['track_number'] as int?,
|
||||||
|
discNumber: data['disc_number'] as int?,
|
||||||
|
releaseDate: data['release_date']?.toString(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
void _onScroll() {
|
void _onScroll() {
|
||||||
final shouldShow = _scrollController.offset > 280;
|
final shouldShow = _scrollController.offset > 280;
|
||||||
if (shouldShow != _showTitleInAppBar) {
|
if (shouldShow != _showTitleInAppBar) {
|
||||||
@@ -211,15 +272,15 @@ class _PlaylistScreenState extends ConsumerState<PlaylistScreen> {
|
|||||||
children: [
|
children: [
|
||||||
Icon(Icons.playlist_play, size: 14, color: colorScheme.onTertiaryContainer),
|
Icon(Icons.playlist_play, size: 14, color: colorScheme.onTertiaryContainer),
|
||||||
const SizedBox(width: 4),
|
const SizedBox(width: 4),
|
||||||
Text(context.l10n.tracksCount(widget.tracks.length), style: TextStyle(color: colorScheme.onTertiaryContainer, fontWeight: FontWeight.w600, fontSize: 12)),
|
Text(context.l10n.tracksCount(_tracks.length), style: TextStyle(color: colorScheme.onTertiaryContainer, fontWeight: FontWeight.w600, fontSize: 12)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 16),
|
const SizedBox(height: 16),
|
||||||
FilledButton.icon(
|
FilledButton.icon(
|
||||||
onPressed: () => _downloadAll(context),
|
onPressed: _tracks.isEmpty ? null : () => _downloadAll(context),
|
||||||
icon: const Icon(Icons.download, size: 18),
|
icon: const Icon(Icons.download, size: 18),
|
||||||
label: Text(context.l10n.downloadAllCount(widget.tracks.length)),
|
label: Text(context.l10n.downloadAllCount(_tracks.length)),
|
||||||
style: FilledButton.styleFrom(
|
style: FilledButton.styleFrom(
|
||||||
minimumSize: const Size.fromHeight(48),
|
minimumSize: const Size.fromHeight(48),
|
||||||
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(24)),
|
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(24)),
|
||||||
@@ -249,10 +310,54 @@ const SizedBox(height: 16),
|
|||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildTrackList(BuildContext context, ColorScheme colorScheme) {
|
Widget _buildTrackList(BuildContext context, ColorScheme colorScheme) {
|
||||||
|
if (_isLoading) {
|
||||||
|
return const SliverToBoxAdapter(
|
||||||
|
child: Padding(
|
||||||
|
padding: EdgeInsets.all(32),
|
||||||
|
child: Center(child: CircularProgressIndicator()),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_error != null) {
|
||||||
|
return SliverToBoxAdapter(
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16),
|
||||||
|
child: Card(
|
||||||
|
color: colorScheme.errorContainer,
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16),
|
||||||
|
child: Row(
|
||||||
|
children: [
|
||||||
|
Icon(Icons.error_outline, color: colorScheme.error),
|
||||||
|
const SizedBox(width: 12),
|
||||||
|
Expanded(child: Text(_error!, style: TextStyle(color: colorScheme.error))),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_tracks.isEmpty) {
|
||||||
|
return SliverToBoxAdapter(
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(32),
|
||||||
|
child: Center(
|
||||||
|
child: Text(
|
||||||
|
context.l10n.errorNoTracksFound,
|
||||||
|
style: TextStyle(color: colorScheme.onSurfaceVariant),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
return SliverList(
|
return SliverList(
|
||||||
delegate: SliverChildBuilderDelegate(
|
delegate: SliverChildBuilderDelegate(
|
||||||
(context, index) {
|
(context, index) {
|
||||||
final track = widget.tracks[index];
|
final track = _tracks[index];
|
||||||
return KeyedSubtree(
|
return KeyedSubtree(
|
||||||
key: ValueKey(track.id),
|
key: ValueKey(track.id),
|
||||||
child: _PlaylistTrackItem(
|
child: _PlaylistTrackItem(
|
||||||
@@ -261,7 +366,7 @@ const SizedBox(height: 16),
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
childCount: widget.tracks.length,
|
childCount: _tracks.length,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@@ -286,21 +391,21 @@ const SizedBox(height: 16),
|
|||||||
}
|
}
|
||||||
|
|
||||||
void _downloadAll(BuildContext context) {
|
void _downloadAll(BuildContext context) {
|
||||||
if (widget.tracks.isEmpty) return;
|
if (_tracks.isEmpty) return;
|
||||||
final settings = ref.read(settingsProvider);
|
final settings = ref.read(settingsProvider);
|
||||||
if (settings.askQualityBeforeDownload) {
|
if (settings.askQualityBeforeDownload) {
|
||||||
DownloadServicePicker.show(
|
DownloadServicePicker.show(
|
||||||
context,
|
context,
|
||||||
trackName: '${widget.tracks.length} tracks',
|
trackName: '${_tracks.length} tracks',
|
||||||
artistName: widget.playlistName,
|
artistName: widget.playlistName,
|
||||||
onSelect: (quality, service) {
|
onSelect: (quality, service) {
|
||||||
ref.read(downloadQueueProvider.notifier).addMultipleToQueue(widget.tracks, service, qualityOverride: quality);
|
ref.read(downloadQueueProvider.notifier).addMultipleToQueue(_tracks, service, qualityOverride: quality);
|
||||||
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text(context.l10n.snackbarAddedTracksToQueue(widget.tracks.length))));
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text(context.l10n.snackbarAddedTracksToQueue(_tracks.length))));
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
ref.read(downloadQueueProvider.notifier).addMultipleToQueue(widget.tracks, settings.defaultService);
|
ref.read(downloadQueueProvider.notifier).addMultipleToQueue(_tracks, settings.defaultService);
|
||||||
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text(context.l10n.snackbarAddedTracksToQueue(widget.tracks.length))));
|
ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text(context.l10n.snackbarAddedTracksToQueue(_tracks.length))));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -366,6 +366,21 @@ final albumKey =
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get short badge text for quality display
|
||||||
|
String _getQualityBadgeText(String quality) {
|
||||||
|
// For lossless: "24-bit/96kHz" -> "24-bit"
|
||||||
|
if (quality.contains('bit')) {
|
||||||
|
return quality.split('/').first;
|
||||||
|
}
|
||||||
|
// For lossy: "OPUS 128kbps" -> "128k", "MP3 320kbps" -> "320k"
|
||||||
|
final bitrateMatch = RegExp(r'(\d+)kbps').firstMatch(quality);
|
||||||
|
if (bitrateMatch != null) {
|
||||||
|
return '${bitrateMatch.group(1)}k';
|
||||||
|
}
|
||||||
|
// Fallback: return format name
|
||||||
|
return quality.split(' ').first;
|
||||||
|
}
|
||||||
|
|
||||||
Future<void> _deleteSelected() async {
|
Future<void> _deleteSelected() async {
|
||||||
final count = _selectedIds.length;
|
final count = _selectedIds.length;
|
||||||
final confirmed = await showDialog<bool>(
|
final confirmed = await showDialog<bool>(
|
||||||
@@ -793,6 +808,8 @@ final queueItems = ref.watch(downloadQueueProvider.select((s) => s.items));
|
|||||||
),
|
),
|
||||||
const Spacer(),
|
const Spacer(),
|
||||||
_buildPauseResumeButton(context, ref, colorScheme),
|
_buildPauseResumeButton(context, ref, colorScheme),
|
||||||
|
const SizedBox(width: 4),
|
||||||
|
_buildClearAllButton(context, ref, colorScheme),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -1177,6 +1194,53 @@ if (queueItems.isEmpty &&
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Widget _buildClearAllButton(
|
||||||
|
BuildContext context,
|
||||||
|
WidgetRef ref,
|
||||||
|
ColorScheme colorScheme,
|
||||||
|
) {
|
||||||
|
return TextButton.icon(
|
||||||
|
onPressed: () => _showClearAllDialog(context, ref, colorScheme),
|
||||||
|
icon: const Icon(Icons.clear_all, size: 18),
|
||||||
|
label: Text(context.l10n.queueClearAll),
|
||||||
|
style: TextButton.styleFrom(
|
||||||
|
visualDensity: VisualDensity.compact,
|
||||||
|
foregroundColor: colorScheme.error,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _showClearAllDialog(
|
||||||
|
BuildContext context,
|
||||||
|
WidgetRef ref,
|
||||||
|
ColorScheme colorScheme,
|
||||||
|
) async {
|
||||||
|
final confirmed = await showDialog<bool>(
|
||||||
|
context: context,
|
||||||
|
builder: (ctx) => AlertDialog(
|
||||||
|
title: Text(context.l10n.queueClearAll),
|
||||||
|
content: Text(context.l10n.queueClearAllMessage),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(ctx, false),
|
||||||
|
child: Text(context.l10n.dialogCancel),
|
||||||
|
),
|
||||||
|
FilledButton(
|
||||||
|
onPressed: () => Navigator.pop(ctx, true),
|
||||||
|
style: FilledButton.styleFrom(
|
||||||
|
backgroundColor: colorScheme.error,
|
||||||
|
),
|
||||||
|
child: Text(context.l10n.dialogClear),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (confirmed == true && context.mounted) {
|
||||||
|
ref.read(downloadQueueProvider.notifier).clearAll();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Widget _buildEmptyState(
|
Widget _buildEmptyState(
|
||||||
BuildContext context,
|
BuildContext context,
|
||||||
ColorScheme colorScheme,
|
ColorScheme colorScheme,
|
||||||
@@ -1723,7 +1787,7 @@ child: CachedNetworkImage(
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
if (item.quality != null && item.quality!.contains('bit'))
|
if (item.quality != null && item.quality!.isNotEmpty)
|
||||||
Positioned(
|
Positioned(
|
||||||
left: 4,
|
left: 4,
|
||||||
top: 4,
|
top: 4,
|
||||||
@@ -1739,7 +1803,7 @@ child: CachedNetworkImage(
|
|||||||
borderRadius: BorderRadius.circular(4),
|
borderRadius: BorderRadius.circular(4),
|
||||||
),
|
),
|
||||||
child: Text(
|
child: Text(
|
||||||
item.quality!.split('/').first,
|
_getQualityBadgeText(item.quality!),
|
||||||
style: Theme.of(context).textTheme.labelSmall
|
style: Theme.of(context).textTheme.labelSmall
|
||||||
?.copyWith(
|
?.copyWith(
|
||||||
color: item.quality!.startsWith('24')
|
color: item.quality!.startsWith('24')
|
||||||
@@ -1974,7 +2038,7 @@ child: CachedNetworkImage(
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
if (item.quality != null &&
|
if (item.quality != null &&
|
||||||
item.quality!.contains('bit')) ...[
|
item.quality!.isNotEmpty) ...[
|
||||||
const SizedBox(width: 8),
|
const SizedBox(width: 8),
|
||||||
Container(
|
Container(
|
||||||
padding: const EdgeInsets.symmetric(
|
padding: const EdgeInsets.symmetric(
|
||||||
|
|||||||
@@ -112,11 +112,10 @@ class AboutPage extends StatelessWidget {
|
|||||||
githubUsername: 'sachinsenal0x64',
|
githubUsername: 'sachinsenal0x64',
|
||||||
showDivider: true,
|
showDivider: true,
|
||||||
),
|
),
|
||||||
_AboutSettingsItem(
|
_ContributorItem(
|
||||||
icon: Icons.cloud_outlined,
|
name: 'sjdonado',
|
||||||
title: context.l10n.aboutDoubleDouble,
|
description: context.l10n.aboutSjdonadoDesc,
|
||||||
subtitle: context.l10n.aboutDoubleDoubleDesc,
|
githubUsername: 'sjdonado',
|
||||||
onTap: () => _launchUrl('https://doubledouble.top'),
|
|
||||||
showDivider: true,
|
showDivider: true,
|
||||||
),
|
),
|
||||||
_AboutSettingsItem(
|
_AboutSettingsItem(
|
||||||
@@ -185,7 +184,7 @@ _AboutSettingsItem(
|
|||||||
icon: Icons.forum_outlined,
|
icon: Icons.forum_outlined,
|
||||||
title: context.l10n.aboutTelegramChat,
|
title: context.l10n.aboutTelegramChat,
|
||||||
subtitle: context.l10n.aboutTelegramChatSubtitle,
|
subtitle: context.l10n.aboutTelegramChatSubtitle,
|
||||||
onTap: () => _launchUrl('https://t.me/spotiflacchat'),
|
onTap: () => _launchUrl('https://t.me/spotiflac_chat'),
|
||||||
showDivider: false,
|
showDivider: false,
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -467,11 +466,29 @@ class _TranslatorsSection extends StatelessWidget {
|
|||||||
flag: '🇷🇺',
|
flag: '🇷🇺',
|
||||||
),
|
),
|
||||||
_Translator(
|
_Translator(
|
||||||
name: 'Max',
|
name: 'Amonoman',
|
||||||
crowdinUsername: 'amonoman',
|
crowdinUsername: 'amonoman',
|
||||||
language: 'German',
|
language: 'German',
|
||||||
flag: '🇩🇪',
|
flag: '🇩🇪',
|
||||||
),
|
),
|
||||||
|
_Translator(
|
||||||
|
name: 'Re*Index.(ot_inc)',
|
||||||
|
crowdinUsername: 'ot_inc',
|
||||||
|
language: 'Japanese',
|
||||||
|
flag: '🇯🇵',
|
||||||
|
),
|
||||||
|
_Translator(
|
||||||
|
name: 'Kaan',
|
||||||
|
crowdinUsername: 'glai',
|
||||||
|
language: 'Turkish',
|
||||||
|
flag: '🇹🇷',
|
||||||
|
),
|
||||||
|
_Translator(
|
||||||
|
name: 'BedirhanGltkn',
|
||||||
|
crowdinUsername: 'bedirhangltkn',
|
||||||
|
language: 'Turkish',
|
||||||
|
flag: '🇹🇷',
|
||||||
|
),
|
||||||
];
|
];
|
||||||
|
|
||||||
@override
|
@override
|
||||||
|
|||||||
@@ -3,18 +3,92 @@ import 'package:flutter/material.dart';
|
|||||||
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
import 'package:flutter_riverpod/flutter_riverpod.dart';
|
||||||
import 'package:file_picker/file_picker.dart';
|
import 'package:file_picker/file_picker.dart';
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
|
import 'package:permission_handler/permission_handler.dart';
|
||||||
|
import 'package:device_info_plus/device_info_plus.dart';
|
||||||
import 'package:spotiflac_android/l10n/l10n.dart';
|
import 'package:spotiflac_android/l10n/l10n.dart';
|
||||||
import 'package:spotiflac_android/providers/settings_provider.dart';
|
import 'package:spotiflac_android/providers/settings_provider.dart';
|
||||||
import 'package:spotiflac_android/providers/extension_provider.dart';
|
import 'package:spotiflac_android/providers/extension_provider.dart';
|
||||||
import 'package:spotiflac_android/widgets/settings_group.dart';
|
import 'package:spotiflac_android/widgets/settings_group.dart';
|
||||||
|
|
||||||
class DownloadSettingsPage extends ConsumerWidget {
|
class DownloadSettingsPage extends ConsumerStatefulWidget {
|
||||||
const DownloadSettingsPage({super.key});
|
const DownloadSettingsPage({super.key});
|
||||||
|
|
||||||
static const _builtInServices = ['tidal', 'qobuz', 'amazon'];
|
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context, WidgetRef ref) {
|
ConsumerState<DownloadSettingsPage> createState() => _DownloadSettingsPageState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _DownloadSettingsPageState extends ConsumerState<DownloadSettingsPage> {
|
||||||
|
static const _builtInServices = ['tidal', 'qobuz', 'amazon'];
|
||||||
|
int _androidSdkVersion = 0;
|
||||||
|
bool _hasAllFilesAccess = false;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_initDeviceInfo();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _initDeviceInfo() async {
|
||||||
|
if (Platform.isAndroid) {
|
||||||
|
final deviceInfo = DeviceInfoPlugin();
|
||||||
|
final androidInfo = await deviceInfo.androidInfo;
|
||||||
|
final sdkVersion = androidInfo.version.sdkInt;
|
||||||
|
final hasAccess = await Permission.manageExternalStorage.isGranted;
|
||||||
|
if (mounted) {
|
||||||
|
setState(() {
|
||||||
|
_androidSdkVersion = sdkVersion;
|
||||||
|
_hasAllFilesAccess = hasAccess;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _requestAllFilesAccess() async {
|
||||||
|
final status = await Permission.manageExternalStorage.request();
|
||||||
|
if (status.isGranted) {
|
||||||
|
ref.read(settingsProvider.notifier).setUseAllFilesAccess(true);
|
||||||
|
if (mounted) {
|
||||||
|
setState(() => _hasAllFilesAccess = true);
|
||||||
|
}
|
||||||
|
} else if (status.isPermanentlyDenied) {
|
||||||
|
if (mounted) {
|
||||||
|
final shouldOpen = await showDialog<bool>(
|
||||||
|
context: context,
|
||||||
|
builder: (context) => AlertDialog(
|
||||||
|
title: Text(context.l10n.setupStorageAccessRequired),
|
||||||
|
content: Text(context.l10n.allFilesAccessDeniedMessage),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.pop(context, false),
|
||||||
|
child: Text(context.l10n.dialogCancel),
|
||||||
|
),
|
||||||
|
FilledButton(
|
||||||
|
onPressed: () => Navigator.pop(context, true),
|
||||||
|
child: Text(context.l10n.setupOpenSettings),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
if (shouldOpen == true) {
|
||||||
|
await openAppSettings();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _disableAllFilesAccess() async {
|
||||||
|
ref.read(settingsProvider.notifier).setUseAllFilesAccess(false);
|
||||||
|
// Note: We can't revoke the permission programmatically,
|
||||||
|
// but we can stop using it in the app
|
||||||
|
if (mounted) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(content: Text(context.l10n.allFilesAccessDisabledMessage)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
final settings = ref.watch(settingsProvider);
|
final settings = ref.watch(settingsProvider);
|
||||||
final colorScheme = Theme.of(context).colorScheme;
|
final colorScheme = Theme.of(context).colorScheme;
|
||||||
final topPadding = MediaQuery.of(context).padding.top;
|
final topPadding = MediaQuery.of(context).padding.top;
|
||||||
@@ -101,15 +175,22 @@ class DownloadSettingsPage extends ConsumerWidget {
|
|||||||
),
|
),
|
||||||
SettingsSwitchItem(
|
SettingsSwitchItem(
|
||||||
icon: Icons.audiotrack,
|
icon: Icons.audiotrack,
|
||||||
title: context.l10n.enableMp3Option,
|
title: context.l10n.enableLossyOption,
|
||||||
subtitle: settings.enableMp3Option
|
subtitle: settings.enableLossyOption
|
||||||
? context.l10n.enableMp3OptionSubtitleOn
|
? context.l10n.enableLossyOptionSubtitleOn
|
||||||
: context.l10n.enableMp3OptionSubtitleOff,
|
: context.l10n.enableLossyOptionSubtitleOff,
|
||||||
value: settings.enableMp3Option,
|
value: settings.enableLossyOption,
|
||||||
onChanged: (value) => ref
|
onChanged: (value) => ref
|
||||||
.read(settingsProvider.notifier)
|
.read(settingsProvider.notifier)
|
||||||
.setEnableMp3Option(value),
|
.setEnableLossyOption(value),
|
||||||
),
|
),
|
||||||
|
if (settings.enableLossyOption)
|
||||||
|
SettingsItem(
|
||||||
|
icon: Icons.tune,
|
||||||
|
title: context.l10n.lossyFormat,
|
||||||
|
subtitle: _getLossyBitrateLabel(settings.lossyBitrate),
|
||||||
|
onTap: () => _showLossyBitratePicker(context, ref, settings.lossyBitrate),
|
||||||
|
),
|
||||||
if (!settings.askQualityBeforeDownload && isBuiltInService) ...[
|
if (!settings.askQualityBeforeDownload && isBuiltInService) ...[
|
||||||
_QualityOption(
|
_QualityOption(
|
||||||
title: context.l10n.qualityFlacLossless,
|
title: context.l10n.qualityFlacLossless,
|
||||||
@@ -134,16 +215,18 @@ class DownloadSettingsPage extends ConsumerWidget {
|
|||||||
onTap: () => ref
|
onTap: () => ref
|
||||||
.read(settingsProvider.notifier)
|
.read(settingsProvider.notifier)
|
||||||
.setAudioQuality('HI_RES_LOSSLESS'),
|
.setAudioQuality('HI_RES_LOSSLESS'),
|
||||||
showDivider: settings.enableMp3Option,
|
showDivider: settings.enableLossyOption,
|
||||||
),
|
),
|
||||||
if (settings.enableMp3Option)
|
if (settings.enableLossyOption)
|
||||||
_QualityOption(
|
_QualityOption(
|
||||||
title: context.l10n.qualityMp3,
|
title: context.l10n.qualityLossy,
|
||||||
subtitle: context.l10n.qualityMp3Subtitle,
|
subtitle: settings.lossyFormat == 'opus'
|
||||||
isSelected: settings.audioQuality == 'MP3',
|
? context.l10n.qualityLossyOpusSubtitle
|
||||||
|
: context.l10n.qualityLossyMp3Subtitle,
|
||||||
|
isSelected: settings.audioQuality == 'LOSSY',
|
||||||
onTap: () => ref
|
onTap: () => ref
|
||||||
.read(settingsProvider.notifier)
|
.read(settingsProvider.notifier)
|
||||||
.setAudioQuality('MP3'),
|
.setAudioQuality('LOSSY'),
|
||||||
showDivider: false,
|
showDivider: false,
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -261,6 +344,59 @@ class DownloadSettingsPage extends ConsumerWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
|
// All Files Access section (Android 13+ only)
|
||||||
|
if (Platform.isAndroid && _androidSdkVersion >= 33) ...[
|
||||||
|
SliverToBoxAdapter(
|
||||||
|
child: SettingsSectionHeader(title: context.l10n.sectionStorageAccess),
|
||||||
|
),
|
||||||
|
SliverToBoxAdapter(
|
||||||
|
child: SettingsGroup(
|
||||||
|
children: [
|
||||||
|
SettingsSwitchItem(
|
||||||
|
icon: Icons.folder_special_outlined,
|
||||||
|
title: context.l10n.allFilesAccess,
|
||||||
|
subtitle: _hasAllFilesAccess
|
||||||
|
? context.l10n.allFilesAccessEnabledSubtitle
|
||||||
|
: context.l10n.allFilesAccessDisabledSubtitle,
|
||||||
|
value: _hasAllFilesAccess && settings.useAllFilesAccess,
|
||||||
|
onChanged: (value) {
|
||||||
|
if (value) {
|
||||||
|
_requestAllFilesAccess();
|
||||||
|
} else {
|
||||||
|
_disableAllFilesAccess();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
showDivider: false,
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
SliverToBoxAdapter(
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(16, 8, 16, 0),
|
||||||
|
child: Row(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Icon(
|
||||||
|
Icons.info_outline,
|
||||||
|
size: 16,
|
||||||
|
color: colorScheme.onSurfaceVariant,
|
||||||
|
),
|
||||||
|
const SizedBox(width: 8),
|
||||||
|
Expanded(
|
||||||
|
child: Text(
|
||||||
|
context.l10n.allFilesAccessDescription,
|
||||||
|
style: Theme.of(context).textTheme.bodySmall?.copyWith(
|
||||||
|
color: colorScheme.onSurfaceVariant,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
|
||||||
const SliverToBoxAdapter(child: SizedBox(height: 32)),
|
const SliverToBoxAdapter(child: SizedBox(height: 32)),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
@@ -722,6 +858,165 @@ class DownloadSettingsPage extends ConsumerWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
String _getLossyBitrateLabel(String bitrate) {
|
||||||
|
switch (bitrate) {
|
||||||
|
case 'mp3_320':
|
||||||
|
return 'MP3 320kbps (Best)';
|
||||||
|
case 'mp3_256':
|
||||||
|
return 'MP3 256kbps';
|
||||||
|
case 'mp3_192':
|
||||||
|
return 'MP3 192kbps';
|
||||||
|
case 'mp3_128':
|
||||||
|
return 'MP3 128kbps';
|
||||||
|
case 'opus_128':
|
||||||
|
return 'Opus 128kbps (Best)';
|
||||||
|
case 'opus_96':
|
||||||
|
return 'Opus 96kbps';
|
||||||
|
case 'opus_64':
|
||||||
|
return 'Opus 64kbps';
|
||||||
|
default:
|
||||||
|
return 'MP3 320kbps';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _showLossyBitratePicker(
|
||||||
|
BuildContext context,
|
||||||
|
WidgetRef ref,
|
||||||
|
String current,
|
||||||
|
) {
|
||||||
|
final colorScheme = Theme.of(context).colorScheme;
|
||||||
|
showModalBottomSheet(
|
||||||
|
context: context,
|
||||||
|
backgroundColor: colorScheme.surfaceContainerHigh,
|
||||||
|
isScrollControlled: true,
|
||||||
|
shape: const RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.vertical(top: Radius.circular(28)),
|
||||||
|
),
|
||||||
|
builder: (context) => SafeArea(
|
||||||
|
child: SingleChildScrollView(
|
||||||
|
child: Column(
|
||||||
|
mainAxisSize: MainAxisSize.min,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(24, 24, 24, 8),
|
||||||
|
child: Text(
|
||||||
|
context.l10n.lossyFormat,
|
||||||
|
style: Theme.of(
|
||||||
|
context,
|
||||||
|
).textTheme.titleLarge?.copyWith(fontWeight: FontWeight.bold),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(24, 0, 24, 16),
|
||||||
|
child: Text(
|
||||||
|
context.l10n.lossyFormatDescription,
|
||||||
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
||||||
|
color: colorScheme.onSurfaceVariant,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
// MP3 Section
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(24, 8, 24, 4),
|
||||||
|
child: Text(
|
||||||
|
'MP3',
|
||||||
|
style: Theme.of(context).textTheme.titleSmall?.copyWith(
|
||||||
|
color: colorScheme.primary,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.audiotrack),
|
||||||
|
title: const Text('320kbps'),
|
||||||
|
subtitle: const Text('Best quality, larger files'),
|
||||||
|
trailing: current == 'mp3_320' ? Icon(Icons.check, color: colorScheme.primary) : null,
|
||||||
|
onTap: () {
|
||||||
|
ref.read(settingsProvider.notifier).setLossyBitrate('mp3_320');
|
||||||
|
Navigator.pop(context);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.audiotrack),
|
||||||
|
title: const Text('256kbps'),
|
||||||
|
subtitle: const Text('High quality'),
|
||||||
|
trailing: current == 'mp3_256' ? Icon(Icons.check, color: colorScheme.primary) : null,
|
||||||
|
onTap: () {
|
||||||
|
ref.read(settingsProvider.notifier).setLossyBitrate('mp3_256');
|
||||||
|
Navigator.pop(context);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.audiotrack),
|
||||||
|
title: const Text('192kbps'),
|
||||||
|
subtitle: const Text('Good quality'),
|
||||||
|
trailing: current == 'mp3_192' ? Icon(Icons.check, color: colorScheme.primary) : null,
|
||||||
|
onTap: () {
|
||||||
|
ref.read(settingsProvider.notifier).setLossyBitrate('mp3_192');
|
||||||
|
Navigator.pop(context);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.audiotrack),
|
||||||
|
title: const Text('128kbps'),
|
||||||
|
subtitle: const Text('Smaller files'),
|
||||||
|
trailing: current == 'mp3_128' ? Icon(Icons.check, color: colorScheme.primary) : null,
|
||||||
|
onTap: () {
|
||||||
|
ref.read(settingsProvider.notifier).setLossyBitrate('mp3_128');
|
||||||
|
Navigator.pop(context);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
const Divider(indent: 24, endIndent: 24),
|
||||||
|
// Opus Section
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.fromLTRB(24, 8, 24, 4),
|
||||||
|
child: Text(
|
||||||
|
'Opus',
|
||||||
|
style: Theme.of(context).textTheme.titleSmall?.copyWith(
|
||||||
|
color: colorScheme.primary,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.graphic_eq),
|
||||||
|
title: const Text('128kbps'),
|
||||||
|
subtitle: const Text('Best quality, efficient codec'),
|
||||||
|
trailing: current == 'opus_128' ? Icon(Icons.check, color: colorScheme.primary) : null,
|
||||||
|
onTap: () {
|
||||||
|
ref.read(settingsProvider.notifier).setLossyBitrate('opus_128');
|
||||||
|
Navigator.pop(context);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.graphic_eq),
|
||||||
|
title: const Text('96kbps'),
|
||||||
|
subtitle: const Text('Good quality'),
|
||||||
|
trailing: current == 'opus_96' ? Icon(Icons.check, color: colorScheme.primary) : null,
|
||||||
|
onTap: () {
|
||||||
|
ref.read(settingsProvider.notifier).setLossyBitrate('opus_96');
|
||||||
|
Navigator.pop(context);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.graphic_eq),
|
||||||
|
title: const Text('64kbps'),
|
||||||
|
subtitle: const Text('Smallest files'),
|
||||||
|
trailing: current == 'opus_64' ? Icon(Icons.check, color: colorScheme.primary) : null,
|
||||||
|
onTap: () {
|
||||||
|
ref.read(settingsProvider.notifier).setLossyBitrate('opus_64');
|
||||||
|
Navigator.pop(context);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
void _showFolderOrganizationPicker(
|
void _showFolderOrganizationPicker(
|
||||||
BuildContext context,
|
BuildContext context,
|
||||||
WidgetRef ref,
|
WidgetRef ref,
|
||||||
|
|||||||
@@ -153,14 +153,6 @@ class OptionsSettingsPage extends ConsumerWidget {
|
|||||||
.read(settingsProvider.notifier)
|
.read(settingsProvider.notifier)
|
||||||
.setUseExtensionProviders(v),
|
.setUseExtensionProviders(v),
|
||||||
),
|
),
|
||||||
SettingsSwitchItem(
|
|
||||||
icon: Icons.lyrics,
|
|
||||||
title: context.l10n.optionsEmbedLyrics,
|
|
||||||
subtitle: context.l10n.optionsEmbedLyricsSubtitle,
|
|
||||||
value: settings.embedLyrics,
|
|
||||||
onChanged: (v) =>
|
|
||||||
ref.read(settingsProvider.notifier).setEmbedLyrics(v),
|
|
||||||
),
|
|
||||||
SettingsSwitchItem(
|
SettingsSwitchItem(
|
||||||
icon: Icons.image,
|
icon: Icons.image,
|
||||||
title: context.l10n.optionsMaxQualityCover,
|
title: context.l10n.optionsMaxQualityCover,
|
||||||
|
|||||||
@@ -67,10 +67,11 @@ class _SetupScreenState extends ConsumerState<SetupScreen> {
|
|||||||
bool storageGranted = false;
|
bool storageGranted = false;
|
||||||
|
|
||||||
if (_androidSdkVersion >= 33) {
|
if (_androidSdkVersion >= 33) {
|
||||||
final manageStatus = await Permission.manageExternalStorage.status;
|
// Android 13+: Only require READ_MEDIA_AUDIO by default
|
||||||
|
// MANAGE_EXTERNAL_STORAGE is optional and can be enabled in settings
|
||||||
final audioStatus = await Permission.audio.status;
|
final audioStatus = await Permission.audio.status;
|
||||||
debugPrint('[Permission] Android 13+ check: MANAGE_EXTERNAL_STORAGE=$manageStatus, READ_MEDIA_AUDIO=$audioStatus');
|
debugPrint('[Permission] Android 13+ check: READ_MEDIA_AUDIO=$audioStatus');
|
||||||
storageGranted = manageStatus.isGranted && audioStatus.isGranted;
|
storageGranted = audioStatus.isGranted;
|
||||||
} else if (_androidSdkVersion >= 30) {
|
} else if (_androidSdkVersion >= 30) {
|
||||||
final manageStatus = await Permission.manageExternalStorage.status;
|
final manageStatus = await Permission.manageExternalStorage.status;
|
||||||
debugPrint('[Permission] Android 11-12 check: MANAGE_EXTERNAL_STORAGE=$manageStatus');
|
debugPrint('[Permission] Android 11-12 check: MANAGE_EXTERNAL_STORAGE=$manageStatus');
|
||||||
@@ -108,44 +109,20 @@ class _SetupScreenState extends ConsumerState<SetupScreen> {
|
|||||||
bool allGranted = false;
|
bool allGranted = false;
|
||||||
|
|
||||||
if (_androidSdkVersion >= 33) {
|
if (_androidSdkVersion >= 33) {
|
||||||
var manageStatus = await Permission.manageExternalStorage.status;
|
// Android 13+: Only request READ_MEDIA_AUDIO by default
|
||||||
if (!manageStatus.isGranted) {
|
// MANAGE_EXTERNAL_STORAGE is optional (can be enabled in Settings)
|
||||||
if (mounted) {
|
|
||||||
final shouldOpen = await showDialog<bool>(
|
|
||||||
context: context,
|
|
||||||
builder: (context) => AlertDialog(
|
|
||||||
title: Text(context.l10n.setupStorageAccessRequired),
|
|
||||||
content: Text(
|
|
||||||
'${context.l10n.setupStorageAccessMessage}\n\n'
|
|
||||||
'${context.l10n.setupAllowAccessToManageFiles}',
|
|
||||||
),
|
|
||||||
actions: [
|
|
||||||
TextButton(
|
|
||||||
onPressed: () => Navigator.pop(context, false),
|
|
||||||
child: Text(context.l10n.dialogCancel),
|
|
||||||
),
|
|
||||||
FilledButton(
|
|
||||||
onPressed: () => Navigator.pop(context, true),
|
|
||||||
child: Text(context.l10n.setupOpenSettings),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
if (shouldOpen == true) {
|
|
||||||
await Permission.manageExternalStorage.request();
|
|
||||||
await Future.delayed(const Duration(milliseconds: 500));
|
|
||||||
manageStatus = await Permission.manageExternalStorage.status;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var audioStatus = await Permission.audio.status;
|
var audioStatus = await Permission.audio.status;
|
||||||
if (!audioStatus.isGranted && manageStatus.isGranted) {
|
if (!audioStatus.isGranted) {
|
||||||
audioStatus = await Permission.audio.request();
|
audioStatus = await Permission.audio.request();
|
||||||
}
|
}
|
||||||
|
|
||||||
allGranted = manageStatus.isGranted && audioStatus.isGranted;
|
allGranted = audioStatus.isGranted;
|
||||||
|
|
||||||
|
if (audioStatus.isPermanentlyDenied) {
|
||||||
|
_showPermissionDeniedDialog('Audio');
|
||||||
|
setState(() => _isLoading = false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
} else if (_androidSdkVersion >= 30) {
|
} else if (_androidSdkVersion >= 30) {
|
||||||
var manageStatus = await Permission.manageExternalStorage.status;
|
var manageStatus = await Permission.manageExternalStorage.status;
|
||||||
|
|||||||
@@ -517,16 +517,27 @@ class _TrackMetadataScreenState extends ConsumerState<TrackMetadataScreen> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildMetadataGrid(BuildContext context, ColorScheme colorScheme) {
|
Widget _buildMetadataGrid(BuildContext context, ColorScheme colorScheme) {
|
||||||
// Determine audio quality string based on file type
|
// Determine audio quality string - prefer stored quality from download
|
||||||
String? audioQualityStr;
|
String? audioQualityStr;
|
||||||
final fileName = item.filePath.split('/').last;
|
final fileName = item.filePath.split('/').last;
|
||||||
final fileExt = fileName.contains('.') ? fileName.split('.').last.toUpperCase() : '';
|
final fileExt = fileName.contains('.') ? fileName.split('.').last.toUpperCase() : '';
|
||||||
|
|
||||||
if (fileExt == 'MP3') {
|
// Use stored quality from download history if available
|
||||||
audioQualityStr = '320kbps';
|
if (item.quality != null && item.quality!.isNotEmpty) {
|
||||||
|
audioQualityStr = item.quality;
|
||||||
} else if (bitDepth != null && sampleRate != null) {
|
} else if (bitDepth != null && sampleRate != null) {
|
||||||
|
// Fallback for FLAC files without stored quality
|
||||||
final sampleRateKHz = (sampleRate! / 1000).toStringAsFixed(1);
|
final sampleRateKHz = (sampleRate! / 1000).toStringAsFixed(1);
|
||||||
audioQualityStr = '$bitDepth-bit/${sampleRateKHz}kHz';
|
audioQualityStr = '$bitDepth-bit/${sampleRateKHz}kHz';
|
||||||
|
} else {
|
||||||
|
// Fallback based on file extension for legacy items
|
||||||
|
if (fileExt == 'MP3') {
|
||||||
|
audioQualityStr = 'MP3';
|
||||||
|
} else if (fileExt == 'OPUS' || fileExt == 'OGG') {
|
||||||
|
audioQualityStr = 'Opus';
|
||||||
|
} else if (fileExt == 'M4A' || fileExt == 'AAC') {
|
||||||
|
audioQualityStr = 'AAC';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
final items = <_MetadataItem>[
|
final items = <_MetadataItem>[
|
||||||
|
|||||||
@@ -1,23 +1,27 @@
|
|||||||
|
import 'dart:convert';
|
||||||
import 'dart:io';
|
import 'dart:io';
|
||||||
import 'package:flutter/services.dart';
|
import 'dart:typed_data';
|
||||||
|
import 'package:ffmpeg_kit_flutter_new_audio/ffmpeg_kit.dart';
|
||||||
|
import 'package:ffmpeg_kit_flutter_new_audio/ffmpeg_kit_config.dart';
|
||||||
|
import 'package:ffmpeg_kit_flutter_new_audio/return_code.dart';
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
import 'package:spotiflac_android/utils/logger.dart';
|
import 'package:spotiflac_android/utils/logger.dart';
|
||||||
|
|
||||||
final _log = AppLogger('FFmpeg');
|
final _log = AppLogger('FFmpeg');
|
||||||
|
|
||||||
/// FFmpeg service for audio conversion and remuxing
|
/// FFmpeg service for audio conversion and remuxing
|
||||||
/// Uses native MethodChannel to call FFmpegKit from local AAR
|
/// Uses ffmpeg_kit_flutter_new_audio plugin
|
||||||
class FFmpegService {
|
class FFmpegService {
|
||||||
static const _channel = MethodChannel('com.zarz.spotiflac/ffmpeg');
|
|
||||||
|
|
||||||
static Future<FFmpegResult> _execute(String command) async {
|
static Future<FFmpegResult> _execute(String command) async {
|
||||||
try {
|
try {
|
||||||
final result = await _channel.invokeMethod('execute', {'command': command});
|
final session = await FFmpegKit.execute(command);
|
||||||
final map = Map<String, dynamic>.from(result);
|
final returnCode = await session.getReturnCode();
|
||||||
|
final output = await session.getOutput() ?? '';
|
||||||
|
|
||||||
return FFmpegResult(
|
return FFmpegResult(
|
||||||
success: map['success'] as bool,
|
success: ReturnCode.isSuccess(returnCode),
|
||||||
returnCode: map['returnCode'] as int,
|
returnCode: returnCode?.getValue() ?? -1,
|
||||||
output: map['output'] as String,
|
output: output,
|
||||||
);
|
);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
_log.e('FFmpeg execute error: $e');
|
_log.e('FFmpeg execute error: $e');
|
||||||
@@ -69,6 +73,61 @@ class FFmpegService {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Future<String?> convertFlacToOpus(
|
||||||
|
String inputPath, {
|
||||||
|
String bitrate = '128k',
|
||||||
|
bool deleteOriginal = true,
|
||||||
|
}) async {
|
||||||
|
final outputPath = inputPath.replaceAll('.flac', '.opus');
|
||||||
|
|
||||||
|
// Opus in OGG container with VBR
|
||||||
|
final command =
|
||||||
|
'-i "$inputPath" -codec:a libopus -b:a $bitrate -vbr on -compression_level 10 -map 0:a -map_metadata 0 "$outputPath" -y';
|
||||||
|
|
||||||
|
final result = await _execute(command);
|
||||||
|
|
||||||
|
if (result.success) {
|
||||||
|
if (deleteOriginal) {
|
||||||
|
try {
|
||||||
|
await File(inputPath).delete();
|
||||||
|
} catch (_) {}
|
||||||
|
}
|
||||||
|
return outputPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
_log.e('FLAC to Opus conversion failed: ${result.output}');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Convert FLAC to lossy format based on format parameter
|
||||||
|
/// format: 'mp3' or 'opus'
|
||||||
|
/// bitrate: e.g., 'mp3_320', 'opus_128' - extracts the kbps value
|
||||||
|
static Future<String?> convertFlacToLossy(
|
||||||
|
String inputPath, {
|
||||||
|
required String format,
|
||||||
|
String? bitrate,
|
||||||
|
bool deleteOriginal = true,
|
||||||
|
}) async {
|
||||||
|
// Extract bitrate value from format like 'mp3_320' -> '320k'
|
||||||
|
String bitrateValue = '320k'; // default for mp3
|
||||||
|
if (bitrate != null && bitrate.contains('_')) {
|
||||||
|
final parts = bitrate.split('_');
|
||||||
|
if (parts.length == 2) {
|
||||||
|
bitrateValue = '${parts[1]}k';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (format.toLowerCase()) {
|
||||||
|
case 'opus':
|
||||||
|
final opusBitrate = bitrate?.startsWith('opus_') == true ? bitrateValue : '128k';
|
||||||
|
return convertFlacToOpus(inputPath, bitrate: opusBitrate, deleteOriginal: deleteOriginal);
|
||||||
|
case 'mp3':
|
||||||
|
default:
|
||||||
|
final mp3Bitrate = bitrate?.startsWith('mp3_') == true ? bitrateValue : '320k';
|
||||||
|
return convertFlacToMp3(inputPath, bitrate: mp3Bitrate, deleteOriginal: deleteOriginal);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static Future<String?> convertFlacToM4a(
|
static Future<String?> convertFlacToM4a(
|
||||||
String inputPath, {
|
String inputPath, {
|
||||||
String codec = 'aac',
|
String codec = 'aac',
|
||||||
@@ -104,8 +163,8 @@ class FFmpegService {
|
|||||||
|
|
||||||
static Future<bool> isAvailable() async {
|
static Future<bool> isAvailable() async {
|
||||||
try {
|
try {
|
||||||
final version = await _channel.invokeMethod('getVersion');
|
final version = await FFmpegKitConfig.getFFmpegVersion();
|
||||||
return version != null && version.toString().isNotEmpty;
|
return version?.isNotEmpty ?? false;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@@ -113,8 +172,7 @@ class FFmpegService {
|
|||||||
|
|
||||||
static Future<String?> getVersion() async {
|
static Future<String?> getVersion() async {
|
||||||
try {
|
try {
|
||||||
final version = await _channel.invokeMethod('getVersion');
|
return await FFmpegKitConfig.getFFmpegVersion();
|
||||||
return version as String?;
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -280,6 +338,210 @@ class FFmpegService {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Embed metadata to Opus file
|
||||||
|
/// Uses METADATA_BLOCK_PICTURE tag for cover art (OGG/Vorbis standard)
|
||||||
|
static Future<String?> embedMetadataToOpus({
|
||||||
|
required String opusPath,
|
||||||
|
String? coverPath,
|
||||||
|
Map<String, String>? metadata,
|
||||||
|
}) async {
|
||||||
|
final tempDir = await getTemporaryDirectory();
|
||||||
|
final uniqueId = DateTime.now().millisecondsSinceEpoch;
|
||||||
|
final tempOutput = '${tempDir.path}/temp_embed_$uniqueId.opus';
|
||||||
|
|
||||||
|
final StringBuffer cmdBuffer = StringBuffer();
|
||||||
|
cmdBuffer.write('-i "$opusPath" ');
|
||||||
|
cmdBuffer.write('-map 0:a ');
|
||||||
|
cmdBuffer.write('-c:a copy ');
|
||||||
|
|
||||||
|
// Embed metadata tags (Vorbis comments)
|
||||||
|
if (metadata != null) {
|
||||||
|
metadata.forEach((key, value) {
|
||||||
|
final sanitizedValue = value.replaceAll('"', '\\"');
|
||||||
|
cmdBuffer.write('-metadata $key="$sanitizedValue" ');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Embed cover art using METADATA_BLOCK_PICTURE
|
||||||
|
if (coverPath != null) {
|
||||||
|
try {
|
||||||
|
final pictureBlock = await _createMetadataBlockPicture(coverPath);
|
||||||
|
if (pictureBlock != null) {
|
||||||
|
// Escape special characters for shell
|
||||||
|
final escapedBlock = pictureBlock.replaceAll('"', '\\"');
|
||||||
|
cmdBuffer.write('-metadata METADATA_BLOCK_PICTURE="$escapedBlock" ');
|
||||||
|
_log.d('Created METADATA_BLOCK_PICTURE for Opus (${pictureBlock.length} chars)');
|
||||||
|
} else {
|
||||||
|
_log.w('Failed to create METADATA_BLOCK_PICTURE, skipping cover');
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
_log.e('Error creating METADATA_BLOCK_PICTURE: $e');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cmdBuffer.write('"$tempOutput" -y');
|
||||||
|
|
||||||
|
final command = cmdBuffer.toString();
|
||||||
|
_log.d('Executing FFmpeg Opus embed command');
|
||||||
|
|
||||||
|
final result = await _execute(command);
|
||||||
|
|
||||||
|
if (result.success) {
|
||||||
|
try {
|
||||||
|
final tempFile = File(tempOutput);
|
||||||
|
final originalFile = File(opusPath);
|
||||||
|
|
||||||
|
if (await tempFile.exists()) {
|
||||||
|
if (await originalFile.exists()) {
|
||||||
|
await originalFile.delete();
|
||||||
|
}
|
||||||
|
await tempFile.copy(opusPath);
|
||||||
|
await tempFile.delete();
|
||||||
|
|
||||||
|
_log.d('Opus metadata embedded successfully');
|
||||||
|
return opusPath;
|
||||||
|
} else {
|
||||||
|
_log.e('Temp Opus output file not found: $tempOutput');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
} catch (e) {
|
||||||
|
_log.e('Failed to replace Opus file after metadata embed: $e');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
final tempFile = File(tempOutput);
|
||||||
|
if (await tempFile.exists()) {
|
||||||
|
await tempFile.delete();
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
_log.w('Failed to cleanup temp Opus file: $e');
|
||||||
|
}
|
||||||
|
|
||||||
|
_log.e('Opus Metadata embed failed: ${result.output}');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create METADATA_BLOCK_PICTURE base64 string for OGG/Opus cover art
|
||||||
|
/// Format follows FLAC picture block specification:
|
||||||
|
/// - 4 bytes: picture type (3 = front cover)
|
||||||
|
/// - 4 bytes: MIME type length
|
||||||
|
/// - n bytes: MIME type string
|
||||||
|
/// - 4 bytes: description length
|
||||||
|
/// - n bytes: description string
|
||||||
|
/// - 4 bytes: width
|
||||||
|
/// - 4 bytes: height
|
||||||
|
/// - 4 bytes: color depth
|
||||||
|
/// - 4 bytes: colors used (0 for non-indexed)
|
||||||
|
/// - 4 bytes: picture data length
|
||||||
|
/// - n bytes: picture data
|
||||||
|
static Future<String?> _createMetadataBlockPicture(String imagePath) async {
|
||||||
|
try {
|
||||||
|
final file = File(imagePath);
|
||||||
|
if (!await file.exists()) {
|
||||||
|
_log.e('Cover image not found: $imagePath');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
final imageData = await file.readAsBytes();
|
||||||
|
|
||||||
|
// Detect MIME type from file extension or magic bytes
|
||||||
|
String mimeType;
|
||||||
|
if (imagePath.toLowerCase().endsWith('.png')) {
|
||||||
|
mimeType = 'image/png';
|
||||||
|
} else if (imagePath.toLowerCase().endsWith('.jpg') ||
|
||||||
|
imagePath.toLowerCase().endsWith('.jpeg')) {
|
||||||
|
mimeType = 'image/jpeg';
|
||||||
|
} else {
|
||||||
|
// Check magic bytes
|
||||||
|
if (imageData.length >= 8 &&
|
||||||
|
imageData[0] == 0x89 && imageData[1] == 0x50 &&
|
||||||
|
imageData[2] == 0x4E && imageData[3] == 0x47) {
|
||||||
|
mimeType = 'image/png';
|
||||||
|
} else if (imageData.length >= 2 &&
|
||||||
|
imageData[0] == 0xFF && imageData[1] == 0xD8) {
|
||||||
|
mimeType = 'image/jpeg';
|
||||||
|
} else {
|
||||||
|
mimeType = 'image/jpeg'; // Default to JPEG
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
final mimeBytes = utf8.encode(mimeType);
|
||||||
|
const description = ''; // Empty description
|
||||||
|
final descBytes = utf8.encode(description);
|
||||||
|
|
||||||
|
// Build the FLAC picture block
|
||||||
|
// Total size: 4 + 4 + mimeLen + 4 + descLen + 4 + 4 + 4 + 4 + 4 + imageLen
|
||||||
|
final blockSize = 4 + 4 + mimeBytes.length + 4 + descBytes.length +
|
||||||
|
4 + 4 + 4 + 4 + 4 + imageData.length;
|
||||||
|
|
||||||
|
final buffer = ByteData(blockSize);
|
||||||
|
var offset = 0;
|
||||||
|
|
||||||
|
// Picture type: 3 = Front cover
|
||||||
|
buffer.setUint32(offset, 3, Endian.big);
|
||||||
|
offset += 4;
|
||||||
|
|
||||||
|
// MIME type length
|
||||||
|
buffer.setUint32(offset, mimeBytes.length, Endian.big);
|
||||||
|
offset += 4;
|
||||||
|
|
||||||
|
// MIME type string
|
||||||
|
final blockBytes = Uint8List(blockSize);
|
||||||
|
blockBytes.setRange(0, offset, buffer.buffer.asUint8List());
|
||||||
|
blockBytes.setRange(offset, offset + mimeBytes.length, mimeBytes);
|
||||||
|
offset += mimeBytes.length;
|
||||||
|
|
||||||
|
// Description length
|
||||||
|
final tempBuffer = ByteData(4);
|
||||||
|
tempBuffer.setUint32(0, descBytes.length, Endian.big);
|
||||||
|
blockBytes.setRange(offset, offset + 4, tempBuffer.buffer.asUint8List());
|
||||||
|
offset += 4;
|
||||||
|
|
||||||
|
// Description string
|
||||||
|
blockBytes.setRange(offset, offset + descBytes.length, descBytes);
|
||||||
|
offset += descBytes.length;
|
||||||
|
|
||||||
|
// Width (0 = unknown)
|
||||||
|
tempBuffer.setUint32(0, 0, Endian.big);
|
||||||
|
blockBytes.setRange(offset, offset + 4, tempBuffer.buffer.asUint8List());
|
||||||
|
offset += 4;
|
||||||
|
|
||||||
|
// Height (0 = unknown)
|
||||||
|
tempBuffer.setUint32(0, 0, Endian.big);
|
||||||
|
blockBytes.setRange(offset, offset + 4, tempBuffer.buffer.asUint8List());
|
||||||
|
offset += 4;
|
||||||
|
|
||||||
|
// Color depth (0 = unknown)
|
||||||
|
tempBuffer.setUint32(0, 0, Endian.big);
|
||||||
|
blockBytes.setRange(offset, offset + 4, tempBuffer.buffer.asUint8List());
|
||||||
|
offset += 4;
|
||||||
|
|
||||||
|
// Colors used (0 for non-indexed)
|
||||||
|
tempBuffer.setUint32(0, 0, Endian.big);
|
||||||
|
blockBytes.setRange(offset, offset + 4, tempBuffer.buffer.asUint8List());
|
||||||
|
offset += 4;
|
||||||
|
|
||||||
|
// Picture data length
|
||||||
|
tempBuffer.setUint32(0, imageData.length, Endian.big);
|
||||||
|
blockBytes.setRange(offset, offset + 4, tempBuffer.buffer.asUint8List());
|
||||||
|
offset += 4;
|
||||||
|
|
||||||
|
// Picture data
|
||||||
|
blockBytes.setRange(offset, offset + imageData.length, imageData);
|
||||||
|
|
||||||
|
// Base64 encode the entire block
|
||||||
|
final base64String = base64Encode(blockBytes);
|
||||||
|
|
||||||
|
return base64String;
|
||||||
|
} catch (e) {
|
||||||
|
_log.e('Error creating METADATA_BLOCK_PICTURE: $e');
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static Map<String, String> _convertToId3Tags(Map<String, String> vorbisMetadata) {
|
static Map<String, String> _convertToId3Tags(Map<String, String> vorbisMetadata) {
|
||||||
final id3Map = <String, String>{};
|
final id3Map = <String, String>{};
|
||||||
|
|
||||||
|
|||||||
@@ -343,11 +343,12 @@ class PlatformBridge {
|
|||||||
await _channel.invokeMethod('clearTrackCache');
|
await _channel.invokeMethod('clearTrackCache');
|
||||||
}
|
}
|
||||||
|
|
||||||
static Future<Map<String, dynamic>> searchDeezerAll(String query, {int trackLimit = 15, int artistLimit = 3}) async {
|
static Future<Map<String, dynamic>> searchDeezerAll(String query, {int trackLimit = 15, int artistLimit = 2, String? filter}) async {
|
||||||
final result = await _channel.invokeMethod('searchDeezerAll', {
|
final result = await _channel.invokeMethod('searchDeezerAll', {
|
||||||
'query': query,
|
'query': query,
|
||||||
'track_limit': trackLimit,
|
'track_limit': trackLimit,
|
||||||
'artist_limit': artistLimit,
|
'artist_limit': artistLimit,
|
||||||
|
'filter': filter ?? '',
|
||||||
});
|
});
|
||||||
return jsonDecode(result as String) as Map<String, dynamic>;
|
return jsonDecode(result as String) as Map<String, dynamic>;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -49,11 +49,11 @@ const _builtInServices = [
|
|||||||
),
|
),
|
||||||
];
|
];
|
||||||
|
|
||||||
/// MP3 quality option (shown when enabled in settings)
|
/// Lossy quality option (shown when enabled in settings)
|
||||||
const _mp3QualityOption = QualityOption(
|
const _lossyQualityOption = QualityOption(
|
||||||
id: 'MP3',
|
id: 'LOSSY',
|
||||||
label: 'MP3',
|
label: 'Lossy',
|
||||||
description: '320kbps (converted from FLAC)',
|
description: 'MP3 320kbps or Opus 128kbps',
|
||||||
);
|
);
|
||||||
|
|
||||||
/// A reusable widget for selecting download service (built-in + extensions)
|
/// A reusable widget for selecting download service (built-in + extensions)
|
||||||
@@ -115,9 +115,9 @@ class _DownloadServicePickerState extends ConsumerState<DownloadServicePicker> {
|
|||||||
final settings = ref.read(settingsProvider);
|
final settings = ref.read(settingsProvider);
|
||||||
final builtIn = _builtInServices.where((s) => s.id == _selectedService).firstOrNull;
|
final builtIn = _builtInServices.where((s) => s.id == _selectedService).firstOrNull;
|
||||||
if (builtIn != null) {
|
if (builtIn != null) {
|
||||||
// Add MP3 option if enabled in settings
|
// Add Lossy option if enabled in settings
|
||||||
if (settings.enableMp3Option) {
|
if (settings.enableLossyOption) {
|
||||||
return [...builtIn.qualityOptions, _mp3QualityOption];
|
return [...builtIn.qualityOptions, _lossyQualityOption];
|
||||||
}
|
}
|
||||||
return builtIn.qualityOptions;
|
return builtIn.qualityOptions;
|
||||||
}
|
}
|
||||||
@@ -125,9 +125,9 @@ class _DownloadServicePickerState extends ConsumerState<DownloadServicePicker> {
|
|||||||
final extensionState = ref.read(extensionProvider);
|
final extensionState = ref.read(extensionProvider);
|
||||||
final ext = extensionState.extensions.where((e) => e.id == _selectedService).firstOrNull;
|
final ext = extensionState.extensions.where((e) => e.id == _selectedService).firstOrNull;
|
||||||
if (ext != null && ext.qualityOptions.isNotEmpty) {
|
if (ext != null && ext.qualityOptions.isNotEmpty) {
|
||||||
// Add MP3 option for extensions too if enabled
|
// Add Lossy option for extensions too if enabled
|
||||||
if (settings.enableMp3Option) {
|
if (settings.enableLossyOption) {
|
||||||
return [...ext.qualityOptions, _mp3QualityOption];
|
return [...ext.qualityOptions, _lossyQualityOption];
|
||||||
}
|
}
|
||||||
return ext.qualityOptions;
|
return ext.qualityOptions;
|
||||||
}
|
}
|
||||||
@@ -136,8 +136,8 @@ class _DownloadServicePickerState extends ConsumerState<DownloadServicePicker> {
|
|||||||
final defaultOptions = [
|
final defaultOptions = [
|
||||||
const QualityOption(id: 'DEFAULT', label: 'Default Quality', description: 'Best available'),
|
const QualityOption(id: 'DEFAULT', label: 'Default Quality', description: 'Best available'),
|
||||||
];
|
];
|
||||||
if (settings.enableMp3Option) {
|
if (settings.enableLossyOption) {
|
||||||
return [...defaultOptions, _mp3QualityOption];
|
return [...defaultOptions, _lossyQualityOption];
|
||||||
}
|
}
|
||||||
return defaultOptions;
|
return defaultOptions;
|
||||||
}
|
}
|
||||||
@@ -259,6 +259,7 @@ class _DownloadServicePickerState extends ConsumerState<DownloadServicePicker> {
|
|||||||
return Icons.music_note;
|
return Icons.music_note;
|
||||||
case 'MP3_320':
|
case 'MP3_320':
|
||||||
case 'MP3':
|
case 'MP3':
|
||||||
|
case 'LOSSY':
|
||||||
return Icons.audiotrack;
|
return Icons.audiotrack;
|
||||||
case 'OPUS':
|
case 'OPUS':
|
||||||
case 'OPUS_128':
|
case 'OPUS_128':
|
||||||
|
|||||||
@@ -297,6 +297,22 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "2.1.4"
|
version: "2.1.4"
|
||||||
|
ffmpeg_kit_flutter_new_audio:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: ffmpeg_kit_flutter_new_audio
|
||||||
|
sha256: "0a698b46cd163c8e9917af75325c84d27871a2a8b2c37de3b40486cd0ab662ae"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "2.0.0"
|
||||||
|
ffmpeg_kit_flutter_platform_interface:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: ffmpeg_kit_flutter_platform_interface
|
||||||
|
sha256: addf046ae44e190ad0101b2fde2ad909a3cd08a2a109f6106d2f7048b7abedee
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "0.2.1"
|
||||||
file:
|
file:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
|||||||
+3
-3
@@ -1,7 +1,7 @@
|
|||||||
name: spotiflac_android
|
name: spotiflac_android
|
||||||
description: Download Spotify tracks in FLAC from Tidal, Qobuz & Amazon Music
|
description: Download Spotify tracks in FLAC from Tidal, Qobuz & Amazon Music
|
||||||
publish_to: "none"
|
publish_to: "none"
|
||||||
version: 3.2.1+64
|
version: 3.3.1+68
|
||||||
|
|
||||||
environment:
|
environment:
|
||||||
sdk: ^3.10.0
|
sdk: ^3.10.0
|
||||||
@@ -59,8 +59,8 @@ dependencies:
|
|||||||
receive_sharing_intent: ^1.8.1
|
receive_sharing_intent: ^1.8.1
|
||||||
logger: ^2.5.0
|
logger: ^2.5.0
|
||||||
|
|
||||||
# FFmpeg - using local custom AAR (arm64-v8a + armeabi-v7a only)
|
# FFmpeg for audio conversion
|
||||||
# ffmpeg_kit_flutter_new_audio: ^2.0.0 # Replaced with local AAR
|
ffmpeg_kit_flutter_new_audio: ^2.0.0
|
||||||
open_filex: ^4.7.0
|
open_filex: ^4.7.0
|
||||||
|
|
||||||
# Notifications
|
# Notifications
|
||||||
|
|||||||
@@ -1,92 +0,0 @@
|
|||||||
name: spotiflac_android
|
|
||||||
description: Download Spotify tracks in FLAC from Tidal, Qobuz & Amazon Music
|
|
||||||
publish_to: "none"
|
|
||||||
version: 3.2.1+64
|
|
||||||
|
|
||||||
environment:
|
|
||||||
sdk: ^3.10.0
|
|
||||||
|
|
||||||
dependencies:
|
|
||||||
flutter:
|
|
||||||
sdk: flutter
|
|
||||||
|
|
||||||
# Localization
|
|
||||||
flutter_localizations:
|
|
||||||
sdk: flutter
|
|
||||||
intl: any
|
|
||||||
|
|
||||||
# State Management
|
|
||||||
flutter_riverpod: ^3.1.0
|
|
||||||
riverpod_annotation: ^4.0.0
|
|
||||||
|
|
||||||
# Navigation
|
|
||||||
go_router: ^17.0.1
|
|
||||||
|
|
||||||
# Storage & Persistence
|
|
||||||
shared_preferences: ^2.5.3
|
|
||||||
path_provider: ^2.1.5
|
|
||||||
path: ^1.9.0
|
|
||||||
sqflite: ^2.4.1
|
|
||||||
|
|
||||||
# HTTP & Network
|
|
||||||
http: ^1.6.0
|
|
||||||
dio: ^5.8.0
|
|
||||||
|
|
||||||
# UI Components
|
|
||||||
cupertino_icons: ^1.0.8
|
|
||||||
cached_network_image: ^3.4.1
|
|
||||||
flutter_cache_manager: ^3.4.1
|
|
||||||
flutter_svg: ^2.1.0
|
|
||||||
|
|
||||||
# Material Expressive 3 / Dynamic Color
|
|
||||||
dynamic_color: ^1.7.0
|
|
||||||
material_color_utilities: ^0.11.1
|
|
||||||
palette_generator: ^0.3.3+4
|
|
||||||
|
|
||||||
# Permissions
|
|
||||||
permission_handler: ^12.0.1
|
|
||||||
|
|
||||||
# File Picker
|
|
||||||
file_picker: ^10.3.8
|
|
||||||
|
|
||||||
# JSON Serialization
|
|
||||||
json_annotation: ^4.9.0
|
|
||||||
|
|
||||||
# Utils
|
|
||||||
url_launcher: ^6.3.1
|
|
||||||
device_info_plus: ^12.3.0
|
|
||||||
share_plus: ^12.0.1
|
|
||||||
receive_sharing_intent: ^1.8.1
|
|
||||||
logger: ^2.5.0
|
|
||||||
|
|
||||||
# FFmpeg for iOS (uses plugin, Android uses custom AAR)
|
|
||||||
ffmpeg_kit_flutter_new_audio: ^2.0.0
|
|
||||||
open_filex: ^4.7.0
|
|
||||||
|
|
||||||
# Notifications
|
|
||||||
flutter_local_notifications: ^19.0.0
|
|
||||||
|
|
||||||
dev_dependencies:
|
|
||||||
flutter_test:
|
|
||||||
sdk: flutter
|
|
||||||
flutter_lints: ^6.0.0
|
|
||||||
build_runner: ^2.10.4
|
|
||||||
riverpod_generator: ^4.0.0
|
|
||||||
json_serializable: ^6.11.2
|
|
||||||
flutter_launcher_icons: ^0.14.3
|
|
||||||
|
|
||||||
flutter_launcher_icons:
|
|
||||||
android: true
|
|
||||||
ios: true
|
|
||||||
image_path: "icon.png"
|
|
||||||
adaptive_icon_background: "#1a1a2e"
|
|
||||||
adaptive_icon_foreground: "icon.png"
|
|
||||||
ios_content_mode: scaleAspectFill
|
|
||||||
remove_alpha_ios: true
|
|
||||||
|
|
||||||
flutter:
|
|
||||||
uses-material-design: true
|
|
||||||
generate: true
|
|
||||||
|
|
||||||
assets:
|
|
||||||
- assets/images/
|
|
||||||
Reference in New Issue
Block a user